| | |
| | | this.env[p] = value; |
| | | }; |
| | | |
| | | |
| | | // add a localized label to the client environment |
| | | this.add_label = function(key, value) |
| | | { |
| | | this.labels[key] = value; |
| | | }; |
| | | |
| | | |
| | | // add a button to the button list |
| | | this.register_button = function(command, id, type, act, sel, over) |
| | |
| | | // enable general commands |
| | | this.enable_command('logout', 'mail', 'addressbook', 'settings', true); |
| | | |
| | | if (this.env.permaurl) |
| | | this.enable_command('permaurl', true); |
| | | |
| | | switch (this.task) |
| | | { |
| | | case 'mail': |
| | |
| | | this.message_list.addEventListener('keypress', function(o){ p.msglist_keypress(o); }); |
| | | this.message_list.addEventListener('select', function(o){ p.msglist_select(o); }); |
| | | this.message_list.addEventListener('dragstart', function(o){ p.drag_active = true; if (p.preview_timer) clearTimeout(p.preview_timer); }); |
| | | this.message_list.addEventListener('dragmove', function(o, e){ p.drag_move(e); }); |
| | | this.message_list.addEventListener('dragend', function(o){ p.drag_active = false; }); |
| | | |
| | | this.message_list.init(); |
| | |
| | | this.set_message_coltypes(this.env.coltypes); |
| | | |
| | | // enable mail commands |
| | | this.enable_command('list', 'checkmail', 'compose', 'add-contact', 'search', 'reset-search', true); |
| | | this.enable_command('list', 'checkmail', 'compose', 'add-contact', 'search', 'reset-search', 'collapse-folder', true); |
| | | |
| | | if (this.env.search_text != null && document.getElementById('quicksearchbox') != null) |
| | | document.getElementById('quicksearchbox').value = this.env.search_text; |
| | | |
| | | if (this.env.action=='show' || this.env.action=='preview') |
| | | { |
| | | this.enable_command('show', 'reply', 'reply-all', 'forward', 'moveto', 'delete', 'mark', 'viewsource', 'print', 'load-attachment', true); |
| | | this.enable_command('show', 'reply', 'reply-all', 'forward', 'moveto', 'delete', 'mark', 'viewsource', 'print', 'load-attachment', 'load-headers', true); |
| | | if (this.env.next_uid) |
| | | { |
| | | this.enable_command('nextmessage', true); |
| | |
| | | if (this.env.messagecount) |
| | | this.enable_command('select-all', 'select-none', 'expunge', true); |
| | | |
| | | if (this.env.messagecount |
| | | && (this.env.mailbox == this.env.trash_mailbox || this.env.mailbox == this.env.junk_mailbox |
| | | || this.env.mailbox.match('^' + RegExp.escape(this.env.trash_mailbox) + RegExp.escape(this.env.delimiter)) |
| | | || this.env.mailbox.match('^' + RegExp.escape(this.env.junk_mailbox) + RegExp.escape(this.env.delimiter)))) |
| | | if (this.purge_mailbox_test()) |
| | | this.enable_command('purge', true); |
| | | |
| | | this.set_page_buttons(); |
| | | |
| | | // focus main window |
| | | if (this.env.framed && window.parent) |
| | | window.parent.focus(); |
| | | else |
| | | window.focus(); |
| | | |
| | | // init message compose form |
| | | if (this.env.action=='compose') |
| | |
| | | // get unread count for each mailbox |
| | | if (this.gui_objects.mailboxlist) |
| | | { |
| | | this.env.unread_counts = {}; |
| | | this.gui_objects.folderlist = this.gui_objects.mailboxlist; |
| | | this.http_request('getunread', ''); |
| | | } |
| | |
| | | this.contact_list.addEventListener('keypress', function(o){ p.contactlist_keypress(o); }); |
| | | this.contact_list.addEventListener('select', function(o){ p.contactlist_select(o); }); |
| | | this.contact_list.addEventListener('dragstart', function(o){ p.drag_active = true; }); |
| | | this.contact_list.addEventListener('dragmove', function(o, e){ p.drag_move(e); }); |
| | | this.contact_list.addEventListener('dragend', function(o){ p.drag_active = false; }); |
| | | this.contact_list.init(); |
| | | |
| | |
| | | if ((this.env.action=='add' || this.env.action=='edit') && this.gui_objects.editform) |
| | | this.enable_command('save', true); |
| | | else |
| | | this.enable_command('search', 'reset-search', 'moveto', true); |
| | | this.enable_command('search', 'reset-search', 'moveto', 'import', true); |
| | | |
| | | if (this.contact_list && this.contact_list.rowcount > 0) |
| | | this.enable_command('export', true); |
| | | |
| | | this.enable_command('list', true); |
| | | break; |
| | |
| | | case 'settings': |
| | | this.enable_command('preferences', 'identities', 'save', 'folders', true); |
| | | |
| | | if (this.env.action=='identities' || this.env.action=='edit-identity' || this.env.action=='add-identity') |
| | | this.enable_command('edit', 'add', 'delete', true); |
| | | if (this.env.action=='identities' || this.env.action=='edit-identity' || this.env.action=='add-identity') { |
| | | this.enable_command('add', 'delete', this.env.multiple_identities); |
| | | this.enable_command('edit', true); |
| | | } |
| | | |
| | | if (this.env.action=='edit-identity' || this.env.action=='add-identity') |
| | | this.enable_command('save', true); |
| | |
| | | case 'login': |
| | | var input_user = rcube_find_object('rcmloginuser'); |
| | | var input_pass = rcube_find_object('rcmloginpwd'); |
| | | var input_tz = rcube_find_object('rcmlogintz'); |
| | | |
| | | if (input_user) |
| | | input_user.onkeyup = function(e){ return rcmail.login_user_keyup(e); }; |
| | | if (input_user && input_user.value=='') |
| | | input_user.focus(); |
| | | else if (input_pass) |
| | | input_pass.focus(); |
| | | |
| | | |
| | | // detect client timezone |
| | | if (input_tz) |
| | | input_tz.value = new Date().getTimezoneOffset() / -60; |
| | | |
| | | this.enable_command('login', true); |
| | | break; |
| | | |
| | | default: |
| | | break; |
| | | } |
| | | |
| | | |
| | | // enable basic commands |
| | | this.enable_command('logout', true); |
| | |
| | | // start keep-alive interval |
| | | this.start_keepalive(); |
| | | |
| | | |
| | | // execute all foreign onload scripts |
| | | for (var i=0; i<this.onloads.length; i++) |
| | | { |
| | |
| | | } |
| | | }; |
| | | |
| | | |
| | | // start interval for keep-alive/recent_check signal |
| | | this.start_keepalive = function() |
| | | { |
| | |
| | | else if (this.env.keep_alive && !this.env.framed && this.task!='login') |
| | | this._int = setInterval(function(){ ref.send_keep_alive(); }, this.env.keep_alive * 1000); |
| | | } |
| | | |
| | | |
| | | this.init_message_row = function(row) |
| | | { |
| | |
| | | row.unread = this.env.messages[uid].unread ? true : false; |
| | | row.replied = this.env.messages[uid].replied ? true : false; |
| | | row.flagged = this.env.messages[uid].flagged ? true : false; |
| | | row.forwarded = this.env.messages[uid].forwarded ? true : false; |
| | | } |
| | | |
| | | // set eventhandler to message icon |
| | |
| | | row.flagged_icon.onmousedown = function(e) { p.command('toggle_flag', this); }; |
| | | } |
| | | }; |
| | | |
| | | |
| | | // init message compose form: set focus and eventhandlers |
| | | this.init_messageform = function() |
| | |
| | | }; |
| | | |
| | | |
| | | |
| | | /*********************************************************/ |
| | | /********* client command interface *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | // execute a specific command on the web client |
| | | this.command = function(command, props, obj) |
| | |
| | | return false; |
| | | } |
| | | |
| | | |
| | | // check input before leaving compose step |
| | | if (this.task=='mail' && this.env.action=='compose' && (command=='list' || command=='mail' || command=='addressbook' || command=='settings')) |
| | | { |
| | | if (this.cmp_hash != this.compose_field_hash() && !confirm(this.get_label('notsentwarning'))) |
| | | return false; |
| | | } |
| | | |
| | | |
| | | // process command |
| | | switch (command) |
| | |
| | | this.switch_task(command); |
| | | break; |
| | | |
| | | case 'permaurl': |
| | | if (obj && obj.href && obj.target) |
| | | return true; |
| | | else if (this.env.permaurl) |
| | | parent.location.href = this.env.permaurl; |
| | | break; |
| | | |
| | | // misc list commands |
| | | case 'list': |
| | |
| | | this.list_contacts(props); |
| | | this.enable_command('add', (this.env.address_sources && !this.env.address_sources[props].readonly)); |
| | | } |
| | | break; |
| | | |
| | | |
| | | case 'load-headers': |
| | | this.load_headers(obj); |
| | | break; |
| | | |
| | | |
| | |
| | | if (props._row.uid) |
| | | { |
| | | uid = props._row.uid; |
| | | this.message_list.dont_select = true; |
| | | // toggle flagged/unflagged |
| | | if (this.message_list.rows[uid].flagged) |
| | | flag = 'unflagged'; |
| | |
| | | case 'compose': |
| | | var url = this.env.comm_path+'&_action=compose'; |
| | | |
| | | if (this.task=='mail' && this.env.mailbox==this.env.drafts_mailbox) |
| | | if (this.task=='mail') |
| | | { |
| | | var uid; |
| | | if (uid = this.get_single_uid()) |
| | | url += '&_draft_uid='+uid+'&_mbox='+urlencode(this.env.mailbox); |
| | | } |
| | | url += '&_mbox='+urlencode(this.env.mailbox); |
| | | |
| | | if (this.env.mailbox==this.env.drafts_mailbox) |
| | | { |
| | | var uid; |
| | | if (uid = this.get_single_uid()) |
| | | url += '&_draft_uid='+uid; |
| | | } |
| | | } |
| | | // modify url if we're in addressbook |
| | | else if (this.task=='addressbook') |
| | | { |
| | | // switch to mail compose step directly |
| | | if (props && props.indexOf('@') > 0) |
| | | { |
| | | { |
| | | url = this.get_task_url('mail', url); |
| | | this.redirect(url + '&_to='+urlencode(props)); |
| | | break; |
| | | } |
| | | } |
| | | |
| | | // use contact_id passed as command parameter |
| | | var a_cids = new Array(); |
| | |
| | | break; |
| | | } |
| | | |
| | | // reset quicksearch |
| | | // reset quicksearch |
| | | case 'reset-search': |
| | | var s = this.env.search_request; |
| | | this.reset_qsearch(); |
| | |
| | | this.list_contacts(this.env.source); |
| | | break; |
| | | |
| | | case 'import': |
| | | if (this.env.action == 'import' && this.gui_objects.importform) { |
| | | var file = document.getElementById('rcmimportfile'); |
| | | if (file && !file.value) { |
| | | alert(this.get_label('selectimportfile')); |
| | | break; |
| | | } |
| | | this.gui_objects.importform.submit(); |
| | | this.set_busy(true, 'importwait'); |
| | | this.lock_form(this.gui_objects.importform, true); |
| | | } |
| | | else |
| | | this.goto_url('import'); |
| | | break; |
| | | |
| | | case 'export': |
| | | if (this.contact_list.rowcount > 0) { |
| | | var add_url = (this.env.source ? '_source='+urlencode(this.env.source)+'&' : ''); |
| | | if (this.env.search_request) |
| | | add_url += '_search='+this.env.search_request; |
| | | |
| | | this.goto_url('export', add_url); |
| | | } |
| | | break; |
| | | |
| | | // collapse/expand folder |
| | | case 'collapse-folder': |
| | | if (props) |
| | | this.collapse_folder(props); |
| | | break; |
| | | |
| | | // user settings commands |
| | | case 'preferences': |
| | |
| | | return obj ? false : true; |
| | | }; |
| | | |
| | | |
| | | // set command enabled or disabled |
| | | this.enable_command = function() |
| | | { |
| | |
| | | } |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | // lock/unlock interface |
| | | this.set_busy = function(a, message) |
| | |
| | | this.request_timer = window.setTimeout(function(){ ref.request_timed_out(); }, this.env.request_timeout * 1000); |
| | | }; |
| | | |
| | | |
| | | // return a localized string |
| | | this.get_label = function(name) |
| | | { |
| | |
| | | else |
| | | return name; |
| | | }; |
| | | |
| | | |
| | | // switch to another application task |
| | | this.switch_task = function(task) |
| | |
| | | this.redirect(url); |
| | | }; |
| | | |
| | | |
| | | this.get_task_url = function(task, url) |
| | | { |
| | | if (!url) |
| | |
| | | return url.replace(/_task=[a-z]+/, '_task='+task); |
| | | }; |
| | | |
| | | |
| | | // called when a request timed out |
| | | this.request_timed_out = function() |
| | | { |
| | |
| | | /********* event handling methods *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | this.doc_mouse_up = function(e) |
| | | { |
| | | if (this.message_list) |
| | | { |
| | | var model, li; |
| | | |
| | | if (this.message_list) { |
| | | this.message_list.blur(); |
| | | else if (this.contact_list) |
| | | model = this.env.mailboxes; |
| | | } |
| | | else if (this.contact_list) { |
| | | this.contact_list.blur(); |
| | | }; |
| | | |
| | | this.focus_folder = function(id) |
| | | { |
| | | var li; |
| | | if (this.drag_active && this.check_droptarget(id) && (li = this.get_folder_li(id))) |
| | | this.set_classname(li, 'droptarget', true); |
| | | model = this.env.address_sources; |
| | | } |
| | | |
| | | this.unfocus_folder = function(id) |
| | | { |
| | | var li; |
| | | if (this.drag_active && (li = this.get_folder_li(id))) |
| | | this.set_classname(li, 'droptarget', false); |
| | | } |
| | | |
| | | // onmouseup handler for folder list item |
| | | this.folder_mouse_up = function(id) |
| | | { |
| | | if (this.drag_active) |
| | | { |
| | | this.unfocus_folder(id); |
| | | this.command('moveto', id); |
| | | |
| | | // handle mouse release when dragging |
| | | if (this.drag_active && model) { |
| | | for (var k in model) { |
| | | if ((li = this.get_folder_li(k)) && rcube_mouse_is_over(e, li.firstChild) && this.check_droptarget(k)) { |
| | | this.set_classname(li, 'droptarget', false); |
| | | this.command('moveto', model[k].id); |
| | | break; |
| | | } |
| | | } |
| | | }; |
| | | } |
| | | }; |
| | | |
| | | this.drag_move = function(e) |
| | | { |
| | | var li; |
| | | var model = this.task == 'mail' ? this.env.mailboxes : this.env.address_sources; |
| | | |
| | | if (this.gui_objects.folderlist && model) { |
| | | for (var k in model) { |
| | | if (li = this.get_folder_li(k)) |
| | | this.set_classname(li, 'droptarget', (rcube_mouse_is_over(e, li.firstChild) && this.check_droptarget(k))); |
| | | } |
| | | } |
| | | }; |
| | | |
| | | this.collapse_folder = function(id) |
| | | { |
| | | var div; |
| | | if ((li = this.get_folder_li(id)) && |
| | | (div = li.getElementsByTagName("div")[0]) && |
| | | (div.className.match(/collapsed/) || div.className.match(/expanded/))) |
| | | { |
| | | var ul = li.getElementsByTagName("ul")[0]; |
| | | if (div.className.match(/collapsed/)) |
| | | { |
| | | ul.style.display = ''; |
| | | this.set_classname(div, 'collapsed', false); |
| | | this.set_classname(div, 'expanded', true); |
| | | var reg = new RegExp('&'+escape(id)+'&'); |
| | | this.set_env('collapsed_folders', this.env.collapsed_folders.replace(reg, '')); |
| | | } |
| | | else |
| | | { |
| | | ul.style.display = 'none'; |
| | | this.set_classname(div, 'expanded', false); |
| | | this.set_classname(div, 'collapsed', true); |
| | | this.set_env('collapsed_folders', this.env.collapsed_folders+'&'+escape(id)+'&'); |
| | | |
| | | // select parent folder if one of its childs is currently selected |
| | | if (this.env.mailbox.indexOf(id + this.env.delimiter) == 0) |
| | | this.command('list', id); |
| | | } |
| | | |
| | | // Work around a bug in IE6 and IE7, see #1485309 |
| | | if ((bw.ie6 || bw.ie7) && |
| | | li.nextSibling && |
| | | (li.nextSibling.getElementsByTagName("ul").length>0) && |
| | | li.nextSibling.getElementsByTagName("ul")[0].style && |
| | | (li.nextSibling.getElementsByTagName("ul")[0].style.display!='none')) |
| | | { |
| | | li.nextSibling.getElementsByTagName("ul")[0].style.display = 'none'; |
| | | li.nextSibling.getElementsByTagName("ul")[0].style.display = ''; |
| | | } |
| | | |
| | | this.http_post('save-pref', '_name=collapsed_folders&_value='+escape(this.env.collapsed_folders)); |
| | | this.set_unread_count_display(id, false); |
| | | } |
| | | } |
| | | |
| | | this.click_on_list = function(e) |
| | | { |
| | |
| | | if (mbox_li = this.get_folder_li()) |
| | | this.set_classname(mbox_li, 'unfocused', true); |
| | | |
| | | rcube_event.cancel(e); |
| | | return rcube_event.get_button(e) == 2 ? true : rcube_event.cancel(e); |
| | | }; |
| | | |
| | | |
| | | this.msglist_select = function(list) |
| | | { |
| | |
| | | |
| | | // start timer for message preview (wait for double click) |
| | | if (selected && this.env.contentframe && !list.multi_selecting) |
| | | this.preview_timer = window.setTimeout(function(){ ref.msglist_get_preview(); }, this.dblclick_time + 10); |
| | | this.preview_timer = window.setTimeout(function(){ ref.msglist_get_preview(); }, 200); |
| | | else if (this.env.contentframe) |
| | | this.show_contentframe(false); |
| | | }; |
| | | |
| | | |
| | | this.msglist_dbl_click = function(list) |
| | | { |
| | |
| | | this.show_message(uid, false, false); |
| | | }; |
| | | |
| | | |
| | | this.msglist_keypress = function(list) |
| | | { |
| | | if (list.key_pressed == list.ENTER_KEY) |
| | | this.command('show'); |
| | | else if (list.key_pressed == list.DELETE_KEY) |
| | | this.command('delete'); |
| | | else if (list.key_pressed == list.BACKSPACE_KEY) |
| | | this.command('delete'); |
| | | else |
| | | list.shiftkey = false; |
| | | }; |
| | | |
| | | |
| | | this.msglist_get_preview = function() |
| | | { |
| | |
| | | this.show_contentframe(false); |
| | | }; |
| | | |
| | | |
| | | this.check_droptarget = function(id) |
| | | { |
| | | if (this.task == 'mail') |
| | | return (id != this.env.mailbox); |
| | | return (this.env.mailboxes[id] && this.env.mailboxes[id].id != this.env.mailbox && !this.env.mailboxes[id].virtual); |
| | | else if (this.task == 'addressbook') |
| | | return (id != this.env.source && this.env.address_sources[id] && !this.env.address_sources[id].readonly); |
| | | else if (this.task == 'settings') |
| | |
| | | /********* (message) list functionality *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | // when user doble-clicks on a row |
| | | this.show_message = function(id, safe, preview) |
| | | { |
| | | if (!id) return; |
| | | |
| | | var add_url = ''; |
| | | var action = preview ? 'preview': 'show'; |
| | | var target = window; |
| | | |
| | | if (preview && this.env.contentframe && window.frames && window.frames[this.env.contentframe]) |
| | | { |
| | | target = window.frames[this.env.contentframe]; |
| | |
| | | // also send search request to get the right messages |
| | | if (this.env.search_request) |
| | | add_url += '&_search='+this.env.search_request; |
| | | |
| | | if (id) |
| | | var url = '&_action='+action+'&_uid='+id+'&_mbox='+urlencode(this.env.mailbox)+add_url; |
| | | if (action == 'preview' && String(target.location.href).indexOf(url) >= 0) |
| | | this.show_contentframe(true); |
| | | else |
| | | { |
| | | var url = '&_action='+action+'&_uid='+id+'&_mbox='+urlencode(this.env.mailbox)+add_url; |
| | | if (action == 'preview' && String(target.location.href).indexOf(url) >= 0) |
| | | this.show_contentframe(true); |
| | | else |
| | | this.set_busy(true, 'loading'); |
| | | target.location.href = this.env.comm_path+url; |
| | | // mark as read and change mbox unread counter |
| | | if (action == 'preview' && this.message_list && this.message_list.rows[id] && this.message_list.rows[id].unread) |
| | | { |
| | | this.set_busy(true, 'loading'); |
| | | target.location.href = this.env.comm_path+url; |
| | | } |
| | | this.set_message(id, 'unread', false); |
| | | if (this.env.unread_counts[this.env.mailbox]) |
| | | { |
| | | this.env.unread_counts[this.env.mailbox] -= 1; |
| | | this.set_unread_count(this.env.mailbox, this.env.unread_counts[this.env.mailbox], this.env.mailbox == 'INBOX'); |
| | | } |
| | | } |
| | | } |
| | | }; |
| | | |
| | | |
| | | this.show_contentframe = function(show) |
| | | { |
| | |
| | | if (!show && this.busy) |
| | | this.set_busy(false); |
| | | }; |
| | | |
| | | |
| | | // list a specific page |
| | | this.list_page = function(page) |
| | |
| | | this.list_contacts(this.env.source, page); |
| | | } |
| | | }; |
| | | |
| | | |
| | | // list messages of a specific mailbox |
| | | this.list_mailbox = function(mbox, page, sort) |
| | |
| | | } |
| | | }; |
| | | |
| | | |
| | | // send remote request to load message list |
| | | this.list_mailbox_remote = function(mbox, page, add_url) |
| | | { |
| | |
| | | this.set_busy(true, 'loading'); |
| | | this.http_request('list', url+add_url, true); |
| | | }; |
| | | |
| | | |
| | | this.expunge_mailbox = function(mbox) |
| | | { |
| | |
| | | var url = '_mbox='+urlencode(mbox); |
| | | this.http_post('expunge', url+add_url, lock); |
| | | }; |
| | | |
| | | |
| | | this.purge_mailbox = function(mbox) |
| | | { |
| | |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | // test if purge command is allowed |
| | | this.purge_mailbox_test = function() |
| | | { |
| | | return (this.env.messagecount && (this.env.mailbox == this.env.trash_mailbox || this.env.mailbox == this.env.junk_mailbox |
| | | || this.env.mailbox.match('^' + RegExp.escape(this.env.trash_mailbox) + RegExp.escape(this.env.delimiter)) |
| | | || this.env.mailbox.match('^' + RegExp.escape(this.env.junk_mailbox) + RegExp.escape(this.env.delimiter)))); |
| | | }; |
| | | |
| | | // set message icon |
| | | this.set_message_icon = function(uid) |
| | | { |
| | | var icn_src; |
| | | var rows = this.message_list.rows; |
| | | |
| | | if (!rows[uid]) |
| | | return false; |
| | | |
| | | if (rows[uid].deleted && this.env.deletedicon) |
| | | icn_src = this.env.deletedicon; |
| | | else if (rows[uid].replied && this.env.repliedicon) |
| | | { |
| | | if (rows[uid].forwarded && this.env.forwardedrepliedicon) |
| | | icn_src = this.env.forwardedrepliedicon; |
| | | else |
| | | icn_src = this.env.repliedicon; |
| | | } |
| | | else if (rows[uid].forwarded && this.env.forwardedicon) |
| | | icn_src = this.env.forwardedicon; |
| | | else if (rows[uid].unread && this.env.unreadicon) |
| | | icn_src = this.env.unreadicon; |
| | | else if (this.env.messageicon) |
| | | icn_src = this.env.messageicon; |
| | | |
| | | if (icn_src && rows[uid].icon) |
| | | rows[uid].icon.src = icn_src; |
| | | |
| | | icn_src = ''; |
| | | |
| | | if (rows[uid].flagged && this.env.flaggedicon) |
| | | icn_src = this.env.flaggedicon; |
| | | else if (this.env.unflaggedicon) |
| | | icn_src = this.env.unflaggedicon; |
| | | |
| | | if (rows[uid].flagged_icon && icn_src) |
| | | rows[uid].flagged_icon.src = icn_src; |
| | | } |
| | | |
| | | // set message status |
| | | this.set_message_status = function(uid, flag, status) |
| | | { |
| | | var rows = this.message_list.rows; |
| | | |
| | | if (!rows[uid]) return false; |
| | | |
| | | if (flag == 'unread') |
| | | rows[uid].unread = status; |
| | | else if(flag == 'deleted') |
| | | rows[uid].deleted = status; |
| | | else if (flag == 'replied') |
| | | rows[uid].replied = status; |
| | | else if (flag == 'forwarded') |
| | | rows[uid].forwarded = status; |
| | | else if (flag == 'flagged') |
| | | rows[uid].flagged = status; |
| | | } |
| | | |
| | | // set message row status, class and icon |
| | | this.set_message = function(uid, flag, status) |
| | | { |
| | | var rows = this.message_list.rows; |
| | | |
| | | if (!rows[uid]) return false; |
| | | |
| | | if (flag) |
| | | this.set_message_status(uid, flag, status); |
| | | |
| | | if (rows[uid].unread && rows[uid].classname.indexOf('unread')<0) |
| | | { |
| | | rows[uid].classname += ' unread'; |
| | | this.set_classname(rows[uid].obj, 'unread', true); |
| | | } |
| | | else if (!rows[uid].unread && rows[uid].classname.indexOf('unread')>=0) |
| | | { |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*unread/, ''); |
| | | this.set_classname(rows[uid].obj, 'unread', false); |
| | | } |
| | | |
| | | if (rows[uid].deleted && rows[uid].classname.indexOf('deleted')<0) |
| | | { |
| | | rows[uid].classname += ' deleted'; |
| | | this.set_classname(rows[uid].obj, 'deleted', true); |
| | | } |
| | | else if (!rows[uid].deleted && rows[uid].classname.indexOf('deleted')>=0) |
| | | { |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*deleted/, ''); |
| | | this.set_classname(rows[uid].obj, 'deleted', false); |
| | | } |
| | | |
| | | if (rows[uid].flagged && rows[uid].classname.indexOf('flagged')<0) |
| | | { |
| | | rows[uid].classname += ' flagged'; |
| | | this.set_classname(rows[uid].obj, 'flagged', true); |
| | | } |
| | | else if (!rows[uid].flagged && rows[uid].classname.indexOf('flagged')>=0) |
| | | { |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*flagged/, ''); |
| | | this.set_classname(rows[uid].obj, 'flagged', false); |
| | | } |
| | | |
| | | this.set_message_icon(uid); |
| | | } |
| | | |
| | | // move selected messages to the specified mailbox |
| | | this.move_messages = function(mbox) |
| | | { |
| | |
| | | this.permanently_remove_messages(); |
| | | }; |
| | | |
| | | |
| | | // delete the selected messages permanently |
| | | this.permanently_remove_messages = function() |
| | | { |
| | |
| | | else |
| | | { |
| | | rows[id].deleted = true; |
| | | |
| | | if (rows[id].classname.indexOf('deleted')<0) |
| | | { |
| | | rows[id].classname += ' deleted'; |
| | | this.set_classname(rows[id].obj, 'deleted', true); |
| | | } |
| | | if (this.env.read_when_deleted) |
| | | { |
| | | rows[id].classname = rows[id].classname.replace(/\s*unread/, ''); |
| | | this.set_classname(rows[id].obj, 'unread', false); |
| | | } |
| | | |
| | | if (rows[id].icon && this.env.deletedicon) |
| | | rows[id].icon.src = this.env.deletedicon; |
| | | rows[id].unread = false; |
| | | this.set_message(id); |
| | | } |
| | | } |
| | | } |
| | |
| | | // send request to server |
| | | this.http_post(action, '_uid='+a_uids.join(',')+'&_mbox='+urlencode(this.env.mailbox)+add_url, lock); |
| | | }; |
| | | |
| | | |
| | | // set a specific flag to one or more messages |
| | | this.mark_message = function(flag, uid) |
| | |
| | | this.toggle_read_status = function(flag, a_uids) |
| | | { |
| | | // mark all message rows as read/unread |
| | | var icn_src; |
| | | var rows = this.message_list.rows; |
| | | for (var i=0; i<a_uids.length; i++) |
| | | { |
| | | uid = a_uids[i]; |
| | | if (rows[uid]) |
| | | { |
| | | rows[uid].unread = (flag=='unread' ? true : false); |
| | | |
| | | if (rows[uid].classname.indexOf('unread')<0 && rows[uid].unread) |
| | | { |
| | | rows[uid].classname += ' unread'; |
| | | this.set_classname(rows[uid].obj, 'unread', true); |
| | | |
| | | if (this.env.unreadicon) |
| | | icn_src = this.env.unreadicon; |
| | | } |
| | | else if (!rows[uid].unread) |
| | | { |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*unread/, ''); |
| | | this.set_classname(rows[uid].obj, 'unread', false); |
| | | |
| | | if (this.env.messageicon) |
| | | icn_src = this.env.messageicon; |
| | | } |
| | | |
| | | if (rows[uid].icon && icn_src |
| | | && !(rows[uid].replied && this.env.repliedicon) |
| | | && !(rows[uid].deleted && this.env.deletedicon)) |
| | | rows[uid].icon.src = icn_src; |
| | | } |
| | | } |
| | | this.set_message(a_uids[i], 'unread', (flag=='unread' ? true : false)); |
| | | |
| | | this.http_post('mark', '_uid='+a_uids.join(',')+'&_flag='+flag); |
| | | }; |
| | | |
| | | // set class to read/unread |
| | | this.mark_as_read_from_preview = function(uid) |
| | | { |
| | | var icn_src; |
| | | var rows = parent.rcmail.message_list.rows; |
| | | if(rows[uid].unread) |
| | | { |
| | | rows[uid].unread = false; |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*unread/, ''); |
| | | parent.rcmail.set_classname(rows[uid].obj, 'unread', false); |
| | | |
| | | if (rows[uid].replied && parent.rcmail.env.repliedicon) |
| | | icn_src = parent.rcmail.env.repliedicon; |
| | | else if (rows[uid].deleted && parent.rcmail.env.deletedicon) |
| | | icn_src = parent.rcmail.env.deletedicon; |
| | | else if (parent.rcmail.env.messageicon) |
| | | icn_src = parent.rcmail.env.messageicon; |
| | | |
| | | if (rows[uid].icon && icn_src) |
| | | rows[uid].icon.src = icn_src; |
| | | } |
| | | } |
| | | |
| | | |
| | | // set image to flagged or unflagged |
| | | this.toggle_flagged_status = function(flag, a_uids) |
| | | { |
| | | // mark all message rows as flagged/unflagged |
| | | var icn_src; |
| | | var rows = this.message_list.rows; |
| | | for (var i=0; i<a_uids.length; i++) |
| | | { |
| | | uid = a_uids[i]; |
| | | if (rows[uid]) |
| | | { |
| | | rows[uid].flagged = (flag=='flagged' ? true : false); |
| | | |
| | | if (rows[uid].flagged && this.env.flaggedicon) |
| | | icn_src = this.env.flaggedicon; |
| | | else if (this.env.unflaggedicon) |
| | | icn_src = this.env.unflaggedicon; |
| | | |
| | | if (rows[uid].flagged_icon && icn_src) |
| | | rows[uid].flagged_icon.src = icn_src; |
| | | } |
| | | } |
| | | this.set_message(a_uids[i], 'flagged', (flag=='flagged' ? true : false)); |
| | | |
| | | this.http_post('mark', '_uid='+a_uids.join(',')+'&_flag='+flag); |
| | | }; |
| | |
| | | |
| | | if (a_uids.length==1) |
| | | { |
| | | if (!rows.length || (rows[a_uids[0]] && rows[a_uids[0]].classname.indexOf('deleted') < 0)) |
| | | if (!rows.length || (rows[a_uids[0]] && !rows[a_uids[0]].deleted)) |
| | | this.flag_as_deleted(a_uids); |
| | | else |
| | | this.flag_as_undeleted(a_uids); |
| | |
| | | { |
| | | uid = a_uids[i]; |
| | | if (rows[uid]) { |
| | | if (rows[uid].classname.indexOf('deleted')<0) |
| | | if (!rows[uid].deleted) |
| | | { |
| | | all_deleted = false; |
| | | break; |
| | |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.flag_as_undeleted = function(a_uids) |
| | | { |
| | | var icn_src; |
| | | var rows = this.message_list ? this.message_list.rows : new Array(); |
| | | |
| | | for (var i=0; i<a_uids.length; i++) |
| | | { |
| | | uid = a_uids[i]; |
| | | if (rows[uid]) { |
| | | rows[uid].deleted = false; |
| | | |
| | | if (rows[uid].classname.indexOf('deleted') > 0) |
| | | { |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*deleted/, ''); |
| | | this.set_classname(rows[uid].obj, 'deleted', false); |
| | | } |
| | | if (rows[uid].unread && this.env.unreadicon) |
| | | icn_src = this.env.unreadicon; |
| | | else if (rows[uid].replied && this.env.repliedicon) |
| | | icn_src = this.env.repliedicon; |
| | | else if (this.env.messageicon) |
| | | icn_src = this.env.messageicon; |
| | | |
| | | if (rows[uid].icon && icn_src) |
| | | rows[uid].icon.src = icn_src; |
| | | } |
| | | } |
| | | this.set_message(a_uids[i], 'deleted', false); |
| | | |
| | | this.http_post('mark', '_uid='+a_uids.join(',')+'&_flag=undelete'); |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.flag_as_deleted = function(a_uids) |
| | | { |
| | | var add_url = ''; |
| | |
| | | uid = a_uids[i]; |
| | | if (rows[uid]) |
| | | { |
| | | rows[uid].deleted = true; |
| | | |
| | | if (rows[uid].classname.indexOf('deleted')<0) |
| | | { |
| | | rows[uid].classname += ' deleted'; |
| | | this.set_classname(rows[uid].obj, 'deleted', true); |
| | | } |
| | | if (this.env.read_when_deleted) |
| | | { |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*unread/, ''); |
| | | this.set_classname(rows[uid].obj, 'unread', false); |
| | | } |
| | | |
| | | if (rows[uid].icon && this.env.deletedicon) |
| | | rows[uid].icon.src = this.env.deletedicon; |
| | | |
| | | this.set_message(uid, 'deleted', true); |
| | | if (rows[uid].unread) |
| | | r_uids[r_uids.length] = uid; |
| | | } |
| | |
| | | this.http_post('mark', '_uid='+a_uids.join(',')+'&_flag=delete'+add_url); |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | // flag as read without mark request (called from backend) |
| | | // argument should be a coma-separated list of uids |
| | |
| | | { |
| | | rows[uid].unread = false; |
| | | rows[uid].read = true; |
| | | |
| | | rows[uid].classname = rows[uid].classname.replace(/\s*unread/, ''); |
| | | this.set_classname(rows[uid].obj, 'unread', false); |
| | | |
| | | if (rows[uid].icon) |
| | | rows[uid].icon.src = this.env.deletedicon; |
| | | this.set_message(uid); |
| | | } |
| | | } |
| | | }; |
| | | |
| | | |
| | | |
| | | /*********************************************************/ |
| | | /********* login form methods *********/ |
| | | /*********************************************************/ |
| | |
| | | /********* message compose methods *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | // checks the input fields before sending a message |
| | | this.check_compose_input = function() |
| | | { |
| | |
| | | var input_to = rcube_find_object('_to'); |
| | | var input_cc = rcube_find_object('_cc'); |
| | | var input_bcc = rcube_find_object('_bcc'); |
| | | var input_from = rcube_find_object('_from'); |
| | | var input_subject = rcube_find_object('_subject'); |
| | | var input_message = rcube_find_object('_message'); |
| | | |
| | | // check sender (if have no identities) |
| | | if (input_from.type == 'text' && !rcube_check_email(input_from.value, true)) |
| | | { |
| | | alert(this.get_label('nosenderwarning')); |
| | | input_from.focus(); |
| | | return false; |
| | | } |
| | | |
| | | // check for empty recipient |
| | | var recipients = input_to.value ? input_to.value : (input_cc.value ? input_cc.value : input_bcc.value); |
| | |
| | | this.display_spellcheck_controls = function(vis) |
| | | { |
| | | if (this.env.spellcheck) { |
| | | // stop spellchecking process |
| | | if (!vis && !this.spellcheck_ready) |
| | | { |
| | | exec_event(this.env.spellcheck.check_link, 'click'); |
| | | this.set_spellcheck_state('ready'); |
| | | } |
| | | |
| | | this.env.spellcheck.check_link.style.visibility = vis ? 'visible' : 'hidden'; |
| | | this.env.spellcheck.switch_lan_pic.style.visibility = vis ? 'visible' : 'hidden'; |
| | | } |
| | |
| | | this.spellcheck_ready = (s=='check_spelling' || s=='ready'); |
| | | this.enable_command('spellcheck', this.spellcheck_ready); |
| | | }; |
| | | |
| | | |
| | | this.set_draft_id = function(id) |
| | | { |
| | |
| | | // Unlock interface now that saving is complete |
| | | this.busy = false; |
| | | }; |
| | | |
| | | |
| | | this.compose_field_hash = function(save) |
| | | { |
| | |
| | | return str; |
| | | }; |
| | | |
| | | |
| | | this.change_identity = function(obj) |
| | | { |
| | | if (!obj || !obj.options) |
| | |
| | | // remove the 'old' signature |
| | | if (this.env.identity && this.env.signatures && this.env.signatures[this.env.identity]) |
| | | { |
| | | sig = this.env.signatures[this.env.identity]['text']; |
| | | if (sig.indexOf('-- ')!=0) |
| | | if (this.env.signatures[this.env.identity]['is_html']) |
| | | sig = this.env.signatures[this.env.identity]['plain_text']; |
| | | else |
| | | sig = this.env.signatures[this.env.identity]['text']; |
| | | |
| | | if (sig.indexOf('-- ')!=0) |
| | | sig = '-- \n'+sig; |
| | | |
| | | p = message.lastIndexOf(sig); |
| | |
| | | { |
| | | var editor = tinyMCE.get('compose-body'); |
| | | |
| | | if (this.env.signatures && this.env.signatures[id]) |
| | | if (this.env.signatures) |
| | | { |
| | | // Append the signature as a span within the body |
| | | // Append the signature as a div within the body |
| | | var sigElem = editor.dom.get("_rc_sig"); |
| | | var newsig = ''; |
| | | var htmlsig = true; |
| | | |
| | | if (!sigElem) |
| | | { |
| | | sigElem = editor.getDoc().createElement("span"); |
| | | sigElem = editor.getDoc().createElement("div"); |
| | | sigElem.setAttribute("id", "_rc_sig"); |
| | | editor.getBody().appendChild(sigElem); |
| | | } |
| | | if (this.env.signatures[id]['is_html']) |
| | | { |
| | | sigElem.innerHTML = this.env.signatures[id]['text']; |
| | | } |
| | | |
| | | if (this.env.signatures[id]) |
| | | { |
| | | newsig = this.env.signatures[id]['text']; |
| | | htmlsig = this.env.signatures[id]['is_html']; |
| | | } |
| | | |
| | | if (htmlsig) |
| | | sigElem.innerHTML = newsig; |
| | | else |
| | | { |
| | | sigElem.innerHTML = '<pre>' + this.env.signatures[id]['text'] + '</pre>'; |
| | | } |
| | | sigElem.innerHTML = '<pre>' + newsig + '</pre>'; |
| | | } |
| | | } |
| | | |
| | |
| | | this.env.identity = id; |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.show_attachment_form = function(a) |
| | | { |
| | |
| | | |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | // upload attachment file |
| | | this.upload_file = function(form) |
| | |
| | | this.gui_objects.attachmentform = form; |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | // add file name to attachment list |
| | | // called from upload page |
| | |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.sent_successfully = function(type, msg) |
| | | { |
| | | this.list_mailbox(); |
| | |
| | | /*********************************************************/ |
| | | /********* keyboard live-search methods *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | // handler for keyboard events on address-fields |
| | | this.ksearch_keypress = function(e, obj) |
| | |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.insert_recipient = function(id) |
| | | { |
| | | if (!this.env.contacts[id] || !this.ksearch_input) |
| | |
| | | if (this.ksearch_input.setSelectionRange) |
| | | this.ksearch_input.setSelectionRange(cpos, cpos); |
| | | }; |
| | | |
| | | |
| | | // address search processor |
| | | this.ksearch_get_results = function() |
| | |
| | | this.ksearch_hide(); |
| | | }; |
| | | |
| | | |
| | | this.ksearch_blur = function(e, obj) |
| | | { |
| | | if (this.ksearch_timer) |
| | |
| | | }; |
| | | |
| | | |
| | | |
| | | /*********************************************************/ |
| | | /********* address book methods *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | this.contactlist_keypress = function(list) |
| | | { |
| | | if (list.key_pressed == list.DELETE_KEY) |
| | | this.command('delete'); |
| | | }; |
| | | |
| | | |
| | | this.contactlist_select = function(list) |
| | | { |
| | |
| | | |
| | | var id, frame, ref = this; |
| | | if (id = list.get_single_selection()) |
| | | this.preview_timer = window.setTimeout(function(){ ref.load_contact(id, 'show'); }, this.dblclick_time + 10); |
| | | this.preview_timer = window.setTimeout(function(){ ref.load_contact(id, 'show'); }, 200); |
| | | else if (this.env.contentframe) |
| | | this.show_contentframe(false); |
| | | |
| | | this.enable_command('edit', id?true:false); |
| | | this.enable_command('compose', list.selection.length > 0); |
| | | this.enable_command('edit', (id && this.env.address_sources && !this.env.address_sources[this.env.source].readonly) ? true : false); |
| | | this.enable_command('delete', list.selection.length && this.env.address_sources && !this.env.address_sources[this.env.source].readonly); |
| | | |
| | | return false; |
| | | }; |
| | | |
| | | |
| | | this.list_contacts = function(src, page) |
| | | { |
| | |
| | | target.location.href = this.env.comm_path+(src ? '&_source='+urlencode(src) : '')+(page ? '&_page='+page : '')+add_url; |
| | | }; |
| | | |
| | | |
| | | // send remote request to load contacts list |
| | | this.list_contacts_remote = function(src, page) |
| | | { |
| | |
| | | this.set_busy(true, 'loading'); |
| | | this.http_request('list', url, true); |
| | | }; |
| | | |
| | | |
| | | // load contact record |
| | | this.load_contact = function(cid, action, framed) |
| | |
| | | |
| | | // copy a contact to the specified target (group or directory) |
| | | this.copy_contact = function(cid, to) |
| | | { |
| | | { |
| | | if (!cid) |
| | | cid = this.contact_list.get_selection().join(','); |
| | | |
| | | if (to != this.env.source && cid && this.env.address_sources[to] && !this.env.address_sources[to].readonly) |
| | | this.http_post('copy', '_cid='+urlencode(cid)+'&_source='+urlencode(this.env.source)+'&_to='+urlencode(to)); |
| | | }; |
| | | }; |
| | | |
| | | |
| | | this.delete_contacts = function() |
| | |
| | | this.http_post('delete', '_cid='+urlencode(a_cids.join(','))+'&_source='+urlencode(this.env.source)+'&_from='+(this.env.action ? this.env.action : '')+qs); |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | // update a contact record in the list |
| | | this.update_contact_row = function(cid, cols_arr) |
| | |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.delete_identity = function(id) |
| | | { |
| | | // exit if no mailbox specified or if selection is empty |
| | |
| | | this.goto_url('delete-identity', '_iid='+id, true); |
| | | return true; |
| | | }; |
| | | |
| | | |
| | | this.focus_subscription = function(id) |
| | | { |
| | |
| | | } |
| | | } |
| | | |
| | | |
| | | this.unfocus_subscription = function(id) |
| | | { |
| | | var row; |
| | |
| | | else |
| | | this.set_classname(this.subscription_list.frame, 'droptarget', false); |
| | | } |
| | | |
| | | |
| | | this.subscription_select = function(list) |
| | | { |
| | |
| | | this.gui_objects.createfolderhint.innerHTML = this.env.folder ? this.get_label('addsubfolderhint') : ''; |
| | | }; |
| | | |
| | | |
| | | this.subscription_move_folder = function(list) |
| | | { |
| | | var reg = RegExp('['+RegExp.escape(this.env.delimiter)+']?[^'+RegExp.escape(this.env.delimiter)+']+$'); |
| | |
| | | this.drag_active = false; |
| | | this.unfocus_subscription(this.get_folder_row_id(this.env.dstfolder)); |
| | | }; |
| | | |
| | | |
| | | // tell server to create and subscribe a new mailbox |
| | | this.create_folder = function(name) |
| | |
| | | if (this.env.folder && name != '') |
| | | name = this.env.folder+this.env.delimiter+name; |
| | | |
| | | this.set_busy(true, 'foldercreating'); |
| | | this.http_post('create-folder', '_name='+urlencode(name), true); |
| | | } |
| | | else if (form.elements['_folder_name']) |
| | | form.elements['_folder_name'].focus(); |
| | | }; |
| | | |
| | | |
| | | // start renaming the mailbox name. |
| | | // this will replace the name string with an input field |
| | |
| | | } |
| | | }; |
| | | |
| | | |
| | | // remove the input field and write the current mailbox name to the table cell |
| | | this.reset_folder_rename = function() |
| | | { |
| | |
| | | |
| | | this.edit_folder = null; |
| | | }; |
| | | |
| | | |
| | | // handler for keyboard events on the input field |
| | | this.name_input_keypress = function(e) |
| | |
| | | if (this.name_input.__parent) |
| | | newname = this.name_input.__parent + this.env.delimiter + newname; |
| | | |
| | | this.set_busy(true, 'folderrenaming'); |
| | | this.http_post('rename-folder', '_folder_oldname='+urlencode(this.env.subscriptionrows[this.edit_folder][0])+'&_folder_newname='+urlencode(newname), true); |
| | | } |
| | | } |
| | |
| | | else if (key==27) |
| | | this.reset_folder_rename(); |
| | | }; |
| | | |
| | | |
| | | // delete a specific mailbox with all its messages |
| | | this.delete_folder = function(id) |
| | |
| | | |
| | | if (folder && confirm(this.get_label('deletefolderconfirm'))) |
| | | { |
| | | this.http_post('delete-folder', '_mboxes='+urlencode(folder)); |
| | | this.set_busy(true, 'folderdeleting'); |
| | | this.http_post('delete-folder', '_mboxes='+urlencode(folder), true); |
| | | this.set_env('folder', null); |
| | | |
| | | if (this.gui_objects.createfolderhint) |
| | | this.gui_objects.createfolderhint.innerHTML = ''; |
| | | } |
| | | }; |
| | | |
| | | |
| | | // add a new folder to the subscription list by cloning a folder row |
| | | this.add_folder_row = function(name, display_name, replace, before) |
| | |
| | | row.id = id; |
| | | |
| | | if (before && (before = this.get_folder_row_id(before))) |
| | | tbody.insertBefore(row, document.getElementById(before)); |
| | | tbody.insertBefore(row, document.getElementById(before)); |
| | | else |
| | | tbody.appendChild(row); |
| | | tbody.appendChild(row); |
| | | |
| | | if (replace) |
| | | tbody.removeChild(replace); |
| | | tbody.removeChild(replace); |
| | | } |
| | | |
| | | // add to folder/row-ID map |
| | |
| | | document.getElementById(id).scrollIntoView(); |
| | | }; |
| | | |
| | | |
| | | // replace an existing table row with a new folder line |
| | | this.replace_folder_row = function(oldfolder, newfolder, display_name, before) |
| | | { |
| | |
| | | form.elements['_folder_newname'].value = ''; |
| | | } |
| | | }; |
| | | |
| | | |
| | | // remove the table row of a specific mailbox from the table |
| | | // (the row will not be removed, just hidden) |
| | |
| | | form.elements['_folder_newname'].value = ''; |
| | | }; |
| | | |
| | | |
| | | this.subscribe_folder = function(folder) |
| | | { |
| | | if (folder) |
| | | this.http_post('subscribe', '_mbox='+urlencode(folder)); |
| | | }; |
| | | |
| | | |
| | | this.unsubscribe_folder = function(folder) |
| | | { |
| | |
| | | this.http_post('unsubscribe', '_mbox='+urlencode(folder)); |
| | | }; |
| | | |
| | | |
| | | // helper method to find a specific mailbox row ID |
| | | this.get_folder_row_id = function(folder) |
| | | { |
| | |
| | | { |
| | | var cell, td; |
| | | var new_row = document.createElement('TR'); |
| | | for(var n=0; n<row.childNodes.length; n++) |
| | | for(var n=0; n<row.cells.length; n++) |
| | | { |
| | | cell = row.childNodes[n]; |
| | | cell = row.cells[n]; |
| | | td = document.createElement('TD'); |
| | | |
| | | if (cell.className) |
| | |
| | | /********* GUI functionality *********/ |
| | | /*********************************************************/ |
| | | |
| | | |
| | | // eable/disable buttons for page shifting |
| | | this.set_page_buttons = function() |
| | | { |
| | |
| | | this.enable_command('previouspage', (this.env.current_page > 1)); |
| | | this.enable_command('firstpage', (this.env.current_page > 1)); |
| | | } |
| | | |
| | | |
| | | // set button to a specific state |
| | | this.set_button = function(command, state) |
| | |
| | | } |
| | | }; |
| | | |
| | | |
| | | // set/unset a specific class name |
| | | this.set_classname = function(obj, classname, set) |
| | | { |
| | |
| | | obj.className += ' '+classname; |
| | | }; |
| | | |
| | | |
| | | // write to the document/window title |
| | | this.set_pagetitle = function(title) |
| | | { |
| | | if (title && document.title) |
| | | document.title = title; |
| | | } |
| | | |
| | | |
| | | // display a system message |
| | | this.display_message = function(msg, type, hold) |
| | |
| | | this.message_timer = window.setTimeout(function(){ ref.hide_message(); }, this.message_time); |
| | | }; |
| | | |
| | | |
| | | // make a message row disapear |
| | | this.hide_message = function() |
| | | { |
| | |
| | | this.gui_objects.message.onmousedown = null; |
| | | } |
| | | }; |
| | | |
| | | |
| | | // mark a mailbox as selected and set environment variable |
| | | this.select_folder = function(name, old) |
| | |
| | | |
| | | return null; |
| | | }; |
| | | |
| | | |
| | | // for reordering column array, Konqueror workaround |
| | | this.set_message_coltypes = function(coltypes) |
| | |
| | | this.env.messages[uid] = {deleted:flags.deleted?1:0, |
| | | replied:flags.replied?1:0, |
| | | unread:flags.unread?1:0, |
| | | forwarded:flags.forwarded?1:0, |
| | | flagged:flags.flagged?1:0}; |
| | | |
| | | var row = document.createElement('TR'); |
| | | row.id = 'rcmrow'+uid; |
| | | row.className = 'message '+(even ? 'even' : 'odd')+(flags.unread ? ' unread' : '')+(flags.deleted ? ' deleted' : ''); |
| | | row.className = 'message' |
| | | + (even ? ' even' : ' odd') |
| | | + (flags.unread ? ' unread' : '') |
| | | + (flags.deleted ? ' deleted' : '') |
| | | + (flags.flagged ? ' flagged' : ''); |
| | | |
| | | if (this.message_list.in_selection(uid)) |
| | | row.className += ' selected'; |
| | | |
| | | var icon = flags.deleted && this.env.deletedicon ? this.env.deletedicon: |
| | | (flags.unread && this.env.unreadicon ? this.env.unreadicon : |
| | | (flags.replied && this.env.repliedicon ? this.env.repliedicon : this.env.messageicon)); |
| | | |
| | | var icon = this.env.messageicon; |
| | | if (flags.deleted && this.env.deletedicon) |
| | | icon = this.env.deletedicon; |
| | | else if (flags.replied && this.env.repliedicon) |
| | | { |
| | | if (flags.forwarded && this.env.forwardedrepliedicon) |
| | | icon = this.env.forwardedrepliedicon; |
| | | else |
| | | icon = this.env.repliedicon; |
| | | } |
| | | else if (flags.forwarded && this.env.forwardedicon) |
| | | icon = this.env.forwardedicon; |
| | | else if(flags.unread && this.env.unreadicon) |
| | | icon = this.env.unreadicon; |
| | | |
| | | var col = document.createElement('TD'); |
| | | col.className = 'icon'; |
| | | col.innerHTML = icon ? '<img src="'+icon+'" alt="" />' : ''; |
| | |
| | | row.appendChild(col); |
| | | |
| | | this.message_list.insert_row(row, attop); |
| | | }; |
| | | |
| | | // remove 'old' row |
| | | if (attop && this.env.pagesize && this.message_list.rowcount > this.env.pagesize) |
| | | { |
| | | var uid = this.message_list.get_last_row(); |
| | | this.message_list.remove_row(uid); |
| | | this.message_list.clear_selection(uid); |
| | | } |
| | | }; |
| | | |
| | | // replace content of row count display |
| | | this.set_rowcount = function(text) |
| | |
| | | // update page navigation buttons |
| | | this.set_page_buttons(); |
| | | }; |
| | | |
| | | |
| | | // replace content of mailboxname display |
| | | this.set_mailboxname = function(content) |
| | |
| | | this.gui_objects.quotadisplay.innerHTML = content; |
| | | }; |
| | | |
| | | |
| | | // update the mailboxlist |
| | | this.set_unread_count = function(mbox, count, set_title) |
| | | { |
| | | if (!this.gui_objects.mailboxlist) |
| | | return false; |
| | | |
| | | var reg, text_obj, item; |
| | | this.env.unread_counts[mbox] = count; |
| | | this.set_unread_count_display(mbox, set_title); |
| | | } |
| | | |
| | | // update the mailbox count display |
| | | this.set_unread_count_display = function(mbox, set_title) |
| | | { |
| | | var reg, text_obj, item, mycount, childcount, div; |
| | | if (item = this.get_folder_li(mbox)) |
| | | { |
| | | // set new text |
| | | text_obj = item.firstChild; |
| | | mycount = this.env.unread_counts[mbox] ? this.env.unread_counts[mbox] : 0; |
| | | text_obj = item.getElementsByTagName('a')[0]; |
| | | reg = /\s+\([0-9]+\)$/i; |
| | | |
| | | if (count && text_obj.innerHTML.match(reg)) |
| | | text_obj.innerHTML = text_obj.innerHTML.replace(reg, ' ('+count+')'); |
| | | else if (count) |
| | | text_obj.innerHTML += ' ('+count+')'; |
| | | childcount = 0; |
| | | if ((div = item.getElementsByTagName('div')[0]) && |
| | | div.className.match(/collapsed/)) |
| | | { |
| | | // add children's counters |
| | | for (var k in this.env.unread_counts) |
| | | if (k.indexOf(mbox + this.env.delimiter) == 0) { |
| | | childcount += this.env.unread_counts[k]; |
| | | } |
| | | } |
| | | |
| | | if (mycount && text_obj.innerHTML.match(reg)) |
| | | text_obj.innerHTML = text_obj.innerHTML.replace(reg, ' ('+mycount+')'); |
| | | else if (mycount) |
| | | text_obj.innerHTML += ' ('+mycount+')'; |
| | | else |
| | | text_obj.innerHTML = text_obj.innerHTML.replace(reg, ''); |
| | | |
| | | // set parent's display |
| | | reg = new RegExp(RegExp.escape(this.env.delimiter) + '[^' + RegExp.escape(this.env.delimiter) + ']+$'); |
| | | if (mbox.match(reg)) |
| | | this.set_unread_count_display(mbox.replace(reg, ''), false); |
| | | |
| | | // set the right classes |
| | | this.set_classname(item, 'unread', count>0 ? true : false); |
| | | this.set_classname(item, 'unread', (mycount+childcount)>0 ? true : false); |
| | | } |
| | | |
| | | // set unread count to window title |
| | |
| | | var doc_title = String(document.title); |
| | | var new_title = ""; |
| | | |
| | | if (count && doc_title.match(reg)) |
| | | new_title = doc_title.replace(reg, '('+count+') '); |
| | | else if (count) |
| | | new_title = '('+count+') '+doc_title; |
| | | if (mycount && doc_title.match(reg)) |
| | | new_title = doc_title.replace(reg, '('+mycount+') '); |
| | | else if (mycount) |
| | | new_title = '('+mycount+') '+doc_title; |
| | | else |
| | | new_title = doc_title.replace(reg, ''); |
| | | |
| | |
| | | } |
| | | }; |
| | | |
| | | // update parent's mailboxlist (from preview) |
| | | this.set_unread_count_from_preview = function(mbox, count, set_title) |
| | | { |
| | | parent.rcmail.set_unread_count(mbox, count, set_title); |
| | | } |
| | | |
| | | // add row to contacts list |
| | | this.add_contact_row = function(cid, cols, select) |
| | | { |
| | |
| | | } |
| | | |
| | | this.contact_list.insert_row(row); |
| | | this.enable_command('export', (this.contact_list.rowcount > 0)); |
| | | }; |
| | | |
| | | |
| | | this.toggle_editor = function(checkbox, textAreaId) |
| | | { |
| | | var ischecked = checkbox.checked; |
| | | var composeElement = document.getElementById(textAreaId); |
| | | |
| | | if (ischecked) |
| | | { |
| | | tinyMCE.execCommand('mceAddControl', true, textAreaId); |
| | | var existingPlainText = composeElement.value; |
| | | var htmlText = "<pre>" + existingPlainText + "</pre>"; |
| | | composeElement.value = htmlText; |
| | | tinyMCE.execCommand('mceAddControl', true, textAreaId); |
| | | } |
| | | else |
| | | { |
| | | tinyMCE.execCommand('mceRemoveControl', true, textAreaId); |
| | | var thisMCE = tinyMCE.get(textAreaId); |
| | | var existingHtml = thisMCE.getContent(); |
| | | this.html2plain(existingHtml, textAreaId); |
| | | tinyMCE.execCommand('mceRemoveControl', true, textAreaId); |
| | | } |
| | | }; |
| | | |
| | | |
| | | this.toggle_prefer_html = function(checkbox) |
| | | { |
| | |
| | | addrbook_show_images.disabled = !checkbox.checked; |
| | | } |
| | | |
| | | // display fetched raw headers |
| | | this.set_headers = function(content) |
| | | { |
| | | if (this.gui_objects.all_headers_row && this.gui_objects.all_headers_box && content) |
| | | { |
| | | var box = this.gui_objects.all_headers_box; |
| | | box.innerHTML = content; |
| | | box.style.display = 'block'; |
| | | |
| | | if (this.env.framed && parent.rcmail) |
| | | parent.rcmail.set_busy(false); |
| | | else |
| | | this.set_busy(false); |
| | | } |
| | | }; |
| | | |
| | | // display all-headers row and fetch raw message headers |
| | | this.load_headers = function(elem) |
| | | { |
| | | if (!this.gui_objects.all_headers_row || !this.gui_objects.all_headers_box || !this.env.uid) |
| | | return; |
| | | |
| | | this.set_classname(elem, 'show-headers', false); |
| | | this.set_classname(elem, 'hide-headers', true); |
| | | this.gui_objects.all_headers_row.style.display = bw.ie ? 'block' : 'table-row'; |
| | | elem.onclick = function() { rcmail.hide_headers(elem); } |
| | | |
| | | // fetch headers only once |
| | | if (!this.gui_objects.all_headers_box.innerHTML) |
| | | { |
| | | this.display_message(this.get_label('loading'), 'loading', true); |
| | | this.http_post('headers', '_uid='+this.env.uid); |
| | | } |
| | | } |
| | | |
| | | // hide all-headers row |
| | | this.hide_headers = function(elem) |
| | | { |
| | | if (!this.gui_objects.all_headers_row || !this.gui_objects.all_headers_box) |
| | | return; |
| | | |
| | | this.set_classname(elem, 'hide-headers', false); |
| | | this.set_classname(elem, 'show-headers', true); |
| | | this.gui_objects.all_headers_row.style.display = 'none'; |
| | | elem.onclick = function() { rcmail.load_headers(elem); } |
| | | } |
| | | |
| | | |
| | | /********************************************************/ |
| | | /********* html to text conversion functions *********/ |
| | | /********************************************************/ |
| | | |
| | | this.html2plain = function(htmlText, id) |
| | | { |
| | | var http_request = new rcube_http_request(); |
| | | var url = this.env.bin_path+'html2text.php'; |
| | | var rcmail = this; |
| | | |
| | | this.set_busy(true, 'converting'); |
| | | console.log('HTTP POST: '+url); |
| | | |
| | | http_request.onerror = function(o) { rcmail.http_error(o); }; |
| | | http_request.oncomplete = function(o) { rcmail.set_text_value(o, id); }; |
| | | http_request.POST(url, htmlText, 'application/octet-stream'); |
| | | } |
| | | |
| | | this.set_text_value = function(httpRequest, id) |
| | | { |
| | | this.set_busy(false); |
| | | document.getElementById(id).value = httpRequest.get_text(); |
| | | console.log(httpRequest.get_text()); |
| | | } |
| | | |
| | | |
| | | /********************************************************/ |
| | |
| | | this.redirect(this.env.comm_path+'&_action='+action+querystring, lock); |
| | | }; |
| | | |
| | | |
| | | this.http_sockets = new Array(); |
| | | |
| | | // find a non-busy socket or create a new one |
| | |
| | | return this.http_sockets[i]; |
| | | }; |
| | | |
| | | |
| | | // send a http request to the server |
| | | this.http_request = function(action, querystring, lock) |
| | | { |
| | |
| | | this.http_response = function(request_obj) |
| | | { |
| | | var ctype = request_obj.get_header('Content-Type'); |
| | | if (ctype){ |
| | | if (ctype) |
| | | { |
| | | ctype = String(ctype).toLowerCase(); |
| | | var ctype_array=ctype.split(";"); |
| | | ctype = ctype_array[0]; |
| | | } |
| | | } |
| | | |
| | | if (request_obj.__lock) |
| | | this.set_busy(false); |
| | | this.set_busy(false); |
| | | |
| | | console.log(request_obj.get_text()); |
| | | |
| | |
| | | eval(request_obj.get_text()); |
| | | |
| | | // process the response data according to the sent action |
| | | switch (request_obj.__action) |
| | | { |
| | | |
| | | switch (request_obj.__action) { |
| | | case 'delete': |
| | | if (this.task == 'addressbook') { |
| | | var uid = this.contact_list.get_selection(); |
| | | this.enable_command('compose', (uid && this.contact_list.rows[uid])); |
| | | this.enable_command('delete', 'edit', (uid && this.contact_list.rows[uid] && this.env.address_sources && !this.env.address_sources[this.env.source].readonly)); |
| | | this.enable_command('export', (this.contact_list && this.contact_list.rowcount > 0)); |
| | | } |
| | | |
| | | case 'moveto': |
| | | if (this.env.action=='show') |
| | | if (this.env.action == 'show') |
| | | this.command('list'); |
| | | else if (this.message_list) |
| | | this.message_list.init(); |
| | | |
| | | break; |
| | | |
| | | case 'purge': |
| | | case 'expunge': |
| | | if (!this.env.messagecount) |
| | | { |
| | | // clear preview pane content |
| | | if (this.env.contentframe) |
| | | this.show_contentframe(false); |
| | | // disable commands useless when mailbox is empty |
| | | this.enable_command('show', 'reply', 'reply-all', 'forward', 'moveto', 'delete', 'mark', 'viewsource', |
| | | 'print', 'load-attachment', 'purge', 'expunge', 'select-all', 'select-none', 'sort', false); |
| | | } |
| | | |
| | | break; |
| | | |
| | | case 'list': |
| | | this.msglist_select(this.message_list); |
| | | if (!this.env.messagecount && this.task == 'mail') { |
| | | // clear preview pane content |
| | | if (this.env.contentframe) |
| | | this.show_contentframe(false); |
| | | // disable commands useless when mailbox is empty |
| | | this.enable_command('show', 'reply', 'reply-all', 'forward', 'moveto', 'delete', 'mark', 'viewsource', |
| | | 'print', 'load-attachment', 'purge', 'expunge', 'select-all', 'select-none', 'sort', false); |
| | | } |
| | | break; |
| | | |
| | | case 'check-recent': |
| | | case 'getunread': |
| | | this.enable_command('show', 'expunge', 'select-all', 'select-none', 'sort', (this.env.messagecount > 0)); |
| | | var mailboxtest = (this.env.mailbox == this.env.trash_mailbox || this.env.mailbox == this.env.junk_mailbox |
| | | || this.env.mailbox.match('^' + RegExp.escape(this.env.trash_mailbox) + RegExp.escape(this.env.delimiter)) |
| | | || this.env.mailbox.match('^' + RegExp.escape(this.env.junk_mailbox) + RegExp.escape(this.env.delimiter))) ? true : false; |
| | | |
| | | this.enable_command('purge', (this.env.messagecount && mailboxtest)); |
| | | case 'list': |
| | | if (this.task == 'mail') { |
| | | if (this.message_list && request_obj.__action == 'list') |
| | | this.msglist_select(this.message_list); |
| | | this.enable_command('show', 'expunge', 'select-all', 'select-none', 'sort', (this.env.messagecount > 0)); |
| | | this.enable_command('purge', this.purge_mailbox_test()); |
| | | } |
| | | else if (this.task == 'addressbook') |
| | | this.enable_command('export', (this.contact_list && this.contact_list.rowcount > 0)); |
| | | |
| | | break; |
| | | |
| | | break; |
| | | } |
| | | |
| | | request_obj.reset(); |
| | | }; |
| | | |
| | | |
| | | // handle HTTP request errors |
| | | this.http_error = function(request_obj) |
| | | { |
| | | //alert('Error sending request: '+request_obj.url); |
| | | |
| | | //alert('Error sending request: '+request_obj.url+' => HTTP '+request_obj.xmlhttp.status); |
| | | if (request_obj.__lock) |
| | | this.set_busy(false); |
| | | |
| | | request_obj.reset(); |
| | | request_obj.__lock = false; |
| | | this.display_message('Unknown Server Error!', 'error'); |
| | | }; |
| | | |
| | | |
| | | // use an image to send a keep-alive siganl to the server |
| | | this.send_keep_alive = function() |
| | |
| | | this.http_request('keep-alive', '_t='+d.getTime()); |
| | | }; |
| | | |
| | | |
| | | // send periodic request to check for recent messages |
| | | this.check_for_recent = function() |
| | | { |
| | |
| | | return obj.value.length; |
| | | }; |
| | | |
| | | |
| | | this.set_caret2start = function(obj) |
| | | { |
| | | if (obj.createTextRange) |
| | |
| | | |
| | | obj.focus(); |
| | | }; |
| | | |
| | | |
| | | // set all fields of a form disabled |
| | | this.lock_form = function(form, lock) |
| | |
| | | } // end object rcube_webmail |
| | | |
| | | |
| | | |
| | | /** |
| | | * Class for sending HTTP requests |
| | | * @constructor |
| | |
| | | this.url = ''; |
| | | this.busy = false; |
| | | this.xmlhttp = null; |
| | | |
| | | |
| | | // reset object properties |
| | | this.reset = function() |
| | |
| | | this.busy = false; |
| | | this.xmlhttp = null; |
| | | } |
| | | |
| | | |
| | | // create HTMLHTTP object |
| | | this.build = function() |
| | |
| | | this.busy = true; |
| | | |
| | | this.xmlhttp.onreadystatechange = function(){ _ref.xmlhttp_onreadystatechange(); }; |
| | | this.xmlhttp.open('GET', url); |
| | | this.xmlhttp.open('GET', url, true); |
| | | this.xmlhttp.setRequestHeader('X-RoundCube-Referer', bw.get_cookie('roundcube_sessid')); |
| | | this.xmlhttp.send(null); |
| | | }; |
| | | |
| | | |
| | | this.POST = function(url, body, contentType) |
| | | { |
| | |
| | | this.xmlhttp.setRequestHeader('X-RoundCube-Referer', bw.get_cookie('roundcube_sessid')); |
| | | this.xmlhttp.send(req_body); |
| | | }; |
| | | |
| | | |
| | | // handle onreadystatechange event |
| | | this.xmlhttp_onreadystatechange = function() |
| | |
| | | window.setTimeout('if (window[\''+o+'\'] && window[\''+o+'\'].init) { '+o+'.init(); }', |
| | | bw.win ? 500 : 200); |
| | | } |
| | | |