Source for file compose.php

Documentation is available at compose.php

  1. <?php
  2. /**
  3.  * compose.php
  4.  *
  5.  * This code sends a mail.
  6.  *
  7.  * There are 4 modes of operation:
  8.  *    - Start new mail
  9.  *    - Add an attachment
  10.  *    - Send mail
  11.  *    - Save As Draft
  12.  *
  13.  * @copyright 1999-2014 The SquirrelMail Project Team
  14.  * @license http://opensource.org/licenses/gpl-license.php GNU Public License
  15.  * @version $Id: compose.php 14437 2014-01-21 22:15:03Z pdontthink $
  16.  * @package squirrelmail
  17.  */
  18.  
  19. /** This is the compose page */
  20. define('PAGE_NAME''compose');
  21.  
  22. /**
  23.  * Include the SquirrelMail initialization file.
  24.  */
  25. require('../include/init.php');
  26.  
  27. /* If email_address not set and admin wants us to ask user for it,
  28.  * redirect to options page. */
  29. if $ask_user_info && getPref($data_dir$username,'email_address'== "" {
  30.     header("Location: " get_location("/options.php?optpage=personal");
  31.     exit;
  32. }
  33.  
  34. /* SquirrelMail required files. */
  35. require_once(SM_PATH 'functions/imap_general.php');
  36. require_once(SM_PATH 'functions/imap_messages.php');
  37. require_once(SM_PATH 'functions/date.php');
  38. require_once(SM_PATH 'functions/mime.php');
  39. require_once(SM_PATH 'functions/compose.php');
  40. require_once(SM_PATH 'class/deliver/Deliver.class.php');
  41. require_once(SM_PATH 'functions/addressbook.php');
  42. require_once(SM_PATH 'functions/forms.php');
  43. require_once(SM_PATH 'functions/identity.php');
  44. global $imap_stream_options// in case not defined in config
  45.  
  46. /* --------------------- Get globals ------------------------------------- */
  47.  
  48. /** SESSION VARS */
  49. sqgetGlobalVar('delimiter'$delimiter,     SQ_SESSION);
  50.  
  51. sqgetGlobalVar('delayed_errors',  $delayed_errors,  SQ_SESSION);
  52. sqgetGlobalVar('composesession',    $composesession,    SQ_SESSION);
  53. sqgetGlobalVar('compose_messages',  $compose_messages,  SQ_SESSION);
  54.  
  55. // compose_messages only useful in SESSION when a forward-as-attachment
  56. // has been preconstructed for us and passed in via that mechanism; once
  57. // we have it, we can clear it from the SESSION
  58. sqsession_unregister('compose_messages');
  59.  
  60. // Turn on delayed error handling in case we wind up redirecting below
  61. $oErrorHandler->setDelayedErrors(true);
  62.  
  63. /** SESSION/POST/GET VARS */
  64. sqgetGlobalVar('send_button_count'$send_button_countSQ_POST1SQ_TYPE_INT);
  65. for ($i 1$i <= $send_button_count$i++)
  66.    if (sqgetGlobalVar('send' $i$sendSQ_POST)) break;
  67. // Send can only be achieved by setting $_POST var. If Send = true then
  68. // retrieve other form fields from $_POST
  69. if (isset($send&& $send{
  70.     $SQ_GLOBAL SQ_POST;
  71. else {
  72.     $SQ_GLOBAL SQ_FORM;
  73. }
  74. sqgetGlobalVar('session',$session$SQ_GLOBAL);
  75. sqgetGlobalVar('mailbox',$mailbox$SQ_GLOBAL);
  76. if(!sqgetGlobalVar('identity',$identity$SQ_GLOBAL)) {
  77.     $identity=0;
  78. }
  79. sqgetGlobalVar('send_to',$send_to$SQ_GLOBAL);
  80. sqgetGlobalVar('send_to_cc',$send_to_cc$SQ_GLOBAL);
  81. sqgetGlobalVar('send_to_bcc',$send_to_bcc$SQ_GLOBAL);
  82. sqgetGlobalVar('subject',$subject$SQ_GLOBAL);
  83. sqgetGlobalVar('body',$body$SQ_GLOBAL);
  84. sqgetGlobalVar('mailprio',$mailprio$SQ_GLOBAL);
  85. sqgetGlobalVar('request_mdn',$request_mdn$SQ_GLOBAL);
  86. sqgetGlobalVar('request_dr',$request_dr$SQ_GLOBAL);
  87. sqgetGlobalVar('html_addr_search',$html_addr_search$SQ_GLOBAL);
  88. sqgetGlobalVar('mail_sent',$mail_sent$SQ_GLOBAL);
  89. sqgetGlobalVar('passed_id',$passed_id$SQ_GLOBALNULLSQ_TYPE_BIGINT);
  90. sqgetGlobalVar('passed_ent_id',$passed_ent_id$SQ_GLOBAL);
  91. sqgetGlobalVar('fwduid',$fwduid$SQ_GLOBAL'');
  92.  
  93. sqgetGlobalVar('attach',$attachSQ_POST);
  94. sqgetGlobalVar('draft',$draftSQ_POST);
  95. sqgetGlobalVar('draft_id',$draft_id$SQ_GLOBAL);
  96. sqgetGlobalVar('ent_num',$ent_num$SQ_GLOBAL);
  97. sqgetGlobalVar('saved_draft',$saved_draftSQ_FORM);
  98.  
  99. if sqgetGlobalVar('delete_draft',$delete_draft) ) {
  100.     $delete_draft = (int)$delete_draft;
  101. }
  102.  
  103. if sqgetGlobalVar('startMessage',$startMessage) ) {
  104.     $startMessage = (int)$startMessage;
  105. else {
  106.     $startMessage 1;
  107. }
  108.  
  109.  
  110. /** POST VARS */
  111. sqgetGlobalVar('sigappend',             $sigappend,                 SQ_POST);
  112. sqgetGlobalVar('from_htmladdr_search',  $from_htmladdr_search,      SQ_POST);
  113. sqgetGlobalVar('addr_search_done',      $html_addr_search_done,     SQ_POST);
  114. sqgetGlobalVar('addr_search_cancel',    $html_addr_search_cancel,   SQ_POST);
  115. sqgetGlobalVar('send_to_search',        $send_to_search,            SQ_POST);
  116. sqgetGlobalVar('do_delete',             $do_delete,                 SQ_POST);
  117. sqgetGlobalVar('delete',                $delete,                    SQ_POST);
  118. sqgetGlobalVar('attachments',           $attachments,               SQ_POST);
  119. if sqgetGlobalVar('return'$tempSQ_POST) ) {
  120.     $html_addr_search_done 'Use Addresses';
  121. }
  122.  
  123. /** GET VARS */
  124. if sqgetGlobalVar('account'$temp,  SQ_GET) ) {
  125.     $iAccount = (int) $temp;
  126. else {
  127.     $iAccount 0;
  128. }
  129.  
  130.  
  131. /** get smaction */
  132. if !sqgetGlobalVar('smaction',$action) )
  133. {
  134.     if sqgetGlobalVar('smaction_reply',$tmp) )      $action 'reply';
  135.     if sqgetGlobalVar('smaction_reply_all',$tmp) )  $action 'reply_all';
  136.     if sqgetGlobalVar('smaction_forward',$tmp) )    $action 'forward';
  137.     if sqgetGlobalVar('smaction_attache',$tmp) )    $action 'forward_as_attachment';
  138.     if sqgetGlobalVar('smaction_draft',$tmp) )      $action 'draft';
  139.     if sqgetGlobalVar('smaction_edit_new',$tmp) )   $action 'edit_as_new';
  140. }
  141.  
  142. sqgetGlobalVar('smtoken'$submitted_token$SQ_GLOBAL'');
  143.  
  144. /**
  145.  * Here we decode the data passed in from mailto.php.
  146.  */
  147. if sqgetGlobalVar('mailtodata'$mailtodataSQ_GET) ) {
  148.     $trtable array('to'       => 'send_to',
  149.                  'cc'           => 'send_to_cc',
  150.                  'bcc'          => 'send_to_bcc',
  151.                  'body'         => 'body',
  152.                  'subject'      => 'subject');
  153.     $mtdata unserialize($mailtodata);
  154.  
  155.     foreach ($trtable as $f => $t{
  156.         if !empty($mtdata[$f]) ) {
  157.             $$t $mtdata[$f];
  158.         }
  159.     }
  160.     unset($mailtodata,$mtdata$trtable);
  161. }
  162.  
  163. /* Location (For HTTP 1.1 header("Location: ...") redirects) */
  164. $location get_location();
  165. /* Identities (fetch only once) */
  166. $idents get_identities();
  167.  
  168. /* --------------------- Specific Functions ------------------------------ */
  169.  
  170. function replyAllString($header{
  171.     global $include_self_reply_all$idents;
  172.     $excl_ar array();
  173.     /**
  174.      * 1) Remove the addresses we'll be sending the message 'to'
  175.      */
  176.     if (isset($header->reply_to)) {
  177.         $excl_ar $header->getAddr_a('reply_to');
  178.     }
  179.     /**
  180.      * 2) Remove our identities from the CC list (they still can be in the
  181.      * TO list) only if $include_self_reply_all is turned off
  182.      */
  183.     if (!$include_self_reply_all{
  184.         foreach($idents as $id{
  185.             $excl_ar[strtolower(trim($id['email_address']))'';
  186.         }
  187.     }
  188.  
  189.     /**
  190.      * 3) get the addresses.
  191.      */
  192.     $url_replytoall_ar $header->getAddr_a(array('to','cc')$excl_ar);
  193.  
  194.     /**
  195.      * 4) generate the string.
  196.      */
  197.     $url_replytoallcc '';
  198.     foreach$url_replytoall_ar as $email => $personal{
  199.         if ($personal{
  200.             // always quote personal name (can't just quote it if
  201.             // it contains a comma separator, since it might still
  202.             // be encoded)
  203.             $url_replytoallcc .= ", \"$personal\" <$email>";
  204.         else {
  205.             $url_replytoallcc .= ', '$email;
  206.         }
  207.     }
  208.     $url_replytoallcc substr($url_replytoallcc,2);
  209.  
  210.     return $url_replytoallcc;
  211. }
  212.  
  213. /**
  214.  * creates top line in reply citations
  215.  *
  216.  * Line style depends on user preferences.
  217.  * $orig_date argument is available only from 1.4.3 and 1.5.1 version.
  218.  * @param object $orig_from From: header object.
  219.  * @param integer $orig_date email's timestamp
  220.  * @return string reply citation
  221.  */
  222. function getReplyCitation($orig_from$orig_date{
  223.     global $reply_citation_style$reply_citation_start$reply_citation_end;
  224.  
  225.     if (!is_object($orig_from)) {
  226.         $sOrig_from '';
  227.     else {
  228.         $sOrig_from decodeHeader($orig_from->getAddress(false),false,false,true);
  229.     }
  230.  
  231.     /* First, return an empty string when no citation style selected. */
  232.     if (($reply_citation_style == ''|| ($reply_citation_style == 'none')) {
  233.         return '';
  234.     }
  235.  
  236.     /* Make sure our final value isn't an empty string. */
  237.     if ($sOrig_from == ''{
  238.         return '';
  239.     }
  240.  
  241.     /* Otherwise, try to select the desired citation style. */
  242.     switch ($reply_citation_style{
  243.     case 'author_said':
  244.         // i18n: %s is for author's name
  245.         $full_reply_citation sprintf(_("%s wrote:"),$sOrig_from);
  246.         break;
  247.     case 'quote_who':
  248.         $start '<quote who="';
  249.         $end   '">';
  250.         $full_reply_citation $start $sOrig_from $end;
  251.         break;
  252.     case 'date_time_author':
  253.         // i18n:
  254.         // The first %s is for date string, the second %s is for author's name.
  255.         // The date uses formating from "D, F j, Y g:i a" and "D, F j, Y H:i"
  256.         // translations.
  257.         // Example string:
  258.         // "On Sat, December 24, 2004 23:59, Santa wrote:"
  259.         // If you have to put author's name in front of date string, check comments about
  260.         // argument swapping at http://php.net/sprintf
  261.         $full_reply_citation sprintf(_("On %s, %s wrote:")getLongDateString($orig_date)$sOrig_from);
  262.         break;
  263.     case 'user-defined':
  264.         $start $reply_citation_start .
  265.             ($reply_citation_start == '' '' ' ');
  266.         $end   $reply_citation_end;
  267.         $full_reply_citation $start $sOrig_from $end;
  268.         break;
  269.     default:
  270.         return '';
  271.     }
  272.  
  273.     /* Add line feed and return the citation string. */
  274.     return ($full_reply_citation "\n");
  275. }
  276.  
  277. /**
  278.  * Creates header fields in forwarded email body
  279.  *
  280.  * $default_charset global must be set correctly before you call this function.
  281.  * @param object $orig_header 
  282.  * @return $string 
  283.  */
  284. function getforwardHeader($orig_header{
  285.     global $editor_size$default_charset;
  286.  
  287.     // using own strlen function in order to detect correct string length
  288.     $display array_("Subject"=> sq_strlen(_("Subject"),$default_charset),
  289.             _("From")    => sq_strlen(_("From"),$default_charset),
  290.             _("Date")    => sq_strlen(_("Date"),$default_charset),
  291.             _("To")      => sq_strlen(_("To"),$default_charset),
  292.             _("Cc")      => sq_strlen(_("Cc"),$default_charset) );
  293.     $maxsize max($display);
  294.     $indent str_pad('',$maxsize+2);
  295.     foreach($display as $key => $val{
  296.         $display[$key$key .': 'str_pad(''$maxsize $val);
  297.     }
  298.     $from decodeHeader($orig_header->getAddr_s('from',"\n$indent"),false,false,true);
  299.     $from str_replace('&nbsp;',' ',$from);
  300.     $to decodeHeader($orig_header->getAddr_s('to',"\n$indent"),false,false,true);
  301.     $to str_replace('&nbsp;',' ',$to);
  302.     $subject decodeHeader($orig_header->subject,false,false,true);
  303.     $subject str_replace('&nbsp;',' ',$subject);
  304.  
  305.     // using own str_pad function in order to create correct string pad
  306.     $bodyTop =  sq_str_pad(' '._("Original Message").' ',$editor_size -2,'-',STR_PAD_BOTH,$default_charset.
  307.         "\n"$display[_("Subject")$subject "\n" .
  308.         $display[_("From")$from "\n" .
  309.         $display[_("Date")getLongDateString$orig_header->date$orig_header->date_unparsed )"\n" .
  310.         $display[_("To")$to "\n";
  311.     if ($orig_header->cc != array(&& $orig_header->cc !=''{
  312.         $cc decodeHeader($orig_header->getAddr_s('cc',"\n$indent"),false,false,true);
  313.         $cc str_replace('&nbsp;',' ',$cc);
  314.         $bodyTop .= $display[_("Cc").$cc "\n";
  315.     }
  316.     $bodyTop .= str_pad(''$editor_size -'-'.
  317.         "\n\n";
  318.     return $bodyTop;
  319. }
  320. /* ----------------------------------------------------------------------- */
  321.  
  322. /*
  323.  * If the session is expired during a post this restores the compose session
  324.  * vars.
  325.  */
  326. $session_expired false;
  327. if (sqsession_is_registered('session_expired_post')) {
  328.     sqgetGlobalVar('session_expired_post'$session_expired_postSQ_SESSION);
  329.     /*
  330.      * extra check for username so we don't display previous post data from
  331.      * another user during this session.
  332.      */
  333.     if (!empty($session_expired_post['username']
  334.      && $session_expired_post['username'== $username{
  335.         // these are the vars that we can set from the expired composed session
  336.         $compo_var_list array ('send_to''send_to_cc''body',
  337.             'startMessage''passed_body''use_signature''signature',
  338.             'subject''newmail''send_to_bcc''passed_id''mailbox'
  339.             'from_htmladdr_search''identity''draft_id''delete_draft'
  340.             'mailprio''edit_as_new''attachments''composesession'
  341.             'request_mdn''request_dr''fwduid');
  342.  
  343.         foreach ($compo_var_list as $var{
  344.             if isset($session_expired_post[$var]&& !isset($$var) ) {
  345.                 $$var $session_expired_post[$var];
  346.             }
  347.         }
  348.  
  349.         if (!empty($attachments))
  350.             $attachments unserialize(urldecode($attachments));
  351.  
  352.         sqsession_register($composesession,'composesession');
  353.  
  354.         if (isset($send)) {
  355.             unset($send);
  356.         }
  357.         $session_expired true;
  358.     }
  359.     unset($session_expired_post);
  360.     sqsession_unregister('session_expired_post');
  361.     if (!isset($mailbox)) {
  362.         $mailbox '';
  363.     }
  364.     if ($compose_new_win == '1'{
  365.         compose_Header($color$mailbox);
  366.     else {
  367.         $sHeaderJs (isset($sHeaderJs)) $sHeaderJs '';
  368.         if (strpos($action'reply'!== false && $reply_focus{
  369.             $sOnload 'checkForm(\''.$replyfocus.'\');';
  370.         else {
  371.             $sOnload 'checkForm();';
  372.         }
  373.         displayPageHeader($color$mailbox,$sHeaderJs,$sOnload);
  374.     }
  375.     showInputForm($sessionfalse);
  376.     exit();
  377. }
  378.  
  379. if (!isset($composesession)) {
  380.     $composesession 0;
  381.     sqsession_register(0,'composesession');
  382. else {
  383.     $composesession = (int)$composesession;
  384. }
  385.  
  386. if (!isset($session|| (isset($newmessage&& $newmessage)) {
  387.     sqsession_unregister('composesession');
  388.     $session "$composesession+1;
  389.     $composesession $session;
  390.     sqsession_register($composesession,'composesession');
  391. }
  392. if (!empty($compose_messages[$session])) {
  393.     $composeMessage $compose_messages[$session];
  394. else {
  395.     $composeMessage new Message();
  396.     $rfc822_header new Rfc822Header();
  397.     $composeMessage->rfc822_header $rfc822_header;
  398.     $composeMessage->reply_rfc822_header '';
  399. }
  400.  
  401. // re-add attachments that were already in this message
  402. // FIXME: note that technically this is very bad form -
  403. // should never directly manipulate an object like this
  404. if (!empty($attachments)) {
  405.     $attachments unserialize(urldecode($attachments));
  406.     if (!empty($attachments&& is_array($attachments))
  407.         $composeMessage->entities $attachments;
  408. }
  409.  
  410. if (empty($mailbox)) {
  411.     $mailbox 'INBOX';
  412. }
  413.  
  414. if ($draft{
  415.  
  416.     // validate security token
  417.     //
  418.     sm_validate_security_token($submitted_token-1TRUE);
  419.  
  420.     /*
  421.      * Set $default_charset to correspond with the user's selection
  422.      * of language interface.
  423.      */
  424.     set_my_charset();
  425.     if (deliverMessage($composeMessagetrue)) {
  426.         showInputForm($session);
  427.         exit();
  428.     else {
  429.         $draft_message _("Draft Email Saved");
  430.         /* If this is a resumed draft, then delete the original */
  431.         if(isset($delete_draft)) {
  432.             $imap_stream sqimap_login($usernamefalse$imapServerAddress$imapPortfalse$imap_stream_options);
  433.             sqimap_mailbox_select($imap_stream$draft_folder);
  434.             // force bypass_trash=true because message should be saved when deliverMessage() returns true.
  435.             // in current implementation of sqimap_msgs_list_flag() single message id can
  436.             // be submitted as string. docs state that it should be array.
  437.             sqimap_msgs_list_delete($imap_stream$draft_folder$delete_drafttrue);
  438.             if ($auto_expunge{
  439.                 sqimap_mailbox_expunge($imap_stream$draft_foldertrue);
  440.             }
  441.             sqimap_logout($imap_stream);
  442.         }
  443.  
  444.         $oErrorHandler->saveDelayedErrors();
  445.         session_write_close();
  446.  
  447.         if ($compose_new_win == '1'{
  448.             if !isset($pageheader_sent|| !$pageheader_sent {
  449.                 header("Location: $location/compose.php?saved_draft=yes&session=$composesession");
  450.             else {
  451. //FIXME: DON'T ECHO HTML FROM CORE!
  452.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  453.                     . '/compose.php?saved_sent=yes&amp;session=' $composesession '">'
  454.                     . _("Return"'</a></div>';
  455.             }
  456.             exit();
  457.         else {
  458.             if !isset($pageheader_sent|| !$pageheader_sent {
  459.                 header("Location: $location/right_main.php?mailbox=urlencode($draft_folder.
  460.                    "&startMessage=1&note=".urlencode($draft_message));
  461.             else {
  462. //FIXME: DON'T ECHO HTML FROM CORE!
  463.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  464.                     . '/right_main.php?mailbox=' urlencode($draft_folder)
  465.                     . '&amp;startMessage=1&amp;note=' urlencode($draft_message.'">'
  466.                     . _("Return"'</a></div>';
  467.             }
  468.             exit();
  469.         }
  470.     }
  471. }
  472.  
  473. if ($send{
  474.  
  475.     // validate security token
  476.     //
  477.     sm_validate_security_token($submitted_token-1TRUE);
  478.  
  479.     if (isset($_FILES['attachfile']&&
  480.             $_FILES['attachfile']['tmp_name'&&
  481.             $_FILES['attachfile']['tmp_name'!= 'none'{
  482.         $AttachFailure saveAttachedFiles($session);
  483.     }
  484.     
  485.     if (checkInput(false&& !isset($AttachFailure)) {
  486.         if ($mailbox == "All Folders"{
  487.             /* We entered compose via the search results page */
  488.             $mailbox 'INBOX'/* Send 'em to INBOX, that's safe enough */
  489.         }
  490.         $urlMailbox urlencode($mailbox);
  491.         if (isset($passed_id)) {
  492.             $passed_id 0;
  493.         }
  494.         /**
  495.          * Set $default_charset to correspond with the user's selection
  496.          * of language interface.
  497.          */
  498.         set_my_charset();
  499.         /**
  500.          * This is to change all newlines to \n
  501.          * We'll change them to \r\n later (in the sendMessage function)
  502.          */
  503.         $body str_replace("\r\n""\n"$body);
  504.         $body str_replace("\r""\n"$body);
  505.  
  506.         /**
  507.          * Rewrap $body so that no line is bigger than $editor_size
  508.          */
  509.         $body explode("\n"$body);
  510.         $newBody '';
  511.         foreach ($body as $line{
  512.             if$line <> '-- ' {
  513.                 $line rtrim($line);
  514.             }
  515.             if (sq_strlen($line$default_charset<= $editor_size 1{
  516.                 $newBody .= $line "\n";
  517.             else {
  518.                 sqWordWrap($line$editor_size$default_charset);
  519.                 $newBody .= $line "\n";
  520.  
  521.             }
  522.  
  523.         }
  524.         $body $newBody;
  525.  
  526.         $Result deliverMessage($composeMessage);
  527.  
  528.         if ($Result)
  529.             $mail_sent 'yes';
  530.         else
  531.             $mail_sent 'no';
  532.  
  533.         // NOTE: this hook changed in 1.5.2 from sending $Result and
  534.         //       $composeMessage as args #2 and #3 to being in an array
  535.         //       under arg #2
  536.         $temp array(&$Result&$composeMessage&$mail_sent);
  537.         do_hook('compose_send_after'$temp);
  538.         if ($Result{
  539.             showInputForm($session);
  540.             exit();
  541.         }
  542.  
  543.         /* if it is resumed draft, delete draft message */
  544.         if isset($delete_draft)) {
  545.             $imap_stream sqimap_login($usernamefalse$imapServerAddress$imapPortfalse$imap_stream_options);
  546.             sqimap_mailbox_select($imap_stream$draft_folder);
  547.             // bypass_trash=true because message should be saved when deliverMessage() returns true.
  548.             // in current implementation of sqimap_msgs_list_flag() single message id can
  549.             // be submitted as string. docs state that it should be array.
  550.             sqimap_msgs_list_delete($imap_stream$draft_folder$delete_drafttrue);
  551.             if ($auto_expunge{
  552.                 sqimap_mailbox_expunge($imap_stream$draft_foldertrue);
  553.             }
  554.             sqimap_logout($imap_stream);
  555.         }
  556.         /*
  557.          * Store the error array in the session because they will be lost on a redirect
  558.          */
  559.         $oErrorHandler->saveDelayedErrors();
  560.         session_write_close();
  561.  
  562.         if ($compose_new_win == '1'{
  563.             if !isset($pageheader_sent|| !$pageheader_sent {
  564.                 header("Location: $location/compose.php?mail_sent=$mail_sent");
  565.             else {
  566. //FIXME: DON'T ECHO HTML FROM CORE!
  567.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  568.                     . '/compose.php?mail_sent=$mail_sent">'
  569.                     . _("Return"'</a></div>';
  570.             }
  571.             exit();
  572.         else {
  573.             if !isset($pageheader_sent|| !$pageheader_sent {
  574.                 header("Location: $location/right_main.php?mailbox=$urlMailbox".
  575.                     "&startMessage=$startMessage&mail_sent=$mail_sent");
  576.             else {
  577. //FIXME: DON'T ECHO HTML FROM CORE!
  578.                 echo '   <br><br><div style="text-align: center;"><a href="' $location
  579.                     . "/right_main.php?mailbox=$urlMailbox"
  580.                     . "&amp;startMessage=$startMessage&amp;mail_sent=$mail_sent\">"
  581.                     . _("Return"'</a></div>';
  582.             }
  583.             exit();
  584.         }
  585.     else {
  586.         if ($compose_new_win == '1'{
  587.             compose_Header($color$mailbox);
  588.         }
  589.         else {
  590.             displayPageHeader($color$mailbox);
  591.         }
  592.         if (isset($AttachFailure)) {
  593.             plain_error_message(_("Could not move/copy file. File not attached"),
  594.                     $color);
  595.         }
  596.         checkInput(true);
  597.         showInputForm($session);
  598.         /* sqimap_logout($imapConnection); */
  599.     }
  600. elseif (isset($html_addr_search_done)) {
  601.  
  602.     // validate security token
  603.     //
  604.     sm_validate_security_token($submitted_token-1TRUE);
  605.  
  606.     if ($compose_new_win == '1'{
  607.         compose_Header($color$mailbox);
  608.     }
  609.     else {
  610.         displayPageHeader($color$mailbox);
  611.     }
  612.  
  613.     if (isset($send_to_search&& is_array($send_to_search)) {
  614.         foreach ($send_to_search as $k => $v{
  615.             if (substr($k01== 'T'{
  616.                 if ($send_to{
  617.                     $send_to .= ', ';
  618.                 }
  619.                 $send_to .= $v;
  620.             }
  621.             elseif (substr($k01== 'C'{
  622.                 if ($send_to_cc{
  623.                     $send_to_cc .= ', ';
  624.                 }
  625.                 $send_to_cc .= $v;
  626.             }
  627.             elseif (substr($k01== 'B'{
  628.                 if ($send_to_bcc{
  629.                     $send_to_bcc .= ', ';
  630.                 }
  631.                 $send_to_bcc .= $v;
  632.             }
  633.         }
  634.     }
  635.     showInputForm($session);
  636. elseif (isset($html_addr_search&& !isset($html_addr_search_cancel)) {
  637.     if (isset($_FILES['attachfile']&&
  638.             $_FILES['attachfile']['tmp_name'&&
  639.             $_FILES['attachfile']['tmp_name'!= 'none'{
  640.         if(saveAttachedFiles($session)) {
  641.             plain_error_message(_("Could not move/copy file. File not attached"));
  642.         }
  643.     }
  644.     /*
  645.      * I am using an include so as to elminiate an extra unnecessary
  646.      * click.  If you can think of a better way, please implement it.
  647.      */
  648.     include_once('./addrbook_search_html.php');
  649. elseif (isset($attach)) {
  650.  
  651.     // validate security token
  652.     //
  653.     sm_validate_security_token($submitted_token-1TRUE);
  654.  
  655.     if ($compose_new_win == '1'{
  656.         compose_Header($color$mailbox);
  657.     else {
  658.         displayPageHeader($color$mailbox);
  659.     }
  660.     if (saveAttachedFiles($session)) {
  661.         plain_error_message(_("Could not move/copy file. File not attached"));
  662.     }
  663.     showInputForm($session);
  664. }
  665. elseif (isset($sigappend)) {
  666.  
  667.     // validate security token
  668.     //
  669.     sm_validate_security_token($submitted_token-1TRUE);
  670.  
  671.     $signature $idents[$identity]['signature'];
  672.  
  673.     $body .= "\n\n".($prefix_sig==true"-- \n":'').$signature;
  674.     if ($compose_new_win == '1'{
  675.         compose_Header($color$mailbox);
  676.     else {
  677.         displayPageHeader($color$mailbox);
  678.     }
  679.     showInputForm($session);
  680. elseif (isset($do_delete)) {
  681.  
  682.     // validate security token
  683.     //
  684.     sm_validate_security_token($submitted_token-1TRUE);
  685.  
  686.     if ($compose_new_win == '1'{
  687.         compose_Header($color$mailbox);
  688.     else {
  689.         displayPageHeader($color$mailbox);
  690.     }
  691.  
  692.     if (isset($delete&& is_array($delete)) {
  693.         foreach($delete as $index{
  694.             if (!empty($composeMessage->entities&& isset($composeMessage->entities[$index])) {
  695.                 $composeMessage->entities[$index]->purgeAttachments();
  696.                 // FIXME: one person reported that unset() didn't do anything at all here, so this is a work-around... but it triggers PHP notices if the unset() doesn't work, which should be fixed... but bigger question is if unset() doesn't work here, what about everywhere else? Anyway, uncomment this if you think you need it
  697.                 //$composeMessage->entities[$index] = NULL;
  698.                 unset ($composeMessage->entities[$index]);
  699.             }
  700.         }
  701.         $new_entities array();
  702.         foreach ($composeMessage->entities as $entity{
  703.             $new_entities[$entity;
  704.         }
  705.         $composeMessage->entities $new_entities;
  706.     }
  707.     showInputForm($session);
  708. else {
  709.     /*
  710.      * This handles the default case as well as the error case
  711.      * (they had the same code) --> if (isset($smtpErrors))
  712.      */
  713.  
  714.     if ($compose_new_win == '1'{
  715.         compose_Header($color$mailbox);
  716.     else {
  717.         displayPageHeader($color$mailbox);
  718.     }
  719.  
  720.     $newmail true;
  721.  
  722.     if (!isset($passed_ent_id)) {
  723.         $passed_ent_id '';
  724.     }
  725.     if (!isset($passed_id)) {
  726.         $passed_id '';
  727.     }
  728.     if (!isset($mailbox)) {
  729.         $mailbox '';
  730.     }
  731.     if (!isset($action)) {
  732.         $action '';
  733.     }
  734.  
  735.     $values newMail($mailbox,$passed_id,$passed_ent_id$action$session);
  736.  
  737.     // forward as attachment - subject is in the message in session
  738.     //
  739.     if ($action == 'forward_as_attachment' && empty($values['subject']))
  740.         $subject $composeMessage->rfc822_header->subject;
  741.  
  742.     /* in case the origin is not read_body.php */
  743.     if (isset($send_to)) {
  744.         $values['send_to'$send_to;
  745.     }
  746.     if (isset($send_to_cc)) {
  747.         $values['send_to_cc'$send_to_cc;
  748.     }
  749.     if (isset($send_to_bcc)) {
  750.         $values['send_to_bcc'$send_to_bcc;
  751.     }
  752.     if (isset($subject)) {
  753.         $values['subject'$subject;
  754.     }
  755.     showInputForm($session$values);
  756. }
  757.  
  758. exit();
  759.  
  760. /**************** Only function definitions go below *************/
  761.  
  762. function getforwardSubject($subject)
  763. {
  764.     if ((substr(strtolower($subject)04!= 'fwd:'&&
  765.             (substr(strtolower($subject)05!= '[fwd:'&&
  766.             (substr(strtolower($subject)06!= '[ fwd:')) {
  767.         $subject '[Fwd: ' $subject ']';
  768.     }
  769.     return $subject;
  770. }
  771.  
  772. /* This function is used when not sending or adding attachments */
  773. function newMail ($mailbox=''$passed_id=''$passed_ent_id=''$action=''$session=''{
  774.     global $editor_size$default_use_priority$body$idents,
  775.         $use_signature$data_dir$username,
  776.         $key$imapServerAddress$imapPort$imap_stream_options,
  777.         $composeMessage$body_quote$request_mdn$request_dr,
  778.         $mdn_user_support$languages$squirrelmail_language,
  779.         $default_charset$do_not_reply_to_self;
  780.  
  781.     /*
  782.      * Set $default_charset to correspond with the user's selection
  783.      * of language interface. $default_charset global is not correct,
  784.      * if message is composed in new window.
  785.      */
  786.     set_my_charset();
  787.  
  788.     $send_to $send_to_cc $send_to_bcc $subject $identity '';
  789.     $mailprio 3;
  790.  
  791.     if ($passed_id{
  792.         $imapConnection sqimap_login($usernamefalse$imapServerAddress,
  793.                 $imapPort0$imap_stream_options);
  794.  
  795.         sqimap_mailbox_select($imapConnection$mailbox);
  796.         $message sqimap_get_message($imapConnection$passed_id$mailbox);
  797.  
  798.         $body '';
  799.         if ($passed_ent_id{
  800.             /* redefine the messsage in case of message/rfc822 */
  801.             $message $message->getEntity($passed_ent_id);
  802.             /* message is an entity which contains the envelope and type0=message
  803.              * and type1=rfc822. The actual entities are childs from
  804.              * $message->entities[0]. That's where the encoding and is located
  805.              */
  806.  
  807.             $entities $message->entities[0]->findDisplayEntity
  808.                 (array()$alt_order array('text/plain'));
  809.             if (!count($entities)) {
  810.                 $entities $message->entities[0]->findDisplayEntity
  811.                     (array()$alt_order array('text/plain','text/html'));
  812.             }
  813.             $orig_header $message->rfc822_header/* here is the envelope located */
  814.             /* redefine the message for picking up the attachments */
  815.             $message $message->entities[0];
  816.  
  817.         else {
  818.             $entities $message->findDisplayEntity (array()$alt_order array('text/plain'));
  819.             if (!count($entities)) {
  820.                 $entities $message->findDisplayEntity (array()$alt_order array('text/plain','text/html'));
  821.             }
  822.             $orig_header $message->rfc822_header;
  823.         }
  824.  
  825.         $type0 $message->type0;
  826.         $type1 $message->type1;
  827.         foreach ($entities as $ent{
  828.             $msg $message->getEntity($ent);
  829.             $type0 $msg->type0;
  830.             $type1 $msg->type1;
  831.             $unencoded_bodypart mime_fetch_body($imapConnection$passed_id$ent);
  832.             $body_part_entity $message->getEntity($ent);
  833.             $bodypart decodeBody($unencoded_bodypart,
  834.                     $body_part_entity->header->encoding);
  835.             if ($type1 == 'html'{
  836.                 $bodypart str_replace("\n"' '$bodypart);
  837.                 $bodypart preg_replace(array('/<\/?p>/i','/<div><\/div>/i','/<br\s*(\/)*>/i','/<\/?div>/i')"\n"$bodypart);
  838.                 $bodypart str_replace(array('&nbsp;','&gt;','&lt;'),array(' ','>','<'),$bodypart);
  839.                 $bodypart strip_tags($bodypart);
  840.             }
  841.             if (isset($languages[$squirrelmail_language]['XTRA_CODE']&&
  842.                     function_exists($languages[$squirrelmail_language]['XTRA_CODE''_decode')) {
  843.                 if (mb_detect_encoding($bodypart!= 'ASCII'{
  844.                     $bodypart call_user_func($languages[$squirrelmail_language]['XTRA_CODE''_decode'$bodypart);
  845.                 }
  846.             }
  847.  
  848.             // charset encoding in compose form stuff
  849.             if (isset($body_part_entity->header->parameters['charset'])) {
  850.                 $actual $body_part_entity->header->parameters['charset'];
  851.             else {
  852.                 $actual 'us-ascii';
  853.             }
  854.  
  855.             if $actual && is_conversion_safe($actual&& $actual != $default_charset){
  856.                 $bodypart charset_convert($actual,$bodypart,$default_charset,false);
  857.             }
  858.             // end of charset encoding in compose
  859.  
  860.             $body .= $bodypart;
  861.         }
  862.         if ($default_use_priority{
  863.             $mailprio substr($orig_header->priority,0,1);
  864.             if (!$mailprio{
  865.                 $mailprio 3;
  866.             }
  867.         else {
  868.             $mailprio '';
  869.         }
  870.  
  871.         $from_o $orig_header->from;
  872.         if (is_array($from_o)) {
  873.             if (isset($from_o[0])) {
  874.                 $from_o $from_o[0];
  875.             }
  876.         }
  877.         if (is_object($from_o)) {
  878.             $orig_from $from_o->getAddress();
  879.         else {
  880.             $orig_from '';
  881.         }
  882.  
  883.         $identities array();
  884.         if (count($idents1{
  885.             foreach($idents as $nr=>$data{
  886.                 $enc_from_name '"'.$data['full_name'].'" <'$data['email_address'].'>';
  887.                 $identities[$enc_from_name;
  888.             }
  889.  
  890.             $identity_match $orig_header->findAddress($identities);
  891.             if ($identity_match !== FALSE{
  892.                 $identity $identity_match;
  893.             }
  894.         }
  895.  
  896.         switch ($action{
  897.             case ('draft'):
  898.                 $use_signature FALSE;
  899.                 $composeMessage->rfc822_header $orig_header;
  900.                 $send_to decodeHeader($orig_header->getAddr_s('to'),false,false,true);
  901.                 $send_to_cc decodeHeader($orig_header->getAddr_s('cc'),false,false,true);
  902.                 $send_to_bcc decodeHeader($orig_header->getAddr_s('bcc'),false,false,true);
  903.                 $send_from $orig_header->getAddr_s('from');
  904.                 $send_from_parts new AddressStructure();
  905.                 $send_from_parts $orig_header->parseAddress($send_from);
  906.                 $send_from_add $send_from_parts->mailbox '@' $send_from_parts->host;
  907.                 $identity find_identity(array($send_from_add));
  908.                 $subject decodeHeader($orig_header->subject,false,false,true);
  909.  
  910.                 // Remember the receipt settings
  911.                 $request_mdn $mdn_user_support && !empty($orig_header->dnt'1' '0';
  912.                 $request_dr $mdn_user_support && !empty($orig_header->drnt'1' '0';
  913.  
  914.                 /* remember the references and in-reply-to headers in case of an reply */
  915. //FIXME: it would be better to fiddle with headers inside of the message object or possibly when delivering the message to its destination (drafts folder?); is this possible?
  916.                 $composeMessage->rfc822_header->more_headers['References'$orig_header->references;
  917.                 $composeMessage->rfc822_header->more_headers['In-Reply-To'$orig_header->in_reply_to;
  918.                 // rewrap the body to clean up quotations and line lengths
  919.                 sqBodyWrap($body$editor_size);
  920.                 $composeMessage getAttachments($message$composeMessage$passed_id$entities$imapConnection);
  921.                 break;
  922.             case ('edit_as_new'):
  923.                 $send_to decodeHeader($orig_header->getAddr_s('to'),false,false,true);
  924.                 $send_to_cc decodeHeader($orig_header->getAddr_s('cc'),false,false,true);
  925.                 $send_to_bcc decodeHeader($orig_header->getAddr_s('bcc'),false,false,true);
  926.                 $subject decodeHeader($orig_header->subject,false,false,true);
  927.                 $mailprio $orig_header->priority;
  928.                 $orig_from '';
  929.                 $composeMessage getAttachments($message$composeMessage$passed_id$entities$imapConnection);
  930.                 // rewrap the body to clean up quotations and line lengths
  931.                 sqBodyWrap($body$editor_size);
  932.                 break;
  933.             case ('forward'):
  934.                 $send_to '';
  935.                 $subject getforwardSubject(decodeHeader($orig_header->subject,false,false,true));
  936.                 $body getforwardHeader($orig_header$body;
  937.                 // the logic for calling sqUnWordWrap here would be to allow the browser to wrap the lines
  938.                 // forwarded message text should be as undisturbed as possible, so commenting out this call
  939.                 // sqUnWordWrap($body);
  940.                 $composeMessage getAttachments($message$composeMessage$passed_id$entities$imapConnection);
  941.  
  942.                 //add a blank line after the forward headers
  943.                 $body "\n" $body;
  944.                 break;
  945.             case ('forward_as_attachment'):
  946.                 $subject getforwardSubject(decodeHeader($orig_header->subject,false,false,true));
  947.                 $composeMessage getMessage_RFC822_Attachment($message$composeMessage$passed_id$passed_ent_id$imapConnection);
  948.                 $subject decodeHeader($orig_header->subject,false,false,true);
  949.                 $subject str_replace('"'"'"$subject);
  950.                 $subject trim($subject);
  951.                 if (substr(strtolower($subject)04!= 'fwd:'{
  952.                     $subject 'Fwd: ' $subject;
  953.                 }
  954.                 $body '';
  955.                 break;
  956.             case ('reply_all'):
  957.                 if(isset($orig_header->mail_followup_to&& $orig_header->mail_followup_to{
  958.                     $send_to $orig_header->getAddr_s('mail_followup_to');
  959.                 else {
  960.                     $send_to_cc replyAllString($orig_header);
  961.                     $send_to_cc decodeHeader($send_to_cc,false,false,true);
  962.                     $send_to_cc str_replace('""''"'$send_to_cc);
  963.                 }
  964.             case ('reply'):
  965.                 // skip this if send_to was already set right above here
  966.                 if(!$send_to{
  967.                     $send_to $orig_header->reply_to;
  968.                     if (is_array($send_to&& count($send_to)) {
  969.                         $send_to $orig_header->getAddr_s('reply_to'','FALSETRUE);
  970.                     else if (is_object($send_to)) /* unneccesarry, just for failsafe purpose */
  971.                         $send_to $orig_header->getAddr_s('reply_to'','FALSETRUE);
  972.                     else {
  973.                         $send_to $orig_header->getAddr_s('from'','FALSETRUE);
  974.                     }
  975.                 }
  976.                 $send_to decodeHeader($send_to,false,false,true);
  977.                 $send_to str_replace('""''"'$send_to);
  978.  
  979.  
  980.                 // If user doesn't want replies to her own messages
  981.                 // going back to herself (instead send again to the
  982.                 // original recipient of the message being replied to),
  983.                 // then iterate through identities, checking if the TO
  984.                 // field is one of them (if the reply is to ourselves)
  985.                 //
  986.                 // Note we don't bother if the original message doesn't
  987.                 // have anything in the TO field itself (because that's
  988.                 // what we use if we change the recipient to be that of
  989.                 // the previous message)
  990.                 //
  991.                 if ($do_not_reply_to_self && !empty($orig_header->to)) {
  992.  
  993.                     $orig_to '';
  994.  
  995.                     foreach($idents as $id{
  996.  
  997.                         if (!empty($id['email_address'])
  998.                          && strpos($send_to$id['email_address']!== FALSE{
  999.  
  1000.                             // if this is a reply-all, the original recipient
  1001.                             // is already in the CC field, so we can just blank
  1002.                             // the recipient (TO field) (as long as the CC field
  1003.                             // isn't empty that is)... but then move the CC into
  1004.                             // the TO, so TO isn't empty
  1005.                             //
  1006.                             if ($action == 'reply_all' && !empty($send_to_cc)) {
  1007.                                 $orig_to $send_to_cc;
  1008.                                 $send_to_cc '';
  1009.                                 break;
  1010.                             }
  1011.  
  1012.                             $orig_to $orig_header->to;
  1013.                             if (is_array($orig_to&& count($orig_to)) {
  1014.                                 $orig_to $orig_header->getAddr_s('to'','FALSETRUE);
  1015.                             else if (is_object($orig_to)) /* unneccesarry, just for failsafe purpose */
  1016.                                 $orig_to $orig_header->getAddr_s('to'','FALSETRUE);
  1017.                             else {
  1018.                                 $orig_to '';
  1019.                             }
  1020.                             $orig_to decodeHeader($orig_to,false,false,true);
  1021.                             $orig_to str_replace('""''"'$orig_to);
  1022.  
  1023.                             break;
  1024.                         }
  1025.                     }
  1026.  
  1027.                     // if the reply was addressed back to ourselves,
  1028.                     // we will send it to the TO of the previous message
  1029.                     //
  1030.                     if (!empty($orig_to)) {
  1031.  
  1032.                         $send_to $orig_to;
  1033.  
  1034.                         // in this case, we also want to reset the FROM
  1035.                         // identity as well (it should match the original
  1036.                         // *FROM* header instead of TO or CC)
  1037.                         //
  1038.                         if (count($idents1{
  1039.                             $identity '';
  1040.                             foreach($idents as $i => $id{
  1041.                                 if (!empty($id['email_address'])
  1042.                                  && strpos($orig_from$id['email_address']!== FALSE{
  1043.                                     $identity $i;
  1044.                                     break;
  1045.                                 }
  1046.                             }
  1047.                         }
  1048.  
  1049.                     }
  1050.  
  1051.                 }
  1052.  
  1053.  
  1054.                 $subject decodeHeader($orig_header->subject,false,false,true);
  1055.                 $subject str_replace('"'"'"$subject);
  1056.                 $subject trim($subject);
  1057.                 if (substr(strtolower($subject)03!= 're:'{
  1058.                     $subject 'Re: ' $subject;
  1059.                 }
  1060.                 /* this corrects some wrapping/quoting problems on replies */
  1061.                 $rewrap_body explode("\n"$body);
  1062.                 $from (is_array($orig_header->from&& !empty($orig_header->from)) $orig_header->from[0$orig_header->from;
  1063.                 $body '';
  1064.                 $strip_sigs getPref($data_dir$username'strip_sigs');
  1065.                 foreach ($rewrap_body as $line{
  1066.                     if ($strip_sigs && rtrim($line"\r\n"== '-- '{
  1067.                         break;
  1068.                     }
  1069.                     if (preg_match("/^(>+)/"$line$matches)) {
  1070.                         $gt $matches[1];
  1071.                         $body .= $body_quote str_replace("\n""\n$body_quote$gt "rtrim($line)) ."\n";
  1072.                     else {
  1073.                         $body .= $body_quote (!empty($body_quote' ' ''str_replace("\n""\n$body_quote(!empty($body_quote' ' '')rtrim($line)) "\n";
  1074.                     }
  1075.                 }
  1076.  
  1077.                 //rewrap the body to clean up quotations and line lengths
  1078.                 $body sqBodyWrap ($body$editor_size);
  1079.  
  1080.                 $body getReplyCitation($from $orig_header->date$body;
  1081.                 $composeMessage->reply_rfc822_header $orig_header;
  1082.  
  1083.                 break;
  1084.             default:
  1085.                 break;
  1086.         }
  1087. //FIXME: we used to register $compose_messages in the session here, but not any more - so do we still need the session_write_close() and sqimap_logout() here?  We probably need the IMAP logout, but what about the session closure?
  1088.         session_write_close();
  1089.         sqimap_logout($imapConnection);
  1090.     }
  1091.     $ret array'send_to' => $send_to,
  1092.             'send_to_cc' => $send_to_cc,
  1093.             'send_to_bcc' => $send_to_bcc,
  1094.             'subject' => $subject,
  1095.             'mailprio' => $mailprio,
  1096.             'body' => $body,
  1097.             'identity' => $identity );
  1098.  
  1099.     return ($ret);
  1100. /* function newMail() */
  1101.  
  1102. /**
  1103.  * downloads attachments from original message, stores them in attachment directory and adds
  1104.  * them to composed message.
  1105.  * @param object $message 
  1106.  * @param object $composeMessage 
  1107.  * @param integer $passed_id 
  1108.  * @param mixed $entities 
  1109.  * @param mixed $imapConnection 
  1110.  * @return object 
  1111.  */
  1112. function getAttachments($message&$composeMessage$passed_id$entities$imapConnection{
  1113.     global $squirrelmail_language$languages$username$attachment_dir;
  1114.  
  1115.     if (!count($message->entities||
  1116.             ($message->type0 == 'message' && $message->type1 == 'rfc822')) {
  1117.         if !in_array($message->entity_id$entities&& $message->entity_id{
  1118.             switch ($message->type0{
  1119.                 case 'message':
  1120.                     if ($message->type1 == 'rfc822'{
  1121.                         $filename $message->rfc822_header->subject;
  1122.                         if ($filename == ""{
  1123.                             $filename "untitled-".$message->entity_id;
  1124.                         }
  1125.                         $filename .= '.eml';
  1126.                     else {
  1127.                         $filename $message->getFilename();
  1128.                     }
  1129.                     break;
  1130.                 default:
  1131.                     if (!$message->mime_header/* temporary hack */
  1132.                         $message->mime_header $message->header;
  1133.                     }
  1134.                     $filename $message->getFilename();
  1135.                     break;
  1136.             }
  1137. //FIXME: added three args to the following, so as to set the last one to TRUE, to mimick a fix in 1.4.21 (#2994865), but didn't test this (note that in 1.4.21, the 2nd and 3rd args are FALSE, but here in this code, they weren't being specified (thus defaulting to TRUE), so I don't know if that means this code is outdated and should have been changed to FALSE, FALSE or if this code is completely different and the addition of the TRUE for arg #4 is wrong
  1138.             $filename str_replace('&#32;'' 'decodeHeader($filenametruetruetrue));
  1139.             if (isset($languages[$squirrelmail_language]['XTRA_CODE']&&
  1140.                     function_exists($languages[$squirrelmail_language]['XTRA_CODE''_encode')) {
  1141.                 $filename =  call_user_func($languages[$squirrelmail_language]['XTRA_CODE''_encode'$filename);
  1142.             }
  1143.  
  1144.             $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1145.             $localfilename sq_get_attach_tempfile();
  1146.             $message->att_local_name $localfilename;
  1147.  
  1148.             $composeMessage->initAttachment($message->type0.'/'.$message->type1,$filename,
  1149.                     $localfilename);
  1150.  
  1151.             /* Write Attachment to file */
  1152.             $fp fopen ($hashed_attachment_dir '/' $localfilename'wb');
  1153.             mime_print_body_lines ($imapConnection$passed_id$message->entity_id$message->header->encoding$fp);
  1154.             fclose ($fp);
  1155.         }
  1156.     else {
  1157.         for ($i=0$entCount=count($message->entities)$i<$entCount;$i++{
  1158.             $composeMessage=getAttachments($message->entities[$i]$composeMessage$passed_id$entities$imapConnection);
  1159.         }
  1160.     }
  1161.     return $composeMessage;
  1162. }
  1163.  
  1164. function getMessage_RFC822_Attachment($message$composeMessage$passed_id,
  1165.         $passed_ent_id=''$imapConnection{
  1166.     if (!$passed_ent_id{
  1167.         $body_a sqimap_run_command($imapConnection,
  1168.                 'FETCH '.$passed_id.' RFC822',
  1169.                 TRUE$response$readmessage,
  1170.                 TRUE);
  1171.     else {
  1172.         $body_a sqimap_run_command($imapConnection,
  1173.                 'FETCH '.$passed_id.' BODY['.$passed_ent_id.']',
  1174.                 TRUE$response$readmessageTRUE);
  1175.         $message $message->parent;
  1176.     }
  1177.     if ($response == 'OK'{
  1178.         $subject encodeHeader($message->rfc822_header->subject);
  1179.         array_shift($body_a);
  1180.         array_pop($body_a);
  1181.         $body implode(''$body_a"\r\n";
  1182.  
  1183.         global $username$attachment_dir;
  1184.         $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1185.         $localfilename sq_get_attach_tempfile();
  1186.         $fp fopen($hashed_attachment_dir '/' $localfilename'wb');
  1187.         fwrite ($fp$body);
  1188.         fclose($fp);
  1189.         $composeMessage->initAttachment('message/rfc822',$subject.'.eml',
  1190.                 $localfilename);
  1191.     }
  1192.     return $composeMessage;
  1193. }
  1194.  
  1195. function showInputForm ($session$values=false{
  1196.     global $send_to$send_to_cc$send_to_bcc,
  1197.         $body$startMessage$action$attachments,
  1198.         $use_signature$signature$prefix_sig$session_expired,
  1199.         $editor_size$editor_height$subject$newmail,
  1200.         $use_javascript_addr_book$passed_id$mailbox$fwduid,
  1201.         $from_htmladdr_search$location_of_buttons$attachment_dir,
  1202.         $username$data_dir$identity$idents$delete_draft,
  1203.         $mailprio$compose_new_win$saved_draft$mail_sent$sig_first,
  1204.         $composeMessage$composesession$default_charset,
  1205.         $compose_onsubmit$oTemplate$oErrorHandler;
  1206.  
  1207.     if (checkForJavascript()) {
  1208.         $onfocus ' onfocus="alreadyFocused=true;"';
  1209.         $onfocus_array array('onfocus' => 'alreadyFocused=true;');
  1210.     }
  1211.     else {
  1212.         $onfocus '';
  1213.         $onfocus_array array();
  1214.     }
  1215.  
  1216.     if ($values{
  1217.         $send_to $values['send_to'];
  1218.         $send_to_cc $values['send_to_cc'];
  1219.         $send_to_bcc $values['send_to_bcc'];
  1220.         $subject $values['subject'];
  1221.         $mailprio $values['mailprio'];
  1222.         $body $values['body'];
  1223.         $identity = (int) $values['identity'];
  1224.     else {
  1225.         $send_to decodeHeader($send_totruefalse);
  1226.         $send_to_cc decodeHeader($send_to_cctruefalse);
  1227.         $send_to_bcc decodeHeader($send_to_bcctruefalse);
  1228.     }
  1229.  
  1230.     if ($use_javascript_addr_book{
  1231. //FIXME: NO HTML IN CORE!
  1232.         echo "\n"'<script type="text/javascript">'."\n<!--\n" .
  1233.             'function open_abook() { ' "\n" .
  1234.             '  var nwin = window.open("addrbook_popup.php","abookpopup",' .
  1235.             '"width=670,height=300,resizable=yes,scrollbars=yes");' "\n" .
  1236.             '  if((!nwin.opener) && (document.windows != null))' "\n" .
  1237.             '    nwin.opener = document.windows;' "\n" .
  1238.             "}\n" .
  1239.             "// -->\n</script>\n\n";
  1240.     }
  1241.  
  1242. //FIXME: NO HTML IN CORE!
  1243.     echo "\n" '<form name="compose" action="compose.php" method="post" ' .
  1244.         'enctype="multipart/form-data"';
  1245.  
  1246.     $compose_onsubmit array();
  1247.     global $null;
  1248.     do_hook('compose_form'$null);
  1249.  
  1250.     // Plugins that use compose_form hook can add an array entry
  1251.     // to the globally scoped $compose_onsubmit; we add them up
  1252.     // here and format the form tag's full onsubmit handler.
  1253.     // Each plugin should use "return false" if they need to
  1254.     // stop form submission but otherwise should NOT use "return
  1255.     // true" to give other plugins the chance to do what they need
  1256.     // to do; SquirrelMail itself will add the final "return true".
  1257.     // Onsubmit text is enclosed inside of double quotes, so plugins
  1258.     // need to quote accordingly.
  1259.     //
  1260.     // Also, plugin authors should try to retain compatibility with
  1261.     // the Compose Extras plugin by resetting its compose submit
  1262.     // counter when preventing form submit.  Use this code: 
  1263.     // if (your-code-here) { submit_count = 0; return false; }
  1264.     //
  1265.     if (checkForJavascript()) {
  1266.         if (empty($compose_onsubmit))
  1267.             $compose_onsubmit array();
  1268.         else if (!is_array($compose_onsubmit))
  1269.             $compose_onsubmit array($compose_onsubmit);
  1270.  
  1271.         $onsubmit_text '';
  1272.         foreach ($compose_onsubmit as $text{
  1273.             $text trim($text);
  1274.             if (!empty($text)) {
  1275.                 if (substr($text-1!= ';' && substr($text-1!= '}')
  1276.                     $text .= '; ';
  1277.                 $onsubmit_text .= $text;
  1278.             }
  1279.         }
  1280.  
  1281.         if (!empty($onsubmit_text))
  1282. //FIXME: DON'T ECHO HTML FROM CORE!
  1283.             echo ' onsubmit="' $onsubmit_text ' return true;"';
  1284.     }
  1285.  
  1286.  
  1287. //FIXME: NO HTML IN CORE!
  1288.     echo ">\n";
  1289.  
  1290. //FIXME: DON'T ECHO HTML FROM CORE!
  1291.     echo addHidden('smtoken'sm_generate_security_token());
  1292.  
  1293. //FIXME: DON'T ECHO HTML FROM CORE!
  1294.     echo addHidden('startMessage'$startMessage);
  1295.  
  1296.     if ($action == 'draft'{
  1297. //FIXME: DON'T ECHO HTML FROM CORE!
  1298.         echo addHidden('delete_draft'$passed_id);
  1299.     }
  1300.     if (isset($delete_draft)) {
  1301. //FIXME: DON'T ECHO HTML FROM CORE!
  1302.         echo addHidden('delete_draft'$delete_draft);
  1303.     }
  1304.     if (isset($session)) {
  1305. //FIXME: DON'T ECHO HTML FROM CORE!
  1306.         echo addHidden('session'$session);
  1307.     }
  1308.  
  1309.     if (isset($passed_id)) {
  1310. //FIXME: DON'T ECHO HTML FROM CORE!
  1311.         echo addHidden('passed_id'$passed_id);
  1312.     }
  1313.  
  1314.     if (isset($fwduid)) {
  1315. //FIXME: DON'T ECHO HTML FROM CORE!
  1316.         echo addHidden('fwduid'$fwduid);
  1317.     }
  1318.  
  1319.     if ($saved_draft == 'yes'{
  1320.         $oTemplate->assign('note'_("Your draft has been saved."));
  1321.         $oTemplate->display('note.tpl');
  1322.     }
  1323.     if ($mail_sent == 'yes'{
  1324.         $oTemplate->assign('note'_("Your mail has been sent."));
  1325.         $oTemplate->display('note.tpl');
  1326.     }
  1327.     if ($compose_new_win == '1'{
  1328.         $oTemplate->display('compose_newwin_close.tpl');
  1329.     }
  1330.  
  1331.     if ($location_of_buttons == 'top'{
  1332. //FIXME: DON'T ECHO HTML FROM CORE!
  1333.         showComposeButtonRow();
  1334.     }
  1335.  
  1336.     $identities array();
  1337.     if (count($idents1{
  1338.         reset($idents);
  1339.         foreach($idents as $id => $data{
  1340.             $identities[$id$data['full_name'].' &lt;'.$data['email_address'].'&gt;';
  1341.         }
  1342.     }
  1343.  
  1344.     $oTemplate->assign('identities'$identities);
  1345.     $oTemplate->assign('identity_def'$identity);
  1346.     $oTemplate->assign('input_onfocus''onfocus="'.join(' '$onfocus_array).'"');
  1347.  
  1348.     $oTemplate->assign('to'sm_encode_html_special_chars($send_to));
  1349.     $oTemplate->assign('cc'sm_encode_html_special_chars($send_to_cc));
  1350.     $oTemplate->assign('bcc'sm_encode_html_special_chars($send_to_bcc));
  1351.     $oTemplate->assign('subject'sm_encode_html_special_chars($subject));
  1352.  
  1353.     // access keys...
  1354.     //
  1355.     global $accesskey_compose_to$accesskey_compose_cc,
  1356.            $accesskey_compose_identity$accesskey_compose_bcc,
  1357.            $accesskey_compose_subject;
  1358.     $oTemplate->assign('accesskey_compose_identity'$accesskey_compose_identity);
  1359.     $oTemplate->assign('accesskey_compose_to'$accesskey_compose_to);
  1360.     $oTemplate->assign('accesskey_compose_cc'$accesskey_compose_cc);
  1361.     $oTemplate->assign('accesskey_compose_bcc'$accesskey_compose_bcc);
  1362.     $oTemplate->assign('accesskey_compose_subject'$accesskey_compose_subject);
  1363.  
  1364.     $oTemplate->display('compose_header.tpl');
  1365.  
  1366.     if ($location_of_buttons == 'between'{
  1367. //FIXME: DON'T ECHO HTML FROM CORE!
  1368.         showComposeButtonRow();
  1369.     }
  1370.  
  1371.     $body_str '';
  1372.     if ($use_signature == true && $newmail == true && !isset($from_htmladdr_search)) {
  1373.         $signature $idents[$identity]['signature'];
  1374.  
  1375.         if ($sig_first == '1'{
  1376.             /*
  1377.              * FIXME: test is specific to ja_JP translation implementation.
  1378.              * This test might apply incorrect conversion to other translations, but
  1379.              * use of 7bit iso-2022-jp charset in other translations might have other
  1380.              * issues too.
  1381.              */
  1382.             if ($default_charset == 'iso-2022-jp'{
  1383.                 $body_str "\n\n".($prefix_sig==true"-- \n":'').mb_convert_encoding($signature'EUC-JP');
  1384.             else {
  1385.                 $body_str "\n\n".($prefix_sig==true"-- \n":'').decodeHeader($signature,false,false);
  1386.             }
  1387.             $body_str .= "\n\n".sm_encode_html_special_chars(decodeHeader($body,false,false));
  1388.         else {
  1389.             $body_str "\n\n".sm_encode_html_special_chars(decodeHeader($body,false,false));
  1390.             // FIXME: test is specific to ja_JP translation implementation. See above comments.
  1391.             if ($default_charset == 'iso-2022-jp'{
  1392.                 $body_str .= "\n\n".($prefix_sig==true"-- \n":'').mb_convert_encoding($signature'EUC-JP');
  1393.             else {
  1394.                 $body_str .= "\n\n".($prefix_sig==true"-- \n":'').decodeHeader($signature,false,false);
  1395.             }
  1396.         }
  1397.     else {
  1398.         $body_str sm_encode_html_special_chars(decodeHeader($body,false,false));
  1399.     }
  1400.  
  1401.     $oTemplate->assign('editor_width'(int)$editor_size);
  1402.     $oTemplate->assign('editor_height'(int)$editor_height);
  1403.     $oTemplate->assign('input_onfocus''onfocus="'.join(' '$onfocus_array).'"');
  1404.     $oTemplate->assign('body'$body_str);
  1405.     $oTemplate->assign('show_bottom_send'$location_of_buttons!='bottom');
  1406.  
  1407.     // access keys...
  1408.     //
  1409.     global $accesskey_compose_body$accesskey_compose_send;
  1410.     $oTemplate->assign('accesskey_compose_body'$accesskey_compose_body);
  1411.     $oTemplate->assign('accesskey_compose_send'$accesskey_compose_send);
  1412.  
  1413.     $oTemplate->display ('compose_body.tpl');
  1414.  
  1415.     if ($location_of_buttons == 'bottom'{
  1416. //FIXME: DON'T ECHO HTML FROM CORE!
  1417.         showComposeButtonRow();
  1418.     }
  1419.  
  1420.     // composeMessage can be empty when coming from a restored session
  1421.     if (is_object($composeMessage&& $composeMessage->entities)
  1422.         $attach_array $composeMessage->entities;
  1423.     if ($session_expired && !empty($attachments&& is_array($attachments))
  1424.         $attach_array $attachments;
  1425.  
  1426.     /* This code is for attachments */
  1427.     if ((bool) ini_get('file_uploads')) {
  1428.  
  1429.         /* Calculate the max size for an uploaded file.
  1430.          * This is advisory for the user because we can't actually prevent
  1431.          * people to upload too large files. */
  1432.         $sizes array();
  1433.         /* php.ini vars which influence the max for uploads */
  1434.         $configvars array('post_max_size''memory_limit''upload_max_filesize');
  1435.         foreach($configvars as $var{
  1436.             /* skip 0 or empty values, and -1 which means 'unlimited' */
  1437.             if$size getByteSize(ini_get($var)) ) {
  1438.                 if $size != '-1' {
  1439.                     $sizes[$size;
  1440.                 }
  1441.             }
  1442.         }
  1443.  
  1444.         $attach array();
  1445.         global $username$attachment_dir;
  1446.         $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1447.         if (!empty($attach_array)) {
  1448.             foreach ($attach_array as $key => $attachment{
  1449.                 $attached_file $attachment->att_local_name;
  1450.                 if ($attachment->att_local_name || $attachment->body_part{
  1451.                     $attached_filename decodeHeader($attachment->mime_header->getParameter('name'));
  1452.                     $type $attachment->mime_header->type0.'/'.
  1453.                         $attachment->mime_header->type1;
  1454.  
  1455.                     $a array();
  1456.                     $a['Key'$key;
  1457.                     $a['FileName'$attached_filename;
  1458.                     $a['ContentType'$type;
  1459.                     $a['Size'filesize($hashed_attachment_dir '/' $attached_file);
  1460.                     $attach[$key$a;
  1461.                 }
  1462.             }
  1463.         }
  1464.  
  1465.         $max min($sizes);
  1466.         $oTemplate->assign('max_file_size'empty($max? -$max);
  1467.         $oTemplate->assign('attachments'$attach);
  1468.  
  1469.         // access keys...
  1470.         //
  1471.         global $accesskey_compose_attach_browse$accesskey_compose_attach,
  1472.                $accesskey_compose_delete_attach;
  1473.         $oTemplate->assign('accesskey_compose_attach_browse'$accesskey_compose_attach_browse);
  1474.         $oTemplate->assign('accesskey_compose_attach'$accesskey_compose_attach);
  1475.         $oTemplate->assign('accesskey_compose_delete_attach'$accesskey_compose_delete_attach);
  1476.  
  1477.         $oTemplate->display('compose_attachments.tpl');
  1478.     // End of file_uploads if-block
  1479.     /* End of attachment code */
  1480.  
  1481.     $oTemplate->assign('username'$username);
  1482.     $oTemplate->assign('smaction'$action);
  1483.     $oTemplate->assign('mailbox'$mailbox);
  1484.     sqgetGlobalVar('QUERY_STRING'$queryStringSQ_SERVER);
  1485.     $oTemplate->assign('querystring'$queryString);
  1486.     $oTemplate->assign('composesession'$composesession);
  1487.     $oTemplate->assign('send_button_count'unique_widget_name('send'TRUE));
  1488.     if (!empty($attach_array))
  1489.         $oTemplate->assign('attachments'urlencode(serialize($attach_array)));
  1490.  
  1491.     $aUserNotices array();
  1492.  
  1493.     // File uploads are off, so we didn't show that part of the form.
  1494.     // To avoid bogus bug reports, tell the user why. 
  1495.     if (!(bool) ini_get('file_uploads')) {
  1496.         $aUserNotices[_("Because PHP file uploads are turned off, you can not attach files to this message. Please see your system administrator for details.");
  1497.     }
  1498.  
  1499.     $oTemplate->assign('user_notices'$aUserNotices);
  1500.  
  1501.     $oTemplate->display('compose_form_close.tpl');
  1502.  
  1503.     if ($compose_new_win=='1'{
  1504.         $oTemplate->display('compose_newwin_close.tpl');
  1505.     }
  1506.  
  1507.     $oErrorHandler->setDelayedErrors(false);
  1508.     $oTemplate->display('footer.tpl');
  1509. }
  1510.  
  1511.  
  1512. function showComposeButtonRow({
  1513.     global $use_javascript_addr_book$save_as_draft,
  1514.         $default_use_priority$mailprio$default_use_mdn,
  1515.         $request_mdn$request_dr,
  1516.         $data_dir$username;
  1517.  
  1518.     global $oTemplate$buffer_hook;
  1519.  
  1520.     if ($default_use_priority{
  1521.         $priorities array('1'=>_("High")'3'=>_("Normal")'5'=>_("Low"));
  1522.         $priority = isset($mailprio$mailprio 3;
  1523.     else {
  1524.         $priorities array();
  1525.         $priority NULL;
  1526.     }
  1527.  
  1528.     $mdn_user_support=getPref($data_dir$username'mdn_user_support',$default_use_mdn);
  1529.  
  1530.     $address_book_button_attribs array();
  1531.     global $accesskey_compose_addresses;
  1532.     if ($accesskey_compose_addresses != 'NONE')
  1533.         $address_book_button_attribs['accesskey'$accesskey_compose_addresses;
  1534.     if ($use_javascript_addr_book && checkForJavascript()) {
  1535.         $addr_book addButton(_("Addresses"),
  1536.                                null,
  1537.                                array_merge($address_book_button_attribsarray('onclick' => 'javascript:open_abook();')));
  1538.     else {
  1539.         $addr_book addSubmit(_("Addresses")'html_addr_search'$address_book_button_attribs);
  1540.     }
  1541.  
  1542.     $oTemplate->assign('allow_priority'$default_use_priority==1);
  1543.     $oTemplate->assign('priority_list'$priorities);
  1544.     $oTemplate->assign('current_priority'$priority);
  1545.  
  1546.     $oTemplate->assign('notifications_enabled'$mdn_user_support==1);
  1547.     $oTemplate->assign('read_receipt'$request_mdn=='1');
  1548.     $oTemplate->assign('delivery_receipt'$request_dr=='1');
  1549.  
  1550.     $oTemplate->assign('drafts_enabled'$save_as_draft);
  1551.     $oTemplate->assign('address_book_button'$addr_book);
  1552.  
  1553.     // access keys...
  1554.     //
  1555.     global $accesskey_compose_priority$accesskey_compose_on_read,
  1556.            $accesskey_compose_on_delivery$accesskey_compose_signature,
  1557.            $accesskey_compose_save_draft$accesskey_compose_send;
  1558.     $oTemplate->assign('accesskey_compose_priority'$accesskey_compose_priority);
  1559.     $oTemplate->assign('accesskey_compose_on_read'$accesskey_compose_on_read);
  1560.     $oTemplate->assign('accesskey_compose_on_delivery'$accesskey_compose_on_delivery);
  1561.     $oTemplate->assign('accesskey_compose_signature'$accesskey_compose_signature);
  1562.     $oTemplate->assign('accesskey_compose_save_draft'$accesskey_compose_save_draft);
  1563.     $oTemplate->assign('accesskey_compose_send'$accesskey_compose_send);
  1564.  
  1565.     $oTemplate->display('compose_buttons.tpl');
  1566. }
  1567.  
  1568. function checkInput ($show{
  1569.     /*
  1570.      * I implemented the $show variable because the error messages
  1571.      * were getting sent before the page header.  So, I check once
  1572.      * using $show=false, and then when i'm ready to display the error
  1573.      * message, show=true
  1574.      */
  1575.     global $send_to$send_to_cc$send_to_bcc;
  1576.  
  1577.     $send_to trim($send_to);
  1578.     $send_to_cc trim($send_to_cc);
  1579.     $send_to_bcc trim($send_to_bcc);
  1580.     if (empty($send_to&& empty($send_to_cc&& empty($send_to_bcc)) {
  1581.         if ($show{
  1582.             plain_error_message(_("You have not filled in the \"To:\" field."));
  1583.         }
  1584.         return false;
  1585.     }
  1586.     return true;
  1587. /* function checkInput() */
  1588.  
  1589.  
  1590. /* True if FAILURE */
  1591. function saveAttachedFiles($session{
  1592.     global $composeMessage$username$attachment_dir;
  1593.  
  1594.     /* get out of here if no file was attached at all */
  1595.     if (is_uploaded_file($_FILES['attachfile']['tmp_name']) ) {
  1596.         return true;
  1597.     }
  1598.  
  1599.     $hashed_attachment_dir getHashedDir($username$attachment_dir);
  1600.     $localfilename sq_get_attach_tempfile();
  1601.     $fullpath $hashed_attachment_dir '/' $localfilename;
  1602.  
  1603.     // m_u_f works better with restricted PHP installs (safe_mode, open_basedir),
  1604.     // if that doesn't work, try a simple rename.
  1605.     if (!sq_call_function_suppress_errors('move_uploaded_file'array($_FILES['attachfile']['tmp_name']$fullpath))) {
  1606.         if (!sq_call_function_suppress_errors('rename'array($_FILES['attachfile']['tmp_name']$fullpath))) {
  1607.             return true;
  1608.         }
  1609.     }
  1610.     $type strtolower($_FILES['attachfile']['type']);
  1611.     $name $_FILES['attachfile']['name'];
  1612.     $composeMessage->initAttachment($type$name$localfilename);
  1613. }
  1614.  
  1615. /**
  1616.   * Parse strings such as "8M" and "2k" into their corresponding size in bytes
  1617.   *
  1618.   * NOTE: This function only recognizes the suffixes "K", "M" and "G"
  1619.   *       and will probably break very easily if the given size is in
  1620.   *       some completely different format.
  1621.   *
  1622.   * @param string $ini_size The input string to be converted
  1623.   *
  1624.   * @return mixed Boolean FALSE if something went wrong (the value passed in
  1625.   *                was empty?, the suffix was not recognized?), otherwise, the
  1626.   *                converted size in bytes (just the number (as an integer),
  1627.   *                no unit identifier included)
  1628.   *
  1629.   */
  1630. function getByteSize($ini_size{
  1631.  
  1632.     if(!$ini_size{
  1633.         return FALSE;
  1634.     }
  1635.  
  1636.     $ini_size trim($ini_size);
  1637.  
  1638.     // if there's some kind of letter at the end of the string we need to multiply.
  1639.     if(!is_numeric(substr($ini_size-1))) {
  1640.  
  1641.         switch(strtoupper(substr($ini_size-1))) {
  1642.             case 'G':
  1643.                 $bytesize 1073741824;
  1644.                 break;
  1645.             case 'M':
  1646.                 $bytesize 1048576;
  1647.                 break;
  1648.             case 'K':
  1649.                 $bytesize 1024;
  1650.                 break;
  1651.              default:
  1652.                 return FALSE;
  1653.         }
  1654.  
  1655.         return ($bytesize * (int)substr($ini_size0-1));
  1656.     }
  1657.  
  1658.     return $ini_size;
  1659. }
  1660.  
  1661.  
  1662. /**
  1663.  * temporary function to make use of the deliver class.
  1664.  * In the future the responsible backend should be automaticly loaded
  1665.  * and conf.pl should show a list of available backends.
  1666.  * The message also should be constructed by the message class.
  1667.  *
  1668.  * @param object $composeMessage The message being sent.  Please note
  1669.  *                                that it is passed by reference and
  1670.  *                                will be returned modified, with additional
  1671.  *                                headers, such as Message-ID, Date, In-Reply-To,
  1672.  *                                References, and so forth.
  1673.  *
  1674.  * @return boolean FALSE if delivery failed, or some non-FALSE value
  1675.  *                  upon success.
  1676.  *
  1677.  */
  1678. function deliverMessage(&$composeMessage$draft=false{
  1679.     global $send_to$send_to_cc$send_to_bcc$mailprio$subject$body,
  1680.         $username$identity$idents$data_dir,
  1681.         $request_mdn$request_dr$default_charset$useSendmail,
  1682.         $domain$action$default_move_to_sent$move_to_sent,
  1683.         $imapServerAddress$imapPort$imap_stream_options$sent_folder$key;
  1684.  
  1685.     $rfc822_header $composeMessage->rfc822_header;
  1686.  
  1687.     $abook addressbook_init(falsetrue);
  1688.     $rfc822_header->to $rfc822_header->parseAddress($send_to,truearray()''$domainarray(&$abook,'lookup'));
  1689.     $rfc822_header->cc $rfc822_header->parseAddress($send_to_cc,true,array()'',$domainarray(&$abook,'lookup'));
  1690.     $rfc822_header->bcc $rfc822_header->parseAddress($send_to_bcc,truearray()'',$domainarray(&$abook,'lookup'));
  1691.     $rfc822_header->priority $mailprio;
  1692.     $rfc822_header->subject $subject;
  1693.  
  1694.     $special_encoding='';
  1695.     if (strtolower($default_charset== 'iso-2022-jp'{
  1696.         if (mb_detect_encoding($body== 'ASCII'{
  1697.             $special_encoding '8bit';
  1698.         else {
  1699.             $body mb_convert_encoding($body'JIS');
  1700.             $special_encoding '7bit';
  1701.         }
  1702.     }
  1703.     $composeMessage->setBody($body);
  1704.  
  1705.     $reply_to '';
  1706.     $reply_to  $idents[$identity]['reply_to'];
  1707.     if ($reply_to && strpos($reply_to'@'=== FALSE)
  1708.         $reply_to .= '@' $domain;
  1709.     
  1710.     $from_addr build_from_header($identity);
  1711.     $rfc822_header->from $rfc822_header->parseAddress($from_addr,true);
  1712.     if ($reply_to{
  1713.         $rfc822_header->reply_to $rfc822_header->parseAddress($reply_to,true);
  1714.     }
  1715.     /* Receipt: On Read */
  1716.     if (isset($request_mdn&& $request_mdn{
  1717.         $rfc822_header->dnt $rfc822_header->parseAddress($from_addr,true);
  1718.     elseif (isset($rfc822_header->dnt)) {
  1719.         unset($rfc822_header->dnt);
  1720.     }
  1721.  
  1722.     /* Receipt: On Delivery */
  1723.     if (!empty($request_dr)) {
  1724. //FIXME: it would be better to fiddle with headers inside of the message object or possibly when delivering the message to its destination; is this possible?
  1725.         $rfc822_header->more_headers['Return-Receipt-To'$from_addr;
  1726.     elseif (isset($rfc822_header->more_headers['Return-Receipt-To'])) {
  1727.         unset($rfc822_header->more_headers['Return-Receipt-To']);
  1728.     }
  1729.  
  1730.     /* multipart messages */
  1731.     if (count($composeMessage->entities)) {
  1732.         $message_body new Message();
  1733.         $message_body->body_part $composeMessage->body_part;
  1734.         $composeMessage->body_part '';
  1735.         $mime_header new MessageHeader;
  1736.         $mime_header->type0 'text';
  1737.         $mime_header->type1 'plain';
  1738.         if ($special_encoding{
  1739.             $mime_header->encoding $special_encoding;
  1740.         else {
  1741.             $mime_header->encoding '8bit';
  1742.         }
  1743.         if ($default_charset{
  1744.             $mime_header->parameters['charset'$default_charset;
  1745.         }
  1746.         $message_body->mime_header $mime_header;
  1747.         array_unshift($composeMessage->entities$message_body);
  1748.         $content_type new ContentType('multipart/mixed');
  1749.     else {
  1750.         $content_type new ContentType('text/plain');
  1751.         if ($special_encoding{
  1752.             $rfc822_header->encoding $special_encoding;
  1753.         else {
  1754.             $rfc822_header->encoding '8bit';
  1755.         }
  1756.         if ($default_charset{
  1757.             $content_type->properties['charset']=$default_charset;
  1758.         }
  1759.     }
  1760.  
  1761.     $rfc822_header->content_type $content_type;
  1762.     $composeMessage->rfc822_header $rfc822_header;
  1763.     if ($action == 'reply' || $action == 'reply_all'{
  1764.         global $passed_id$passed_ent_id;
  1765.         $reply_id $passed_id;
  1766.         $reply_ent_id $passed_ent_id;
  1767.     else {
  1768.         $reply_id '';
  1769.         $reply_ent_id '';
  1770.     }
  1771.  
  1772.     /* Here you can modify the message structure just before we hand
  1773.        it over to deliver; plugin authors note that $composeMessage
  1774.        is sent and modified by reference since 1.5.2 */
  1775.     do_hook('compose_send'$composeMessage);
  1776.  
  1777.     if (!$useSendmail && !$draft{
  1778.         require_once(SM_PATH 'class/deliver/Deliver_SMTP.class.php');
  1779.         $deliver new Deliver_SMTP();
  1780.         global $smtpServerAddress$smtpPort$smtp_stream_options$pop_before_smtp$pop_before_smtp_host;
  1781.  
  1782.         $authPop (isset($pop_before_smtp&& $pop_before_smtptrue false;
  1783.         if (empty($pop_before_smtp_host)) $pop_before_smtp_host $smtpServerAddress;
  1784.         get_smtp_user($user$pass);
  1785.         $stream $deliver->initStream($composeMessage,$domain,0,
  1786.                 $smtpServerAddress$smtpPort$user$pass$authPop$pop_before_smtp_host$smtp_stream_options);
  1787.     elseif (!$draft{
  1788.         require_once(SM_PATH 'class/deliver/Deliver_SendMail.class.php');
  1789.         global $sendmail_path$sendmail_args;
  1790.         // Check for outdated configuration
  1791.         if (!isset($sendmail_args)) {
  1792.             if ($sendmail_path=='/var/qmail/bin/qmail-inject'{
  1793.                 $sendmail_args '';
  1794.             else {
  1795.                 $sendmail_args '-i -t';
  1796.             }
  1797.         }
  1798.         $deliver new Deliver_SendMail(array('sendmail_args'=>$sendmail_args));
  1799.         $stream $deliver->initStream($composeMessage,$sendmail_path);
  1800.     elseif ($draft{
  1801.         global $draft_folder;
  1802.         $imap_stream sqimap_login($usernamefalse$imapServerAddress,
  1803.                 $imapPort0$imap_stream_options);
  1804.         if (sqimap_mailbox_exists ($imap_stream$draft_folder)) {
  1805.             require_once(SM_PATH 'class/deliver/Deliver_IMAP.class.php');
  1806.             $imap_deliver new Deliver_IMAP();
  1807.             $success $imap_deliver->mail($composeMessage$imap_stream$reply_id$reply_ent_id$imap_stream$draft_folder);
  1808.             sqimap_logout($imap_stream);
  1809.             unset ($imap_deliver);
  1810.             $composeMessage->purgeAttachments();
  1811.             return $success;
  1812.         else {
  1813.             $msg  '<br />'.sprintf(_("Error: Draft folder %s does not exist.")sm_encode_html_special_chars($draft_folder));
  1814.             plain_error_message($msg);
  1815.             return false;
  1816.         }
  1817.     }
  1818.     $success false;
  1819.     if ($stream{
  1820.         $deliver->mail($composeMessage$stream$reply_id$reply_ent_id);
  1821.         $success $deliver->finalizeStream($stream);
  1822.     }
  1823.     if (!$success{
  1824.         // $deliver->dlv_server_msg is not always server's reply
  1825.         $msg _("Message not sent.")
  1826.              . "<br />\n"
  1827.              . (isset($deliver->dlv_msg$deliver->dlv_msg '');
  1828.         if (!empty($deliver->dlv_server_msg)) {
  1829.             // add 'server replied' part only when it is not empty.
  1830.             // Delivery error can be generated by delivery class itself
  1831.             $msg .= '<br />'
  1832.                   . _("Server replied:"' '
  1833.                   . (isset($deliver->dlv_ret_nr$deliver->dlv_ret_nr ' ' '')
  1834.                   . nl2br(sm_encode_html_special_chars($deliver->dlv_server_msg));
  1835.         }
  1836.         plain_error_message($msg);
  1837.     else {
  1838.         unset ($deliver);
  1839.         $imap_stream sqimap_login($usernamefalse$imapServerAddress$imapPort0$imap_stream_options);
  1840.  
  1841.  
  1842.         // mark as replied or forwarded if applicable
  1843.         //
  1844.         global $what$iAccount$startMessage$passed_id$fwduid$mailbox;
  1845.  
  1846.         if ($action=='reply' || $action=='reply_all' || $action=='forward' || $action=='forward_as_attachment'{
  1847.             require(SM_PATH 'functions/mailbox_display.php');
  1848.             $aMailbox sqm_api_mailbox_select($imap_stream$iAccount$mailbox,array('setindex' => $what'offset' => $startMessage),array());
  1849.             switch($action{
  1850.             case 'reply':
  1851.             case 'reply_all':
  1852.                 // check if we are allowed to set the \\Answered flag
  1853.                 if (in_array('\\answered',$aMailbox['PERMANENTFLAGS']true)) {
  1854.                     $aUpdatedMsgs sqimap_toggle_flag($imap_streamarray($passed_id)'\\Answered'truefalse);
  1855.                     if (isset($aUpdatedMsgs[$passed_id]['FLAGS'])) {
  1856.                         /**
  1857.                         * Only update the cached headers if the header is
  1858.                         * cached.
  1859.                         */
  1860.                         if (isset($aMailbox['MSG_HEADERS'][$passed_id])) {
  1861.                             $aMailbox['MSG_HEADERS'][$passed_id]['FLAGS'$aMsg['FLAGS'];
  1862.                         }
  1863.                     }
  1864.                 }
  1865.                 break;
  1866.             case 'forward':
  1867.             case 'forward_as_attachment':
  1868.                 // check if we are allowed to set the $Forwarded flag (RFC 4550 paragraph 2.8)
  1869.                 if (in_array('$forwarded',$aMailbox['PERMANENTFLAGS']true||
  1870.                     in_array('\\*',$aMailbox['PERMANENTFLAGS'])) {
  1871.  
  1872.                     // when forwarding as an attachment from the message
  1873.                     // list, passed_id is not used, need to get UID(s)
  1874.                     // from the query string
  1875.                     //
  1876.                     if (empty($passed_id&& !empty($fwduid))
  1877.                         $ids explode('_'$fwduid);
  1878.                     else
  1879.                         $ids array($passed_id);
  1880.  
  1881.                     $aUpdatedMsgs sqimap_toggle_flag($imap_stream$ids'$Forwarded'truefalse);
  1882.  
  1883.                     foreach ($ids as $id{
  1884.                         if (isset($aUpdatedMsgs[$id]['FLAGS'])) {
  1885.                             if (isset($aMailbox['MSG_HEADERS'][$id])) {
  1886.                                 $aMailbox['MSG_HEADERS'][$id]['FLAGS'$aMsg['FLAGS'];
  1887.                             }
  1888.                         }
  1889.                     }
  1890.                 }
  1891.                 break;
  1892.             }
  1893.  
  1894.             /**
  1895.              * Write mailbox with updated seen flag information back to cache.
  1896.              */
  1897.             if(isset($aUpdatedMsgs[$passed_id])) {
  1898.                 $mailbox_cache[$iAccount.'_'.$aMailbox['NAME']] $aMailbox;
  1899.                 sqsession_register($mailbox_cache,'mailbox_cache');
  1900.             }
  1901.  
  1902.         }
  1903.  
  1904.  
  1905.         // move to sent folder
  1906.         //
  1907.         $move_to_sent getPref($data_dir,$username,'move_to_sent');
  1908.         if (isset($default_move_to_sent&& ($default_move_to_sent != 0)) {
  1909.             $svr_allow_sent true;
  1910.         else {
  1911.             $svr_allow_sent false;
  1912.         }
  1913.  
  1914.         if (isset($sent_folder&& (($sent_folder != ''|| ($sent_folder != 'none'))
  1915.                 && sqimap_mailbox_exists$imap_stream$sent_folder)) {
  1916.             $fld_sent true;
  1917.         else {
  1918.             $fld_sent false;
  1919.         }
  1920.  
  1921.         if ((isset($move_to_sent&& ($move_to_sent != 0)) || (!isset($move_to_sent))) {
  1922.             $lcl_allow_sent true;
  1923.         else {
  1924.             $lcl_allow_sent false;
  1925.         }
  1926.  
  1927.         if (($fld_sent && $svr_allow_sent && !$lcl_allow_sent|| ($fld_sent && $lcl_allow_sent)) {
  1928.             if ($action == 'reply' || $action == 'reply_all'{
  1929.                 $save_reply_with_orig=getPref($data_dir,$username,'save_reply_with_orig');
  1930.                 if ($save_reply_with_orig{
  1931.                     $sent_folder $mailbox;
  1932.                 }
  1933.             }
  1934.             require_once(SM_PATH 'class/deliver/Deliver_IMAP.class.php');
  1935.             $imap_deliver new Deliver_IMAP();
  1936.             $imap_deliver->mail($composeMessage$imap_stream$reply_id$reply_ent_id$imap_stream$sent_folder);
  1937.             unset ($imap_deliver);
  1938.         }
  1939.  
  1940.  
  1941.         // final cleanup
  1942.         //
  1943.         $composeMessage->purgeAttachments();
  1944.         sqimap_logout($imap_stream);
  1945.  
  1946.     }
  1947.     return $success;
  1948. }

Documentation generated on Mon, 28 Jul 2014 04:17:27 +0200 by phpDocumentor 1.4.3