07 ноя 2010
Форма обратной связи с аттачем
Dmitrius Категория: Web-Мастеру » Уроки
У нас вы можете скачать бесплатно Форма обратной связи с аттачем

Данный материал предоставлен сайтом Skripter.info исключительно в ознакомительных целях. Администрация не несет ответственности за его содержимое.
Форма обратной связи с аттачем


В качестве аттача будем использовать прикрепленный к сообщению файл, выбранный на компьютере пользователя. Список поддерживаемых расширений файлов будет храниться в массиве $allowtypes. Все диагностические сообщения и настройки скрипта будем хранить в одном файле, который назовем mailer.php, в нём и напишем весь скрипт данного урока

Давайте рассмотрим основные моменты в исходном коде этого скрипта:

Основные настройки и диагностические сообщения:

error_reporting(7);
$websitename="Форма обратной связи с аттачем";
$allowtypes=array("zip", "rar", "txt", "doc", "jpg", "png", "gif","odt","xml");
$myemail="youremail@domain.ru";
$priority="3";
$allowattach="1";
$max_file_size="1024";
$max_file_total="2048";
$submitvalue="Отправить";
$resetvalue="Сброс";
$defaultsubject="Без темы";
$use_subject_drop=false;
$subjects=array("Часть 1","Часть 2","Часть 3");
$emails=array("dept_1@domain.com","dept_2@domain.com","dept_3@domain.com");
$thanksmessage="Спасибо! Ваше письмо было отправлено, мы ответим в ближайшее время.";


Настройте для своего случая соответствующие переменные: $myemail и $allowtypes для определения вашего e-mail-адреса и типов поддерживаемых расширений для приаттачиваемого файла. В переменной emails перечисляются дополнительные email-адреса для пересылки на них копий этого же сообщения.

Также особенного внимания заслуживают функции для обработки приаттачиваемого (присоединяемого) файла.

function get_ext($key){
    $key=strtolower(substr(strrchr($key, "."), 1));
    $key=str_replace("jpeg","jpg",$key);
    return $key;
}
function phattach($file,$name){
    global $boundary;    
    $fp=@fopen($file,"r");
    $str=@fread($fp, filesize($file));
    $str=@chunk_split(base64_encode($str));
    $message="--".$boundary."\n";
    $message.="Content-Type: application/octet-stream; name=\"".$name."\"\n";
    $message.="Content-disposition: attachment; filename=\"".$name."\"\n";
    $message.="Content-Transfer-Encoding: base64\n";
    $message.="\n";
    $message.="$str\n";
    $message.="\n";
    return $message;
}
function clean($key){
    $key=str_replace("\r", "", $key);
    $key=str_replace("\n", "", $key);
    $find=array("/bcc\:/i","/Content\-Type\:/i","/Mime\-Type\:/i","/cc\:/i","/to\:/i");
    $key=preg_replace($find,"",$key);
    return $key;
}


Функция get_ext преобразует символы в имени файла к нижнему регистру, а также заменяет расширения jpeg на jpg.

Функция phattach возвращает заголовок типа сообщения для последующей отправки на мыло текстовых файлов внешних приложений формата application/octet-stream. А функция clean находит и удаляет нежелательные символы в имени.

Для проверки и удаления нежелательных типов файлов служит следующая процедура проверки:

$error="";
$types="";
$sent_mail=false;
$ext_count=count($allowtypes);
$i=0;
foreach($allowtypes AS $extension) {    
    If($i <= $ext_count-2) {
        $types .="*.".$extension.", ";
    } Else {
        $types .="*.".$extension;
    }
    $i++;
}
unset($i,$ext_count);


Цикл foreach делает разбор указанных расширений файлов в массиве $allowtypes[] по типам в соответствии с необходимым условием.

Далее привожу исходный код цикла обработки и передачи заполненной пользователем формы на e-mail адрес, указанный в переменной $myemail.

If($_POST['submit']==true) {
    extract($_POST, EXTR_SKIP);    
        If(trim($yourname)=="") {
            $error.="Вы не ввели Ваше имя!
";
        }        
        If(trim($youremail)=="") {
            $error.="Вы не ввели Ваш email!
";
        } Elseif(!eregi("^([a-z0-9_]|\\-|\\.)+@(([a-z0-9_]|\\-)+\\.)+[a-z]{2,4}\$",$youremail)) {
            $error.="Неправильный адрес электронной почты
";
        }
        If(trim($emailsubject)=="") {
            $emailsubject=$defaultsubject;
        }
        If(trim($yourmessage)=="") {
            $error.="Вы не ввели Ваше сообщение!
";
        }        
        If($allowattach > 0) {            
            For($i=0; $i <= $allowattach-1; $i++) {                
                If($_FILES['attachment']['name'][$i]) {                    
                    $ext=get_ext($_FILES['attachment']['name'][$i]);
                    $size=$_FILES['attachment']['size'][$i];
                    $max_bytes=$max_file_size*1024;                    
                    If(!in_array($ext, $allowtypes)) {
                        
                        $error.= "Недопустимое расширение для вашего файла: ".$_FILES['attachment']['name'][$i].", only ".$types." are allowed.
";
                    } Elseif($size > $max_bytes) {
                        $error.= "Ваш файл: ".$_FILES['attachment']['name'][$i]." is to big. Max file size is ".$max_file_size."kb.
";
                    }                    
                }
                
            }            
              $total_size=array_sum($_FILES['attachment']['size']);
              
            $max_file_total_bytes=$max_file_total*1024;
            
            If($total_size > $max_file_total_bytes) {
                $error.="Максимальный допустимый размер вашего файла ".$max_file_total."kb
";
            }            
        }
    If($error) {
        $display_message=$error;
    } Else {        
        If($use_subject_drop AND is_array($subjects) AND is_array($emails)) {
            $subject_count=count($subjects);
            $email_count=count($emails);
            
            If($subject_count==$email_count) {            
                $myemail=$emails[$emailsubject];
                $emailsubject=$subjects[$emailsubject];            
            }        
        }        
        $boundary=md5(uniqid(time()));        
        $yourname=clean($yourname);
        $yourmessage=clean($yourmessage);
        $youremail=clean($youremail);
        $headers="From: ".$yourname." <".$youremail.">\n";
        $headers.="Reply-To: ".$yourname." <".$youremail.">\n";
        $headers.="MIME-Version: 1.0\n";
        $headers.="Content-Type: multipart/mixed; boundary=\"".$boundary."\"\n";
        $headers.="X-Sender: ".$_SERVER['REMOTE_ADDR']."\n";
        $headers.="X-Mailer: PHP/".phpversion()."\n";
        $headers.="X-Priority: ".$priority."\n";
        $headers.="Return-Path: <".$youremail.">\n";
        $headers.="This is a multi-part message in MIME format.\n";            
        $message = "--".$boundary."\n";
        $message.="Content-Type: text/plain; charset=\"windows-1251\"\n";
        $message.="Content-Transfer-Encoding: quoted-printable\n";
        $message.="\n";
        $message.="$yourmessage";
        $message.="\n";    
        If($allowattach > 0) {            
            For($i=0; $i <= $allowattach-1; $i++) {                
                If($_FILES['attachment']['name'][$i]) {                    
                    $message.=phattach($_FILES['attachment']['tmp_name'][$i],$_FILES['attachment']['name'][$i]);
                    
                }                
            }        
        }        
        $message.="--".$boundary."--\n";        
        If(!mail($myemail,$emailsubject,$message,$headers)) {            
            Exit("Произошла ошибка, пожалуйста, сообщите об этом администратору сайта.\n");            
        } Else {        
            $sent_mail=true;            
        }
    }


Работает так: Если переменная отправки $_POST['submit'] вернула положительный результат (true), то выполняем все действия. Иначе у нас будет распечатываться форма, которую напишем ниже. В самом начале мы извлекаем с помощью функции extract все переменные, которые нам необходимы для обработки формы, чтобы лишний раз не писать $_POST для каждой переменной. Это очень удобно, хотя не рекомендуется. Далее производим проверку введенных полей: yourname, youremail, emailsubject, yourmessage, allowattach. Если в них пусто, то формируем строковую переменную сообщения ошибки $error, которую мы определили как $error=""; перед циклом разбора типов файлов. Если ошибка возникнет, то выведится соответствующее сообщение. Для идентификации аттач-файла используем переменную $allowattach, если она имеет место быть, то в цикле перебора её значений формируем временые массивы для хранения параметров файла с присвоением их значений переменным: $ext, $size, $max_bytes, $total_size, $max_file_total_bytes. Также проверяем расширение и размер файла на допустимость, если не существует(массив) или превышает(размер), то присваиваем переменной ошибки соответствующее строковое значение.

Если ошибок в переменной $error нет, то формируем конечные переменные для использования в заголовке отправляемого письма $headers: $boundary, $yourname, $yourmessage, $youremail. Если выбран файл, то формируем и для него заголовок в виде: $message.=phattach($_FILES['attachment']['tmp_name'][$i],$_FILES['attachment']['name'][$i]);

Затем с помощью функции mail() отправляем сформированное письмо. Если функция не сработала, то выводим сообщение, что произошла ошибка. В случае положительного результата присваиваем переменной $sent_mail флаг положительного результата true, значение которой перед отправкой было установлено в false.

Всё. Теперь осталось только создать html-формочку для пущей наглядности. Но перед этим напишем небольшой клиентский скрипт для вывода сообщений об ошибке в стандартном диалоговом окне в случае неправильно введенных или оставленных пустых полей формы. Разместите этот код в вашем html-документе между тэгами ....

<script type="text/javascript">
var error="";
e_regex = /^\w+([\.-]?\w+)*@\w+([\.-]?\w+)*(\.\w{2,})+$/;
function Checkit(theform) {
    
    if(theform.yourname.value=="") {
        error+="Вы не ввели Ваше имя\n";
    }    
    if(theform.youremail.value=="") {
        error+="Вы не ввели Ваш email\n";
    } else if(!e_regex.test(theform.youremail.value)) {
        error+="Неправильный адрес электронной почты\n";
    }        
    if(theform.yourmessage.value=="") {
        error+="Вы не ввели Ваше сообщение\n";
    }    
    if(error) {
        alert('Произошли следующие ошибки:\n\n' + error);
        error="";
        return false;
    } else {
        return true;
    }
}
</script>


И наконец сама форма. Внешний вид можете оформить по своему вкусу. Я не стал приводить для неё стилевые таблицы, чтобы не отвлекать Вас от программерской части и логического вывода некоторых полей формы с выводимыми значениями строковых переменных.

<?If($display_message) {?>
<div align="center" class="error_message"><b><?=$display_message;?></b></div><br />
<?}?>
<?If($sent_mail!=true) {?>
<form method="post" action="<?=$_SERVER['PHP_SELF'];?>" enctype="multipart/form-data" name="phmailer" onsubmit="return Checkit(this);">
<table align="center" class="table">
<tr><td colspan="2" class="table_header" width="100%"><?=$websitename;?></td></tr>
<?If($allowattach > 0) {?>
<tr>
<td width="100%" class="attach_info" colspan="2">
<b>Поддерживаемые типы файлов:</b> <?=$types?><br />
<b>Максимальный размер файла:</b> <?=$max_file_size?>kb.<br />
<b>Максимальный размер архива:</b> <?=$max_file_total?>kb.</td>
</tr>
<?}?>
<tr><td width="30%" class="table_body">Ваше имя:</td>
<td width="70%" class="table_body"><input name="yourname" type="text" size="30" value="<?=stripslashes(htmlspecialchars($yourname));?>" /><span class="error_message">*</span></td>
</tr>
<tr>
<td width="30%" class="table_body">Ваш Email:</td>
<td width="70%" class="table_body"><input name="youremail" type="text" size="30" value="<?=stripslashes(htmlspecialchars($youremail));?>" /><span class="error_message">*</span></td>
</tr>
<tr><td width="30%" class="table_body">Тема:</td><td width="70%" class="table_body">
<?If($use_subject_drop AND is_array($subjects)) {?>
<select name="emailsubject" size="1">
<?while(list($key,$val)=each($subjects)) {?>
<option value="<?=intval($key);?>"><?=htmlspecialchars(stripslashes($val));?></option>
<?}?></select>
<?} Else {?>
<input name="emailsubject" type="text" size="30" value="<?=stripslashes(htmlspecialchars($emailsubject));?>" />
<?}?>
</td></tr>
<?For($i=1;$i <= $allowattach; $i++) {?>
<tr>
<td width="30%" class="table_body">Прикрепить файл:</td>
<td width="70%" class="table_body"><input name="attachment[]" type="file" size="30" /></td>
</tr>
<?}?>
<tr><td colspan="2" width="100%" class="table_body">Ваше сообщение:<span class="error_message">*</span><br />
<div align="center"><textarea name="yourmessage" rows="8" cols="60"><?=stripslashes(htmlspecialchars($yourmessage));?></textarea></div>
</td></tr>
<tr>
<td colspan="2" width="100%" class="table_footer">
<input type="hidden" name="submit" value="true" />
<input type="submit" value="<?=$submitvalue;?>" /> &nbsp;
<input type="reset" value="<?=$resetvalue;?>" />
</td></tr>
</table></form>
<?} Else {?>
<div align="center" class="thanks_message"><?=$thanksmessage;?></div>
<html><head><meta https-equiv='Refresh' content='1; URL=index.php'></head></html>
<?}?>


Думаю в ней особо комментировать нечего. Отмечу только то, что многие служебные переменные для вывода мы определили в самом начале скрипта. Также обратите внимание на наличие обязательного значения атрибута enctype в тэге
, если он не указан, то файл не будет отправляться! В качестве атрибута "action" указано значение глобального массива $_SERVER['PHP_SELF']. За работу клиентского скрипта для проверки заполнения полей отвечает значение обработчика "onsubmit", который возвращает вызов функции Checkit(this).

Очень важный момент: Не забудьте написать в форме скрытое поле с именем submit и значением true. Оно символизирует об отправке данных.

После успешной отправки сообщения выводиться переменная $thanksmessage и перезагружается(обновляется) старничка формы обратной связи.

Внимание! У вас нет прав для просмотра скрытого текста.









Уважаемый посетитель, Вы зашли на сайт как незарегистрированный пользователь.
Мы рекомендуем Вам зарегистрироваться либо войти на сайт под своим именем.
Информация
Посетители, находящиеся в группе Гости, не могут оставлять комментарии к данной публикации.