| +-----------------------------------------------------------------------+ */ $COMPOSE_ID = rcube_utils::get_input_value('_id', rcube_utils::INPUT_GET); $COMPOSE = null; if ($COMPOSE_ID && $_SESSION['compose_data_'.$COMPOSE_ID]) { $COMPOSE =& $_SESSION['compose_data_'.$COMPOSE_ID]; } // give replicated session storage some time to synchronize $retries = 0; while ($COMPOSE_ID && !is_array($COMPOSE) && $RCMAIL->db->is_replicated() && $retries++ < 5) { usleep(500000); $RCMAIL->session->reload(); if ($_SESSION['compose_data_'.$COMPOSE_ID]) { $COMPOSE =& $_SESSION['compose_data_'.$COMPOSE_ID]; } } // Nothing below is called during message composition, only at "new/forward/reply/draft" initialization or // if a compose-ID is given (i.e. when the compose step is opened in a new window/tab). if (!is_array($COMPOSE)) { // Infinite redirect prevention in case of broken session (#1487028) if ($COMPOSE_ID) { // if we know the message with specified ID was already sent // we can ignore the error and compose a new message (#1490009) if ($COMPOSE_ID != $_SESSION['last_compose_session']) { rcube::raise_error(array('code' => 450), false, true); } } $COMPOSE_ID = uniqid(mt_rand()); $params = rcube_utils::request2param(rcube_utils::INPUT_GET, 'task|action', true); $_SESSION['compose_data_'.$COMPOSE_ID] = array( 'id' => $COMPOSE_ID, 'param' => $params, 'mailbox' => $params['mbox'] ?: $RCMAIL->storage->get_folder(), ); $COMPOSE =& $_SESSION['compose_data_'.$COMPOSE_ID]; rcmail_process_compose_params($COMPOSE); // check if folder for saving sent messages exists and is subscribed (#1486802) if ($sent_folder = $COMPOSE['param']['sent_mbox']) { rcmail_sendmail::check_sent_folder($sent_folder, true); } // redirect to a unique URL with all parameters stored in session $OUTPUT->redirect(array( '_action' => 'compose', '_id' => $COMPOSE['id'], '_search' => $_REQUEST['_search'], )); } // add some labels to client $OUTPUT->add_label('notuploadedwarning', 'savingmessage', 'siginserted', 'responseinserted', 'messagesaved', 'converting', 'editorwarning', 'discard', 'fileuploaderror', 'sendmessage', 'newresponse', 'responsename', 'responsetext', 'save', 'savingresponse', 'restoresavedcomposedata', 'restoremessage', 'delete', 'restore', 'ignore', 'selectimportfile', 'messageissent', 'loadingdata', 'nopubkeyfor', 'nopubkeyforsender', 'encryptnoattachments','encryptedsendialog','searchpubkeyservers', 'importpubkeys', 'encryptpubkeysfound', 'search', 'close', 'import', 'keyid', 'keylength', 'keyexpired', 'keyrevoked', 'keyimportsuccess', 'keyservererror', 'attaching', 'namex', 'attachmentrename' ); $OUTPUT->set_pagetitle($RCMAIL->gettext('compose')); $OUTPUT->set_env('compose_id', $COMPOSE['id']); $OUTPUT->set_env('session_id', session_id()); $OUTPUT->set_env('mailbox', $RCMAIL->storage->get_folder()); $OUTPUT->set_env('top_posting', intval($RCMAIL->config->get('reply_mode')) > 0); $OUTPUT->set_env('sig_below', $RCMAIL->config->get('sig_below')); $OUTPUT->set_env('save_localstorage', (bool)$RCMAIL->config->get('compose_save_localstorage')); $OUTPUT->set_env('is_sent', false); $OUTPUT->set_env('mimetypes', rcmail_supported_mimetypes()); $OUTPUT->set_env('keyservers', $RCMAIL->config->keyservers()); $drafts_mbox = $RCMAIL->config->get('drafts_mbox'); $config_show_sig = $RCMAIL->config->get('show_sig', 1); // add config parameters to client script if (strlen($drafts_mbox)) { $OUTPUT->set_env('drafts_mailbox', $drafts_mbox); $OUTPUT->set_env('draft_autosave', $RCMAIL->config->get('draft_autosave')); } // default font for HTML editor $font = rcmail::font_defs($RCMAIL->config->get('default_font')); if ($font && !is_array($font)) { $OUTPUT->set_env('default_font', $font); } // default font size for HTML editor if ($font_size = $RCMAIL->config->get('default_font_size')) { $OUTPUT->set_env('default_font_size', $font_size); } // get reference message and set compose mode if ($msg_uid = $COMPOSE['param']['draft_uid']) { $compose_mode = rcmail_sendmail::MODE_DRAFT; $OUTPUT->set_env('draft_id', $msg_uid); $RCMAIL->storage->set_folder($drafts_mbox); } else if ($msg_uid = $COMPOSE['param']['reply_uid']) { $compose_mode = rcmail_sendmail::MODE_REPLY; } else if ($msg_uid = $COMPOSE['param']['forward_uid']) { $compose_mode = rcmail_sendmail::MODE_FORWARD; $COMPOSE['forward_uid'] = $msg_uid; $COMPOSE['as_attachment'] = !empty($COMPOSE['param']['attachment']); } else if ($msg_uid = $COMPOSE['param']['uid']) { $compose_mode = rcmail_sendmail::MODE_EDIT; } if ($compose_mode) { $COMPOSE['mode'] = $compose_mode; $OUTPUT->set_env('compose_mode', $compose_mode); } if ($compose_mode == rcmail_sendmail::MODE_EDIT || $compose_mode == rcmail_sendmail::MODE_DRAFT) { // don't add signature in draft/edit mode, we'll also not remove the old-one // but only on page display, later we should be able to change identity/sig (#1489229) if ($config_show_sig == 1 || $config_show_sig == 2) { $OUTPUT->set_env('show_sig_later', true); } } else if ($config_show_sig == 1) $OUTPUT->set_env('show_sig', true); else if ($config_show_sig == 2 && empty($compose_mode)) $OUTPUT->set_env('show_sig', true); else if ($config_show_sig == 3 && ($compose_mode == rcmail_sendmail::MODE_REPLY || $compose_mode == rcmail_sendmail::MODE_FORWARD)) $OUTPUT->set_env('show_sig', true); // set line length for body wrapping $LINE_LENGTH = $RCMAIL->config->get('line_length', 72); if (!empty($msg_uid) && empty($COMPOSE['as_attachment'])) { $mbox_name = $RCMAIL->storage->get_folder(); // set format before rcube_message construction // use the same format as for the message view if (isset($_SESSION['msg_formats'][$mbox_name.':'.$msg_uid])) { $RCMAIL->config->set('prefer_html', $_SESSION['msg_formats'][$mbox_name.':'.$msg_uid]); } else { $prefer_html = $RCMAIL->config->get('prefer_html') || $RCMAIL->config->get('htmleditor') || $compose_mode == rcmail_sendmail::MODE_DRAFT || $compose_mode == rcmail_sendmail::MODE_EDIT; $RCMAIL->config->set('prefer_html', $prefer_html); } $MESSAGE = new rcube_message($msg_uid); // make sure message is marked as read if ($MESSAGE->headers && $MESSAGE->context === null && empty($MESSAGE->headers->flags['SEEN'])) { $RCMAIL->storage->set_flag($msg_uid, 'SEEN'); } if (!empty($MESSAGE->headers->charset)) { $RCMAIL->storage->set_charset($MESSAGE->headers->charset); } if (!$MESSAGE->headers) { // error } else if ($compose_mode == rcmail_sendmail::MODE_FORWARD || $compose_mode == rcmail_sendmail::MODE_REPLY) { if ($compose_mode == rcmail_sendmail::MODE_REPLY) { $COMPOSE['reply_uid'] = $MESSAGE->context === null ? $msg_uid : null; if (!empty($COMPOSE['param']['all'])) { $MESSAGE->reply_all = $COMPOSE['param']['all']; } } else { $COMPOSE['forward_uid'] = $msg_uid; } $COMPOSE['reply_msgid'] = $MESSAGE->headers->messageID; $COMPOSE['references'] = trim($MESSAGE->headers->references . " " . $MESSAGE->headers->messageID); // Save the sent message in the same folder of the message being replied to if ($RCMAIL->config->get('reply_same_folder') && ($sent_folder = $COMPOSE['mailbox']) && rcmail_sendmail::check_sent_folder($sent_folder, false) ) { $COMPOSE['param']['sent_mbox'] = $sent_folder; } } else if ($compose_mode == rcmail_sendmail::MODE_DRAFT || $compose_mode == rcmail_sendmail::MODE_EDIT) { if ($compose_mode == rcmail_sendmail::MODE_DRAFT) { if ($draft_info = $MESSAGE->headers->get('x-draft-info')) { // get reply_uid/forward_uid to flag the original message when sending $info = rcmail_sendmail::draftinfo_decode($draft_info); if ($info['type'] == 'reply') $COMPOSE['reply_uid'] = $info['uid']; else if ($info['type'] == 'forward') $COMPOSE['forward_uid'] = $info['uid']; $COMPOSE['mailbox'] = $info['folder']; // Save the sent message in the same folder of the message being replied to if ($RCMAIL->config->get('reply_same_folder') && ($sent_folder = $info['folder']) && rcmail_sendmail::check_sent_folder($sent_folder, false) ) { $COMPOSE['param']['sent_mbox'] = $sent_folder; } } if (($msgid = $MESSAGE->headers->get('message-id')) && !preg_match('/^mid:[0-9]+$/', $msgid)) { $COMPOSE['param']['message-id'] = $msgid; } // use message UID as draft_id $OUTPUT->set_env('draft_id', $msg_uid); } if ($in_reply_to = $MESSAGE->headers->get('in-reply-to')) { $COMPOSE['reply_msgid'] = '<' . $in_reply_to . '>'; } $COMPOSE['references'] = $MESSAGE->headers->references; } } else { $MESSAGE = new stdClass(); // apply mailto: URL parameters if (!empty($COMPOSE['param']['in-reply-to'])) { $COMPOSE['reply_msgid'] = '<' . $COMPOSE['param']['in-reply-to'] . '>'; } if (!empty($COMPOSE['param']['references'])) { $COMPOSE['references'] = $COMPOSE['param']['references']; } } if (!empty($COMPOSE['reply_msgid'])) { $OUTPUT->set_env('reply_msgid', $COMPOSE['reply_msgid']); } // Initialize helper class to build the UI $SENDMAIL = new rcmail_sendmail($COMPOSE, array('message' => $MESSAGE)); // process $MESSAGE body/attachments, set $MESSAGE_BODY/$HTML_MODE vars and some session data $MESSAGE_BODY = rcmail_prepare_message_body(); // register UI objects (Note: some objects are registered by rcmail_sendmail above) $OUTPUT->add_handlers(array( 'composebody' => 'rcmail_compose_body', 'composeattachmentlist' => 'rcmail_compose_attachment_list', 'composeattachmentform' => 'rcmail_compose_attachment_form', 'composeattachment' => 'rcmail_compose_attachment_field', 'filedroparea' => 'rcmail_compose_file_drop_area', 'editorselector' => 'rcmail_editor_selector', 'addressbooks' => 'rcmail_addressbook_list', 'addresslist' => 'rcmail_contacts_list', 'responseslist' => 'rcmail_compose_responses_list', )); $OUTPUT->include_script('publickey.js'); rcmail_spellchecker_init(); $OUTPUT->send('compose'); /****** compose mode functions ********/ // process compose request parameters function rcmail_process_compose_params(&$COMPOSE) { if ($COMPOSE['param']['to']) { $mailto = explode('?', $COMPOSE['param']['to'], 2); // #1486037: remove "mailto:" prefix $COMPOSE['param']['to'] = preg_replace('/^mailto:/i', '', $mailto[0]); // #1490346: decode the recipient address // #1490510: use raw encoding for correct "+" character handling as specified in RFC6068 $COMPOSE['param']['to'] = rawurldecode($COMPOSE['param']['to']); // Supported case-insensitive tokens in mailto URL $url_tokens = array('to', 'cc', 'bcc', 'reply-to', 'in-reply-to', 'references', 'subject', 'body'); if (!empty($mailto[1])) { parse_str($mailto[1], $query); foreach ($query as $f => $val) { if (($key = array_search(strtolower($f), $url_tokens)) !== false) { $f = $url_tokens[$key]; } // merge mailto: addresses with addresses from 'to' parameter if ($f == 'to' && !empty($COMPOSE['param']['to'])) { $to_addresses = rcube_mime::decode_address_list($COMPOSE['param']['to'], null, true, null, true); $add_addresses = rcube_mime::decode_address_list($val, null, true); foreach ($add_addresses as $addr) { if (!in_array($addr['mailto'], $to_addresses)) { $to_addresses[] = $addr['mailto']; $COMPOSE['param']['to'] = (!empty($to_addresses) ? ', ' : '') . $addr['string']; } } } else { $COMPOSE['param'][$f] = $val; } } } } // resolve _forward_uid=* to an absolute list of messages from a search result if ($COMPOSE['param']['forward_uid'] == '*' && is_object($_SESSION['search'][1])) { $COMPOSE['param']['forward_uid'] = $_SESSION['search'][1]->get(); } // clean HTML message body which can be submitted by URL if (!empty($COMPOSE['param']['body'])) { if ($COMPOSE['param']['html'] = strpos($COMPOSE['param']['body'], '<') !== false) { $wash_params = array('safe' => false, 'inline_html' => true); $COMPOSE['param']['body'] = rcmail_wash_html($COMPOSE['param']['body'], $wash_params, array()); $COMPOSE['param']['body'] = preg_replace('//', // remove tags '/]*)>/i', '/<\/body>/i', // convert TinyMCE's empty-line sequence (#1490463) '/

\xC2\xA0<\/p>/', ), array( '', '', '', '


', ), $body ); // replace cid with href in inline images links if (!empty($cid_map)) { $body = str_replace(array_keys($cid_map), array_values($cid_map), $body); } } return $body; } // Removes signature from the message body function rcmail_remove_signature($body) { global $RCMAIL; $body = str_replace("\r\n", "\n", $body); $len = strlen($body); $sig_max_lines = $RCMAIL->config->get('sig_max_lines', 15); while (($sp = strrpos($body, "-- \n", $sp ? -$len+$sp-1 : 0)) !== false) { if ($sp == 0 || $body[$sp-1] == "\n") { // do not touch blocks with more that X lines if (substr_count($body, "\n", $sp) < $sig_max_lines) { $body = substr($body, 0, max(0, $sp-1)); } break; } } return $body; } function rcmail_write_compose_attachments(&$message, $bodyIsHtml) { global $RCMAIL, $COMPOSE; $loaded_attachments = array(); foreach ((array)$COMPOSE['attachments'] as $attachment) { $loaded_attachments[$attachment['name'] . $attachment['mimetype']] = $attachment; } $cid_map = array(); $messages = array(); if ($message->pgp_mime) { return $cid_map; } foreach ((array) $message->mime_parts() as $pid => $part) { if ($part->mimetype == 'message/rfc822') { $messages[] = $part->mime_id; } if ($part->disposition == 'attachment' || ($part->disposition == 'inline' && $bodyIsHtml) || $part->filename) { // skip parts that aren't valid attachments if ($part->ctype_primary == 'multipart' || $part->mimetype == 'application/ms-tnef') { continue; } // skip message attachments in reply mode if ($part->ctype_primary == 'message' && $COMPOSE['mode'] == rcmail_sendmail::MODE_REPLY) { continue; } // skip inline images when forwarding in text mode if ($part->content_id && $part->disposition == 'inline' && !$bodyIsHtml && $COMPOSE['mode'] == rcmail_sendmail::MODE_FORWARD) { continue; } // skip version.txt parts of multipart/encrypted messages if ($message->pgp_mime && $part->mimetype == 'application/pgp-encrypted' && $part->filename == 'version.txt') { continue; } // skip attachments included in message/rfc822 attachment (#1486487, #1490607) foreach ($messages as $mimeid) { if (strpos($part->mime_id, $mimeid . '.') === 0) { continue 2; } } if (($attachment = $loaded_attachments[rcmail_attachment_name($part) . $part->mimetype]) || ($attachment = rcmail_save_attachment($message, $pid, $COMPOSE['id'])) ) { if ($bodyIsHtml && ($part->content_id || $part->content_location)) { $url = sprintf('%s&_id=%s&_action=display-attachment&_file=rcmfile%s', $RCMAIL->comm_path, $COMPOSE['id'], $attachment['id']); if ($part->content_id) $cid_map['cid:'.$part->content_id] = $url; else $cid_map[$part->content_location] = $url; } } } } $COMPOSE['forward_attachments'] = true; return $cid_map; } function rcmail_write_inline_attachments(&$message) { global $RCMAIL, $COMPOSE; $cid_map = array(); $messages = array(); if ($message->pgp_mime) { return $cid_map; } foreach ((array) $message->mime_parts() as $pid => $part) { if ($part->mimetype == 'message/rfc822') { $messages[] = $part->mime_id; } if (($part->content_id || $part->content_location) && $part->filename) { // skip attachments included in message/rfc822 attachment (#1486487, #1490607) foreach ($messages as $mimeid) { if (strpos($part->mime_id, $mimeid . '.') === 0) { continue 2; } } if ($attachment = rcmail_save_attachment($message, $pid, $COMPOSE['id'])) { $url = sprintf('%s&_id=%s&_action=display-attachment&_file=rcmfile%s', $RCMAIL->comm_path, $COMPOSE['id'], $attachment['id']); if ($part->content_id) $cid_map['cid:'.$part->content_id] = $url; else $cid_map[$part->content_location] = $url; } } } return $cid_map; } // Creates attachment(s) from the forwarded message(s) function rcmail_write_forward_attachments() { global $RCMAIL, $COMPOSE, $MESSAGE; if ($MESSAGE->pgp_mime) { return; } $storage = $RCMAIL->get_storage(); $names = array(); $refs = array(); $size_errors = 0; $size_limit = parse_bytes($RCMAIL->config->get('max_message_size')); $total_size = 10 * 1024; // size of message body, to start with $loaded_attachments = array(); foreach ((array)$COMPOSE['attachments'] as $attachment) { $loaded_attachments[$attachment['name'] . $attachment['mimetype']] = $attachment; $total_size += $attachment['size']; } if ($COMPOSE['forward_uid'] == '*') { $index = $storage->index(null, rcmail_sort_column(), rcmail_sort_order()); $COMPOSE['forward_uid'] = $index->get(); } else if (!is_array($COMPOSE['forward_uid']) && strpos($COMPOSE['forward_uid'], ':')) { $COMPOSE['forward_uid'] = rcube_imap_generic::uncompressMessageSet($COMPOSE['forward_uid']); } else if (is_string($COMPOSE['forward_uid'])) { $COMPOSE['forward_uid'] = explode(',', $COMPOSE['forward_uid']); } foreach ((array)$COMPOSE['forward_uid'] as $uid) { $message = new rcube_message($uid); if (empty($message->headers)) { continue; } if (!empty($message->headers->charset)) { $storage->set_charset($message->headers->charset); } if (empty($MESSAGE->subject)) { $MESSAGE->subject = $message->subject; } // generate (unique) attachment name $name = strlen($message->subject) ? mb_substr($message->subject, 0, 64) : 'message_rfc822'; if (!empty($names[$name])) { $names[$name]++; $name .= '_' . $names[$name]; } $names[$name] = 1; $name .= '.eml'; if (!empty($loaded_attachments[$name . 'message/rfc822'])) { continue; } if ($size_limit && $size_limit < $total_size + $message->headers->size) { $size_errors++; continue; } $total_size += $message->headers->size; rcmail_save_attachment($message, null, $COMPOSE['id'], array('filename' => $name)); if ($message->headers->messageID) { $refs[] = $message->headers->messageID; } } // set In-Reply-To and References headers if (count($refs) == 1) { $COMPOSE['reply_msgid'] = $refs[0]; } if (!empty($refs)) { $COMPOSE['references'] = implode(' ', $refs); } if ($size_errors) { $limit = $RCMAIL->show_bytes($size_limit); $error = $RCMAIL->gettext(array('name' => 'msgsizeerrorfwd', 'vars' => array('num' => $size_errors, 'size' => $limit))); $RCMAIL->output->add_script(sprintf("%s.display_message('%s', 'error');", rcmail_output::JS_OBJECT_NAME, rcube::JQ($error)), 'docready'); } } // Saves an image as attachment function rcmail_save_image($path, $mimetype = '', $data = null) { global $COMPOSE; // handle attachments in memory if (empty($data)) { $data = file_get_contents($path); $is_file = true; } $name = rcmail_basename($path); if (empty($mimetype)) { if ($is_file) { $mimetype = rcube_mime::file_content_type($path, $name); } else { $mimetype = rcube_mime::file_content_type($data, $name, 'application/octet-stream', true); } } $attachment = array( 'group' => $COMPOSE['id'], 'name' => $name, 'mimetype' => $mimetype, 'data' => $data, 'size' => strlen($data), ); $attachment = rcmail::get_instance()->plugins->exec_hook('attachment_save', $attachment); if ($attachment['status']) { unset($attachment['data'], $attachment['status'], $attachment['content_id'], $attachment['abort']); return $attachment; } return false; } // Unicode-safe basename() function rcmail_basename($filename) { // basename() is not unicode safe and locale dependent if (stristr(PHP_OS, 'win') || stristr(PHP_OS, 'netware')) { return preg_replace('/^.*[\\\\\\/]/', '', $filename); } else { return preg_replace('/^.*[\/]/', '', $filename); } } /** * Attachments list object for templates */ function rcmail_compose_attachment_list($attrib) { global $RCMAIL, $OUTPUT, $COMPOSE; // add ID if not given if (!$attrib['id']) $attrib['id'] = 'rcmAttachmentList'; $out = "\n"; $jslist = array(); $button = ''; if ($attrib['icon_pos'] == 'left') $COMPOSE['icon_pos'] = 'left'; if (is_array($COMPOSE['attachments'])) { if ($attrib['deleteicon']) { $button = html::img(array( 'src' => $RCMAIL->output->asset_url($attrib['deleteicon'], true), 'alt' => $RCMAIL->gettext('delete') )); } else if (rcube_utils::get_boolean($attrib['textbuttons'])) { $button = rcube::Q($RCMAIL->gettext('delete')); } foreach ($COMPOSE['attachments'] as $id => $a_prop) { if (empty($a_prop)) { continue; } $link_content = sprintf('%s (%s)', rcube::Q($a_prop['name']), $RCMAIL->show_bytes($a_prop['size'])); $content_link = html::a(array( 'href' => "#load", 'class' => 'filename', 'onclick' => sprintf("return %s.command('load-attachment','rcmfile%s', this, event)", rcmail_output::JS_OBJECT_NAME, $id), ), $link_content); $delete_link = html::a(array( 'href' => "#delete", 'title' => $RCMAIL->gettext('delete'), 'onclick' => sprintf("return %s.command('remove-attachment','rcmfile%s', this, event)", rcmail_output::JS_OBJECT_NAME, $id), 'class' => 'delete', 'tabindex' => $attrib['tabindex'] ?: '0', 'aria-label' => $RCMAIL->gettext('delete') . ' ' . $a_prop['name'], ), $button); $out .= html::tag('li', array( 'id' => 'rcmfile'.$id, 'class' => rcube_utils::file2class($a_prop['mimetype'], $a_prop['name']), ), $COMPOSE['icon_pos'] == 'left' ? $delete_link.$content_link : $content_link.$delete_link ); $jslist['rcmfile'.$id] = array( 'name' => $a_prop['name'], 'complete' => true, 'mimetype' => $a_prop['mimetype'] ); } } if ($attrib['deleteicon']) $COMPOSE['deleteicon'] = $RCMAIL->output->asset_url($attrib['deleteicon'], true); else if (rcube_utils::get_boolean($attrib['textbuttons'])) $COMPOSE['textbuttons'] = true; if ($attrib['cancelicon']) $OUTPUT->set_env('cancelicon', $RCMAIL->output->asset_url($attrib['cancelicon'], true)); if ($attrib['loadingicon']) $OUTPUT->set_env('loadingicon', $RCMAIL->output->asset_url($attrib['loadingicon'], true)); $OUTPUT->set_env('attachments', $jslist); $OUTPUT->add_gui_object('attachmentlist', $attrib['id']); // put tabindex value into data-tabindex attribute if (isset($attrib['tabindex'])) { $attrib['data-tabindex'] = $attrib['tabindex']; unset($attrib['tabindex']); } return html::tag('ul', $attrib, $out, html::$common_attrib); } /** * Attachment upload form object for templates */ function rcmail_compose_attachment_form($attrib) { global $RCMAIL; return $RCMAIL->upload_form($attrib, 'uploadform', 'send-attachment', array('multiple' => true)); } /** * Register a certain container as active area to drop files onto */ function rcmail_compose_file_drop_area($attrib) { global $OUTPUT; if ($attrib['id']) { $OUTPUT->add_gui_object('filedrop', $attrib['id']); $OUTPUT->set_env('filedrop', array('action' => 'upload', 'fieldname' => '_attachments')); } } /** * Editor mode selector object for templates */ function rcmail_editor_selector($attrib) { global $RCMAIL; // determine whether HTML or plain text should be checked $useHtml = rcmail_compose_editor_mode(); if (empty($attrib['editorid'])) $attrib['editorid'] = 'rcmComposeBody'; if (empty($attrib['name'])) $attrib['name'] = 'editorSelect'; $attrib['onchange'] = "return rcmail.command('toggle-editor', {id: '".$attrib['editorid']."', html: this.value == 'html'}, '', event)"; $select = new html_select($attrib); $select->add(rcube::Q($RCMAIL->gettext('htmltoggle')), 'html'); $select->add(rcube::Q($RCMAIL->gettext('plaintoggle')), 'plain'); return $select->show($useHtml ? 'html' : 'plain'); } /** * Addressbooks list object for templates */ function rcmail_addressbook_list($attrib = array()) { global $RCMAIL, $OUTPUT; $attrib += array('id' => 'rcmdirectorylist'); $out = ''; $line_templ = html::tag('li', array( 'id' => 'rcmli%s', 'class' => '%s'), html::a(array('href' => '#list', 'rel' => '%s', 'onclick' => "return ".rcmail_output::JS_OBJECT_NAME.".command('list-addresses','%s',this)"), '%s')); foreach ($RCMAIL->get_address_sources(false, true) as $j => $source) { $id = strval(strlen($source['id']) ? $source['id'] : $j); $js_id = rcube::JQ($id); // set class name(s) $class_name = 'addressbook'; if ($source['class_name']) $class_name .= ' ' . $source['class_name']; $out .= sprintf($line_templ, rcube_utils::html_identifier($id,true), $class_name, $source['id'], $js_id, ($source['name'] ?: $id)); } $OUTPUT->add_gui_object('addressbookslist', $attrib['id']); return html::tag('ul', $attrib, $out, html::$common_attrib); } /** * Contacts list object for templates */ function rcmail_contacts_list($attrib = array()) { global $RCMAIL, $OUTPUT; $attrib += array('id' => 'rcmAddressList'); // set client env $OUTPUT->add_gui_object('contactslist', $attrib['id']); $OUTPUT->set_env('pagecount', 0); $OUTPUT->set_env('current_page', 0); $OUTPUT->include_script('list.js'); return $RCMAIL->table_output($attrib, array(), array('name'), 'ID'); } /** * Responses list object for templates */ function rcmail_compose_responses_list($attrib) { global $RCMAIL, $OUTPUT; $attrib += array('id' => 'rcmresponseslist', 'tagname' => 'ul', 'cols' => 1); $jsenv = array(); $list = new html_table($attrib); foreach ($RCMAIL->get_compose_responses(true) as $response) { $key = $response['key']; $item = html::a(array( 'href' => '#' . urlencode($response['name']), 'class' => rtrim('insertresponse ' . $attrib['itemclass']), 'unselectable' => 'on', 'tabindex' => '0', 'rel' => $key, ), rcube::Q($response['name'])); $jsenv[$key] = $response; $list->add(array(), $item); } // set client env $OUTPUT->set_env('textresponses', $jsenv); $OUTPUT->add_gui_object('responseslist', $attrib['id']); return $list->show(); }