【一】概论
(1)简介:
这里说的邮件不是平时说的email邮件(邮件地址带有@符号的),而是指的一般论坛网站的站内信息,也叫私信或者pm(private message私信)
【二】站内信案例
(1)站内信构成:邮件的发送(写邮件)、邮件的接收(收件箱)、邮件的发件箱(发送的邮件列表)
(2)准备工作:
①数据表:sp_email
SQL语句:①后面的comment为注释,方便开发人员分辨;②附件这里数据库设计为只能上传一个附件
create table sp_email( id int(11) not null auto_increment, from_id int(11) not null comment'发送者id', to_id int(11) not null comment'接收者id', title varchar(50) not null comment'标题', file varchar(255) default null comment'文件', //完整路径 hasfile smallint(1) default '0' comment'是否有附件', //有文件为1,没有文件为0.默认0(没有附件) filename varchar(255) default null comment'文件原始名', content text comment'内容', addtime int(11) default null comment'添加时间', //发送时间 isread smallint(1) default '0' comment'是否已读', //是否已读,回值 primary key(id) )engine=myisam auto_increment=1 default charset=utf8;
②修改模板文件index.html,创建导航菜单(发邮件、邮件收件箱、已发邮件)
<li> <a href="javascript:;" class="emailManage">邮件管理</a> <ul> <li><a href="{:U('Email/send')}" class="documentManage">发邮件</a></li> <li><a href="{:U('Email/sendBox')}" class="documentManage">发件箱</a></li> <li><a href="{:U('Email/recBox')}" class="documentManage">收件箱</a></li> </ul> </li>
③创建控制器EmailController.class.php
<?php namespace AdminController; use ThinkController; class EmailController extends Controller{ } ?>
(3)邮件发送
分析
控制器:EmailController.class.php
方法:send(二合一,展示模板和数据保存)
模板:send.html
下面分步操作:
第一步:创建send方法,展示模板页面
<?php namespace AdminController; use ThinkController; class EmailController extends Controller{ //send方法,展示模板+数据保存 public function send(){ $this->display(); } } ?>
第二步:将send.html模板文件复制到指定目录下ApplicationAdminViewEmailsend.html,并替换静态资源路径
第三部:改写send方法,查询收件人,在模板中列出收件人列表
//send方法,展示模板+数据保存 public function send(){ //查询收件人信息,除去自己(id与session('id')相等的),因为自己不可以发送给自己 $data = M('user')->field('id,truename')->where('id != '.session('id'))->select(); $this->assign('data',$data); $this->display(); }
第四步:在模板里展示收件人列表
第五步:检查表单,表单完整代码
<form action="" method="post" enctype="multipart/form-data"> <fieldset> <legend>发送邮件</legend> <p> <label for="to_id">收件人:</label> <select name="to_id" id="to_id"> <option value="0">请选择收件人</option> <volist name="data" id="vol"> <option value="{$vol.id}">{$vol.truename}</option> </volist> </select> </p> <p><label for="title">标题:</label><input type="text" name="title" id="title"></p> <p><label for="file">附件:</label><input type="file" name="file" id="file"></p> <p><label for="author">内容:</label><textarea name="content"></textarea></p> <div> <a href="javascript:;" id="submitBtn">提交</a> <a href="javascript:;" id="resetBtn">清空</a> </div> </fieldset> </form>
第六步:将提交数据保存到数据表
先判断请求类型,若为POST请求,则处理数据,否则展示数据和模板页面。先将大概结构写出来
public function send(){ //判断请求类型 if (IS_POST) { //接收数据 $post = I('post.'); //实例化自定义模型 $model = D('Email'); //调用具体类的方法实现数据的保存 $result = $model -> addData($post,$_FILES('file')); if($result){ $this->success('邮件发送成功',U('sendBox'),3); }else{ $this->error('邮件发送失败'); } }else{ //查询收件人信息,除去自己(id与session('id')相等的) $data = M('user')->field('id,truename')->where('id != '.session('id'))->select(); $this->assign('data',$data); $this->display(); } }
第七步:创建自定义模型EmailModel.class.php,添加保存方法addData。先列出自定义模型的方法架构
<?php namespace AdminModel; use ThinkModel; class EmailModel extends Model{ public function addData($post,$file){ } } ?>
第八步:编写需要的addData方法实现数据处理和保存入库(文件处理+数据处理)
//文件分为文件+字符 if(!$file['error']){//判断是否有文件上传 //1. 配置数组,定义配置 $cfg = array( //配置上传路径 'rootPath' => WORKING_PATH . UPLOAD_ROOT_PATH ); //2. 实例化上传类 $upload = new ThinkUpload($cfg); //3. 上传操作,并接受上传结果 $info = $upload->uploadOne($file); dump($info);die; }else{ }
先判断上传是否成功,上传成功的话会输出有9个元素的一维数组
接下来判断是否上传成功,若成功则继续处理补全字段。完整代码
<?php namespace AdminModel; use ThinkModel; class EmailModel extends Model{ public function addData($post,$file){ //文件分为文件+字符 if(!$file['error']){//判断是否有文件上传 //1. 配置数组,定义配置 $cfg = array( //配置上传路径 'rootPath' => WORKING_PATH . UPLOAD_ROOT_PATH ); //2. 实例化上传类 $upload = new ThinkUpload($cfg); //3. 上传操作,并接受上传结果 $info = $upload->uploadOne($file); if ($info) {//成功后补全字段 //处理数据表中需要的字段file、hasfile、filename $post['file'] = UPLOAD_ROOT_PATH . $info['savepath'] . $info['savename'];//文件在磁盘上的存储路径 $post['hasfile'] = 1;//表示是否有文件,能走进来说明肯定有文件,所以为1 $post['filename'] = $info['name'];//文件的原始名称 }else{} // 补全字段from_id和addtime $post['from_id'] = session('id');//发件人id $post['addtime'] = time();//发送时间 //数据保存 return $this->add($post); }else{ } } } ?>
效果图:
数据成功保存到数据表
(4)发件箱
分析:
控制器:EmailController.class.php
方法:sendBox
模板:sendBox.html
下面分步进行
第一步:创建方法sendBox,获取列表数据展示数据和模板文件
注意:发件箱里需要显示出收件人的名字(数据表里存的是from_id收件人的id,所以需要联表查询数据)
主表:sp_email(别名t1) ;从表:sp_user(别名t2)
联表条件:t1.to_id = t2.id
原生SQL:(两个方法table和join,table相当于join的inner内联,这里用table)
注意:table方法后跟两个表名,join方法后跟一个表名(左联)
select t1.*,t2.truename as truename from sp_email as t1 left join sp_user as t2 on t1.to_id=t2.id where t1.from_id=当前用户id;
将上述SQL代码复制到Navicate中执行结果
接下来参考上述代码到ThinkPHP里去执行连贯操作,然后输出打印下查询结果
浏览器测试后,显示出查询数据。接着将数据传递给模板,代码
//sendBox方法,发件箱 public function sendBox(){ //联表查询当前用户已经发送的邮件 $data = M('Email')->field('t1.*,t2.truename as truename')->alias('t1')->join('left join sp_user as t2 on t1.to_id=t2.id')->
where('t1.from_id = '.session('id'))->select(); $this->assign('data',$data); $this->display(); }
第二步:复制模板文件sendBox.html到指定位置,并换掉静态资源路径
<table border="1" cellspacing="0" cellpadding="10"> <thead> <tr><td>序号</td><td>收件人</td><td>标题</td><td>附件</td><td>内容</td><td>发送时间</td><td>状态</td><td>操作</td></tr> </thead> <volist name="data" id="vol"> <tr> <td>{$vol.id}</td> <td>{$vol.truename}</td> <td>{$vol.title|msubstr=###,0,10}</td> <td>{$vol.filename|msubstr=###,0,16}
<notempty name="vol.filename"><a href="__CONTROLLER__/download/id/{$vol.id}">【下载】</a></notempty>
</td> <td>{$vol.content|msubstr=###,0,10}</td> <td>{$vol.addtime|date='Y-m-d H:i:s',###}</td> <td>
<if condition="$vol.isread == 0"><span style="color:red">未读</span><else/><span style="color:green">已读</span></if>
</td> <td> <a href="javascript:;" class="showBtn" data="{$vol.id}"" data-title="{$vol.title}">查看</a> | <a href="__CONTROLLER__/edit/id/{$vol.id}" class="editBtn">删除</a> </td> </tr> </volist> </table>
显示效果:
注意:
1. 为了区别读取状态,这里我需要加判断。如果isread为0则表示未读,否则表示已读;
2. 为了测试效果,我去Navicate里将isread手动修改为1,然后刷新浏览器测试(为了直观,可以在模板里加上颜色)
第三步:添加附件下载功能
//download下载附件 public function download(){ //接收id $id = I('get.id'); //查询数据 $data = M('Email') -> find($id); //下载代码,主要需要的是路径,接下来拼接路径 $file = WORKING_PATH . $data['file']; //输出文件 header("Content-type: application/octet-stream");//文件流,告诉浏览器输出的东西是文件流 header('Content-Disposition: attachment; filename="' . basename($file) . '"');//文件名 header("Content-Length: ". filesize($file));//文件大小 //输出缓冲区 readfile($file); }