侯体宗的博客
  • 首页
  • Hyperf版
  • beego仿版
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

一步步打造简单的MVC电商网站BooksStore(3)

php  /  管理员 发布于 7年前   434

一步步打造一个简单的 MVC 电商网站 - BooksStore(三)

本系列的 GitHub地址:https://github.com/liqingwen2015/Wen.BooksStore

《一步步打造一个简单的 MVC 电商网站 - BooksStore(一)》

《一步步打造一个简单的 MVC 电商网站 - BooksStore(二)》

《一步步打造一个简单的 MVC 电商网站 - BooksStore(三)》

《一步步打造一个简单的 MVC 电商网站 - BooksStore(四)》

简介

上一节我们完成了两个主要功能:添加到购物车和分类导航,这一节我们会完成整个购物车的流程,以及订单处理。

该系列主要功能与知识点如下:

分类、产品浏览、购物车、结算、CRUD(增删改查) 管理、发邮件、分页、模型绑定、认证过滤器和单元测试等(预计剩余两篇,周三(因为周二不上班)先发布一篇)。

【备注】项目使用 VS2015 + C#6 进行开发,有问题请发表在留言区哦,还有,页面长得比较丑,请见谅。

目录

完成购物车

订单结算

一、完成购物车

上一节其实已经完成了移除购物车和清空购物车的方法,只是尚未将可供用户操作的按钮放在页面区域。除了增加这两个按钮,也会在页面顶部的位置增加购物车的摘要(用于显示用户的购物总额)。

下面是上一节已经写好的 CartController 代码。

/// <summary> /// 购物车 /// </summary> public class CartController : Controller { private readonly IBookRepository _bookRepository; public CartController(IBookRepository bookRepository) {  _bookRepository = bookRepository; } /// <summary> /// 首页 /// </summary> /// <param name="returnUrl"></param> /// <returns></returns> public ViewResult Index(string returnUrl) {  return View(new CartIndexViewModel()  {  Cart = GetCart(),  ReturnUrl = returnUrl  }); } /// <summary> /// 添加到购物车 /// </summary> /// <param name="id"></param> /// <param name="returnUrl"></param> /// <returns></returns> public RedirectToRouteResult AddToCart(int id, string returnUrl) {  var book = _bookRepository.Books.FirstOrDefault(x => x.Id == id);  if (book != null)  {  GetCart().AddBook(book, 1);  }  return RedirectToAction("Index", new { returnUrl }); } /// <summary> /// 从购物车移除 /// </summary> /// <param name="id"></param> /// <param name="returnUrl"></param> /// <returns></returns> public RedirectToRouteResult RemoveFromCart(int id, string returnUrl) {  var book = _bookRepository.Books.FirstOrDefault(x => x.Id == id);  if (book != null)  {  GetCart().RemoveBook(book);  }  return RedirectToAction("Index", new { returnUrl }); } /// <summary> /// 获取购物车 /// </summary> /// <returns></returns> private Cart GetCart() {  var cart = (Cart)Session["Cart"];  if (cart != null) return cart;  cart = new Cart();  Session["Cart"] = cart;  return cart; } }

1.加入移除书籍和清空购物车的功能

Index.cshtml

@model Wen.BooksStore.WebUI.Models.CartIndexViewModel<h2>我的购物车</h2><table class="table"> <thead> <tr>  <th>书名</th>  <th>价格</th>  <th>数量</th>  <th>总计</th>  <th> </th> </tr> </thead> <tbody> @foreach (var item in Model.Cart.GetCartItems) {  <tr>  <td>@item.Book.Name</td>  <td>@item.Book.Price</td>  <td>@item.Quantity</td>  <td>@((item.Book.Price * item.Quantity).ToString("C"))</td>  <td>   @using (Html.BeginForm("RemoveFromCart", "Cart"))   {   @Html.Hidden("id", item.Book.Id)   @Html.HiddenFor(x => x.ReturnUrl)   <input type="submit" value="- 移除" />   }  </td>  </tr> } <tr>  <td> </td>  <td> </td>  <td>总计:</td>  <td>@Model.Cart.ComputeTotalValue().ToString("C")</td>  <td>  @using (Html.BeginForm("Clear", "Cart"))  {   @Html.HiddenFor(x => x.ReturnUrl)   <input type="submit" value="清空购物车" />  }  </td> </tr> </tbody></table>

【备注】@Html.Hidden("id", item.Book.Id) 是用于生成隐藏的字段,如果直接使用@Html.HiddenFor(),生成的 name 将会是 item.Book.Id ,将和 CartController 中 RemoveFromCart(int id, string return) 的参数不匹配。

显示的效果如下:

2.添加摘要:我们在购物车存放了许多东西,通过摘要,可以显示购物总额的缩略图,我们选择的位置在顶部右上角的一个比较显眼的位置进行显示它,当然,还需要有点击的跳转按钮方便显示所有的购物清单页面。

继续在 CartController 下新增一个 Action,名为 Summary,返回值是一个分部视图:

/// <summary> /// 摘要 /// </summary> /// <returns></returns> public PartialViewResult Summary() {  return PartialView(GetCart()); }

对应的Summary.cshtml

@model Wen.BooksStore.Domain.Entities.Cart<div class="bookSummary"> 你的购物车:@Model.ComputeTotalValue()  <span>@Html.ActionLink("结算", "Checkout", "Cart", new { retunUrl = Request.Url.PathAndQuery }, null)</span></div>

对应的布局页_Layout.cshtml 修改的地方为:

_Layout.cshtml

<!DOCTYPE html><html><head> <meta name="viewport" content="width=device-width" /> <title>@ViewBag.Title</title> <link href="https:/article/~/Contents/Site.css" rel="stylesheet" /></head><body> <div id="header"> @{ Html.RenderAction("Summary", "Cart");} <div class="title">图书商城</div> </div> <div id="sideBar"> @{ Html.RenderAction("Sidebar", "Nav"); } </div> <div id="content"> @RenderBody() </div></body></html>

添加了新的东西,css 也要进行修改:

Site.css

body {}#header, #content, #sideBar { display: block;}#header { background-color: green; border-bottom: 2px solid #111; color: White;}#header, .title { font-size: 1.5em; padding: .5em;}#sideBar { float: left; width: 8em; padding: .3em;}#content { border-left: 2px solid gray; margin-left: 10em; padding: 1em;}.pager { text-align: right; padding: .5em 0 0 0; margin-top: 1em;} .pager A { font-size: 1.1em; color: #666; padding: 0 .4em 0 .4em; } .pager A:hover {  background-color: Silver; } .pager A.selected {  background-color: #353535;  color: White; }.item input { float: right; color: White; background-color: green;}.table { width: 100%; padding: 0; margin: 0;} .table th { font: bold 12px "Trebuchet MS", Verdana, Arial, Helvetica, sans-serif; color: #4f6b72; border-right: 1px solid #C1DAD7; border-bottom: 1px solid #C1DAD7; border-top: 1px solid #C1DAD7; letter-spacing: 2px; text-transform: uppercase; text-align: left; padding: 6px 6px 6px 12px; background: #CAE8EA no-repeat; } .table td { border-right: 1px solid #C1DAD7; border-bottom: 1px solid #C1DAD7; background: #fff; font-size: 14px; padding: 6px 6px 6px 12px; color: #4f6b72; } .table td.alt {  background: #F5FAFA;  color: #797268; } .table th.spec, td.spec { border-left: 1px solid #C1DAD7; }.bookSummary { width: 15%; float: right; margin-top: 1.5%;}

二、订单结算

购物完毕就是结算页面了,这里的订单结算并不涉及支付接口的调用,只是使用邮件的形式进行通知而已。

这里,我设计结算的时候需要要求用户输入一些信息,如姓名、地址和邮箱等信息,在点击确定时我再将这些输入的信息与购物清单的信息从系统的邮箱发到你所输入的邮箱当中。一个比较直观的图:

1.在 Entities 中添加一个域模型 Contact.cs 表示联系人的信息。

/// <summary> /// 联系信息 /// </summary> public class Contact { [Required(ErrorMessage = "姓名不能为空")] public string Name { get; set; } [Required(ErrorMessage = "地址不能为空")] public string Address { get; set; } [Required(ErrorMessage = "邮箱不能为空")] [RegularExpression(@"(\w+([-+.]\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*\w\w)", ErrorMessage = "输入的邮箱地址不合法")] public string Email { get; set; } }

CartController.cs 添加一个用于结算的 Action:

/// <summary> /// 结算 /// </summary> /// <returns></returns> public ViewResult Checkout() {  return View(new Contact()); }

Checkout.cshtml 中的:

@model Wen.BooksStore.Domain.Entities.Contact<div> @using (Html.BeginForm()) { <div class="error">@Html.ValidationSummary()</div> <div>姓名: @Html.TextBoxFor(x => x.Name)</div> <div>地址: @Html.TextBoxFor(x => x.Address)</div> <div>邮箱: @Html.TextBoxFor(x => x.Email)</div> <div><input type="submit" value="提交" /></div> }</div>

这里使用的是模型校验,_Layout.cshtml 布局页需要引入js:

<script src="https:/article/~/Scripts/jquery-1.10.2.js"></script><script src="https:/article/~/Scripts/jquery.validate.js"></script><script src="https:/article/~/Scripts/jquery.validate.unobtrusive.js"></script>

<!DOCTYPE html><html><head> <meta name="viewport" content="width=device-width" /> <title>@ViewBag.Title</title> <link href="https:/article/~/Contents/Site.css" rel="stylesheet" /> <script src="https:/article/~/Scripts/jquery-1.10.2.js"></script> <script src="https:/article/~/Scripts/jquery.validate.js"></script> <script src="https:/article/~/Scripts/jquery.validate.unobtrusive.js"></script></head><body> <div id="header"> @{ Html.RenderAction("Summary", "Cart");} <div class="title">图书商城</div> </div> <div id="sideBar"> @{ Html.RenderAction("Sidebar", "Nav"); } </div> <div id="content"> @RenderBody() </div></body></html>

尝试运行,会出现以下页面,如果信息不填的话会出现相关的错误提示:

2.接下来,要进入“提交”后的流程了。

现在还需要一个组件用于处理订单,创建一个用于订单处理的接口,和一个该接口的实现,再通过 Ninject 进行两者的绑定:

/// <summary> /// 订单处理 /// </summary> public interface IOrderProcessor { /// <summary> /// 处理订单 /// </summary> /// <param name="cart"></param> /// <param name="contact"></param> void ProcessOrder(Cart cart, Contact contact); }

建立一个实现该接口用于处理订单的实体类,这里并不是调用支付接口,而是简单通过 BCL 中的进行邮件的发送。

EmailOrderProcessor.cs:

/// <summary> /// 邮件订单处理器 /// </summary> public class EmailOrderProcessor : IOrderProcessor { /// <summary> /// 发送人 /// </summary> public static class Sender {  /// <summary>  /// 账号  /// </summary>  public static string Account = "你的@qq.com";  /// <summary>  /// 密码  /// </summary>  public static string Password = "xxx"; } /// <summary> /// 处理订单 /// </summary> /// <param name="cart"></param> /// <param name="contact"></param> public void ProcessOrder(Cart cart, Contact contact) {  if (string.IsNullOrEmpty(contact.Email))  {  throw new Exception("Email 不能为空!");  }  var sb = new StringBuilder();  foreach (var item in cart.GetCartItems)  {  sb.AppendLine($"《{item.Book.Name}》:{item.Book.Price} * {item.Quantity} = {item.Book.Price * item.Quantity}");  }  sb.AppendLine($"总额:{cart.GetCartItems.Sum(x => x.Quantity * x.Book.Price)}");  sb.AppendLine();  sb.AppendLine($"联系人:{contact.Name} {contact.Address}");  //设置发件人,发件人需要与设置的邮件发送服务器的邮箱一致  var fromAddr = new MailAddress(Sender.Account);  var message = new MailMessage { From = fromAddr };  //设置收件人,可添加多个,添加方法与下面的一样  message.To.Add(contact.Email);  //设置抄送人  message.CC.Add(Sender.Account);  //设置邮件标题  message.Subject = "您的订单正在出库...";  //设置邮件内容  message.Body = sb.ToString();  //设置邮件发送服务器,服务器根据你使用的邮箱而不同,可以到相应的 邮箱管理后台查看,下面是QQ的  var client = new SmtpClient("smtp.qq.com", 25)  {  Credentials = new NetworkCredential(Sender.Account, Sender.Password),  EnableSsl = true  };  //设置发送人的邮箱账号和密码  //启用ssl,也就是安全发送  //发送邮件  client.Send(message); }

CartController 也需要稍作调整:

还要在 CartController 中额外添加一个带[HttPost] 特性的名为 Checkout 方法:

/// <summary> /// 结算 /// </summary> /// <param name="contact"></param> /// <returns></returns> [HttpPost] public ViewResult Checkout(Contact contact) {  if (!ModelState.IsValid)  return View(contact);  var cart = GetCart();  _orderProcessor.ProcessOrder(cart, contact);  cart.Clear();  return View("Thanks"); }

当校验成功时,会调用接口发一条信息,并且清空已有的购物车,然后跳转到指定的一个新视图页:

新建 Thanks.cshtml,内容如下:

Thanks

别忘了添加绑定哦,使用 DI 容器将两者进行绑定:

启动页面,试试效果吧:

看来,好像成功了哦:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


  • 上一条:
    一步步打造简单的MVC电商网站BooksStore(4)
    下一条:
    一步步打造简单的MVC电商网站BooksStore(2)
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • 用Time Warden监控PHP中的代码处理时间(0个评论)
    • 在PHP中使用array_pop + yield实现读取超大型目录功能示例(0个评论)
    • Property Hooks RFC在PHP 8.4中越来越接近现实(0个评论)
    • 近期文章
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf分页文件功能(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 欧盟关于强迫劳动的规定的官方举报渠道及官方举报网站(0个评论)
    • 在go语言中使用github.com/signintech/gopdf实现生成pdf文件功能(0个评论)
    • Laravel从Accel获得5700万美元A轮融资(0个评论)
    • 在go + gin中gorm实现指定搜索/区间搜索分页列表功能接口实例(0个评论)
    • 在go语言中实现IP/CIDR的ip和netmask互转及IP段形式互转及ip是否存在IP/CIDR(0个评论)
    • PHP 8.4 Alpha 1现已发布!(0个评论)
    • Laravel 11.15版本发布 - Eloquent Builder中添加的泛型(0个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-06
    • 2017-07
    • 2017-08
    • 2017-09
    • 2017-11
    • 2017-12
    • 2018-01
    • 2018-02
    • 2018-03
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-09
    • 2021-02
    • 2021-03
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-11
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-11
    • 2023-12
    • 2024-01
    • 2024-02
    • 2024-03
    • 2024-04
    • 2024-05
    • 2024-06
    • 2024-07
    • 2024-09
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客