[TOC]
## 步骤 1 : 先运行,看到效果,再学习
先将完整的 tmall_ssm 项目(向老师要相关资料),配置运行起来,确认可用之后,再学习做了哪些步骤以达到这样的效果。
## 步骤 2 : 模仿和排错
在确保可运行项目能够正确无误地运行之后,再严格照着教程的步骤,对代码模仿一遍。
模仿过程难免代码有出入,导致无法得到期望的运行结果,此时此刻通过比较**正确答案** ( 可运行项目 ) 和自己的代码,来定位问题所在。
采用这种方式,**学习有效果,排错有效率**,可以较为明显地提升学习速度,跨过学习路上的各个槛。
## 步骤 3 : 界面效果
访问如下地址:
`http://127.0.0.1:8080/tmall_ssm/foreproduct?pid=844`
![](https://box.kancloud.cn/7c27cfe9db97344073b7d0abb2fdf8e5_1422x6743.png)
目录结构图:
![](https://box.kancloud.cn/c40a5a46f5963f02b1f06309fa21618f_337x666.png)
## 步骤 4 : Product
修改Product,增加如下属性
1. 单个产品图片集合
`private List<ProductImage> productSingleImages;`
2. 详情产品图片集合
`private List<ProductImage> productDetailImages;`
3. 销量
`private int saleCount;`
4. 累计评价
`private int reviewCount;`
~~~
package com.dodoke.tmall.pojo;
import java.util.Date;
import java.util.List;
public class Product {
private Integer id;
private String name;
private String subTitle;
private Float originalPrice;
private Float promotePrice;
private Integer stock;
private Date createDate;
private Integer categoryId;
/* 非数据库字段 */
private Category category;
private ProductImage firstProductImage;
private List<ProductImage> productSingleImages;
private List<ProductImage> productDetailImages;
private int saleCount;
private int reviewCount;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name == null ? null : name.trim();
}
public String getSubTitle() {
return subTitle;
}
public void setSubTitle(String subTitle) {
this.subTitle = subTitle == null ? null : subTitle.trim();
}
public Float getOriginalPrice() {
return originalPrice;
}
public void setOriginalPrice(Float originalPrice) {
this.originalPrice = originalPrice;
}
public Float getPromotePrice() {
return promotePrice;
}
public void setPromotePrice(Float promotePrice) {
this.promotePrice = promotePrice;
}
public Integer getStock() {
return stock;
}
public void setStock(Integer stock) {
this.stock = stock;
}
public Date getCreateDate() {
return createDate;
}
public void setCreateDate(Date createDate) {
this.createDate = createDate;
}
public Integer getCategoryId() {
return categoryId;
}
public void setCategoryId(Integer categoryId) {
this.categoryId = categoryId;
}
public Category getCategory() {
return category;
}
public void setCategory(Category category) {
this.category = category;
}
public ProductImage getFirstProductImage() {
return firstProductImage;
}
public void setFirstProductImage(ProductImage firstProductImage) {
this.firstProductImage = firstProductImage;
}
public List<ProductImage> getProductSingleImages() {
return productSingleImages;
}
public void setProductSingleImages(List<ProductImage> productSingleImages) {
this.productSingleImages = productSingleImages;
}
public List<ProductImage> getProductDetailImages() {
return productDetailImages;
}
public void setProductDetailImages(List<ProductImage> productDetailImages) {
this.productDetailImages = productDetailImages;
}
public int getSaleCount() {
return saleCount;
}
public void setSaleCount(int saleCount) {
this.saleCount = saleCount;
}
public int getReviewCount() {
return reviewCount;
}
public void setReviewCount(int reviewCount) {
this.reviewCount = reviewCount;
}
}
~~~
## 步骤 5 : Review
修改自动生成的评价类Review, 增加User属性
`private User user;`
~~~
package com.dodoke.tmall.pojo;
import java.util.Date;
public class Review {
private Integer id;
private String content;
private Date createDate;
private Integer productId;
private Integer userId;
/* 非数据库字段 */
private User user;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getContent() {
return content;
}
public void setContent(String content) {
this.content = content == null ? null : content.trim();
}
public Date getCreateDate() {
return createDate;
}
public void setCreateDate(Date createDate) {
this.createDate = createDate;
}
public Integer getProductId() {
return productId;
}
public void setProductId(Integer productId) {
this.productId = productId;
}
public Integer getUserId() {
return userId;
}
public void setUserId(Integer userId) {
this.userId = userId;
}
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
}
~~~
## 步骤 6 : ReviewService
增加ReviewService,提供CURD以及通过产品获取评价方法:
List list(int pid);
int getCount(int pid);
~~~
package com.dodoke.tmall.service;
import java.util.List;
import com.dodoke.tmall.pojo.Review;
public interface ReviewService {
void add(Review c);
void delete(int id);
void update(Review c);
Review get(int id);
/**
* 通过产品获取评价
* @param pid 产品id
* @return List
*/
List list(int pid);
/**
* 获取产品有多少评价
* @param pid 产品id
* @return int
*/
int getCount(int pid);
}
~~~
## 步骤 7 : User
修改User, 增加一个getAnonymousName方法,用于在显示评价者的时候,进行匿名显示
~~~
package com.dodoke.tmall.pojo;
public class User {
private Integer id;
private String name;
private String password;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name == null ? null : name.trim();
}
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password == null ? null : password.trim();
}
/**
* 在显示评价者的时候进行匿名显示
*
* @return String
*/
public String getAnonymousName() {
if (null == name)
return null;
if (name.length() <= 1)
return "*";
if (name.length() == 2)
return name.substring(0, 1) + "*";
char[] cs = name.toCharArray();
for (int i = 1; i < cs.length - 1; i++) {
cs[i] = '*';
}
return new String(cs);
}
}
~~~
## 步骤 8 : ReviewServiceImpl
增加ReviewServiceImpl,实现对应方法
~~~
package com.dodoke.tmall.service.impl;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import com.dodoke.tmall.mapper.ReviewMapper;
import com.dodoke.tmall.pojo.Review;
import com.dodoke.tmall.pojo.ReviewExample;
import com.dodoke.tmall.pojo.User;
import com.dodoke.tmall.service.ReviewService;
import com.dodoke.tmall.service.UserService;
@Service
public class ReviewServiceImpl implements ReviewService {
@Autowired
ReviewMapper reviewMapper;
@Autowired
UserService userService;
@Override
public void add(Review c) {
reviewMapper.insert(c);
}
@Override
public void delete(int id) {
reviewMapper.deleteByPrimaryKey(id);
}
@Override
public void update(Review c) {
reviewMapper.updateByPrimaryKeySelective(c);
}
@Override
public Review get(int id) {
return reviewMapper.selectByPrimaryKey(id);
}
public List<Review> list(int pid) {
ReviewExample example = new ReviewExample();
example.createCriteria().andProductIdEqualTo(pid);
example.setOrderByClause("id desc");
List<Review> result = reviewMapper.selectByExample(example);
setUser(result);
return result;
}
public void setUser(List<Review> reviews) {
for (Review review : reviews) {
setUser(review);
}
}
private void setUser(Review review) {
int uid = review.getUserId();
User user = userService.get(uid);
review.setUser(user);
}
@Override
public int getCount(int pid) {
return list(pid).size();
}
}
~~~
## 步骤 9 : OrderItemService
修改OrderItemService,增加根据产品获取销售量的方法:
`int getSaleCount(int pid);`
~~~
package com.dodoke.tmall.service;
import java.util.List;
import com.dodoke.tmall.pojo.Order;
import com.dodoke.tmall.pojo.OrderItem;
public interface OrderItemService {
void add(OrderItem c);
void delete(int id);
void update(OrderItem c);
OrderItem get(int id);
List list();
void fill(List<Order> os);
void fill(Order o);
int getSaleCount(int pid);
}
~~~
## 步骤 10 : OrderItemServiceImpl
修改OrderItemServiceImpl,实现getSaleCount方法
~~~
package com.dodoke.tmall.service.impl;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import com.dodoke.tmall.mapper.OrderItemMapper;
import com.dodoke.tmall.pojo.Order;
import com.dodoke.tmall.pojo.OrderItem;
import com.dodoke.tmall.pojo.OrderItemExample;
import com.dodoke.tmall.pojo.Product;
import com.dodoke.tmall.service.OrderItemService;
import com.dodoke.tmall.service.ProductService;
@Service
public class OrderItemServiceImpl implements OrderItemService {
@Autowired
OrderItemMapper orderItemMapper;
@Autowired
ProductService productService;
@Override
public void add(OrderItem c) {
orderItemMapper.insert(c);
}
@Override
public void delete(int id) {
orderItemMapper.deleteByPrimaryKey(id);
}
@Override
public void update(OrderItem c) {
orderItemMapper.updateByPrimaryKeySelective(c);
}
@Override
public OrderItem get(int id) {
OrderItem result = orderItemMapper.selectByPrimaryKey(id);
setProduct(result);
return result;
}
public List<OrderItem> list() {
OrderItemExample example = new OrderItemExample();
example.setOrderByClause("id desc");
return orderItemMapper.selectByExample(example);
}
@Override
public void fill(List<Order> os) {
// 遍历每个订单,然后挨个调用fill(Order order)。
for (Order o : os) {
fill(o);
}
}
public void fill(Order o) {
// 1. 根据订单id查询出其对应的所有订单项
OrderItemExample example = new OrderItemExample();
example.createCriteria().andOrderIdEqualTo(o.getId());
example.setOrderByClause("id desc");
List<OrderItem> ois = orderItemMapper.selectByExample(example);
// 2. 通过setProduct为所有的订单项设置Product属性
setProduct(ois);
float total = 0;
int totalNumber = 0;
// 3. 遍历所有的订单项,然后计算出该订单的总金额和总数量
for (OrderItem oi : ois) {
total += oi.getNumber() * oi.getProduct().getPromotePrice();
totalNumber += oi.getNumber();
}
o.setTotal(total);
o.setTotalNumber(totalNumber);
// 4. 最后再把订单项设置在订单的orderItems属性上。
o.setOrderItems(ois);
}
public void setProduct(List<OrderItem> ois) {
for (OrderItem oi : ois) {
setProduct(oi);
}
}
private void setProduct(OrderItem oi) {
Product p = productService.get(oi.getProductId());
oi.setProduct(p);
}
@Override
public int getSaleCount(int pid) {
OrderItemExample example = new OrderItemExample();
example.createCriteria().andProductIdEqualTo(pid).andOrderIdIsNotNull();
List<OrderItem> ois = orderItemMapper.selectByExample(example);
int result = 0;
for (OrderItem oi : ois) {
result += oi.getNumber();
}
return result;
}
}
~~~
> 销量显示问题
把商品加入购物车,还没完成付款,而产品的销量理应完成付款才算一笔销量。应该在OrderItemExample后面加上`.andOidIsNotNull()`这个条件。
## 步骤 11 : ProductService
修改ProductService,增加为产品设置销量和评价数量的方法:
~~~
void setSaleAndReviewNumber(Product p);
void setSaleAndReviewNumber(List<Product> ps);
~~~
~~~
package com.dodoke.tmall.service;
import java.util.List;
import com.dodoke.tmall.pojo.Category;
import com.dodoke.tmall.pojo.Product;
public interface ProductService {
void add(Product c);
void delete(int id);
void update(Product c);
Product get(int id);
List list(int categoryId);
void setFirstProductImage(Product p);
/**
* 为分类填充产品集合
*
* @param categorys
*/
public void fill(List<Category> categorys);
/**
* 为多个分类填充产品集合
*
* @param category
*/
public void fill(Category category);
/**
* 为多个分类填充推荐产品集合,即把分类下的产品集合,按照8个为一行,拆成多行,以利于后续页面上进行显示
*
* @param categorys
*/
public void fillByRow(List<Category> categorys);
/**
* 为产品设置销量和评价数量
* @param p 产品对象
*/
void setSaleAndReviewNumber(Product p);
void setSaleAndReviewNumber(List<Product> ps);
}
~~~
## 步骤 12 : ProductServiceImpl
修改ProductServiceImpl,实现为产品设置销量和评价数量的方法:
void setSaleAndReviewNumber(Product p);
void setSaleAndReviewNumber
~~~
package com.dodoke.tmall.service.impl;
import java.util.ArrayList;
import java.util.Date;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import com.dodoke.tmall.mapper.ProductMapper;
import com.dodoke.tmall.pojo.Category;
import com.dodoke.tmall.pojo.Product;
import com.dodoke.tmall.pojo.ProductExample;
import com.dodoke.tmall.pojo.ProductImage;
import com.dodoke.tmall.service.CategoryService;
import com.dodoke.tmall.service.OrderItemService;
import com.dodoke.tmall.service.ProductImageService;
import com.dodoke.tmall.service.ProductService;
import com.dodoke.tmall.service.ReviewService;
@Service
public class ProductServiceImpl implements ProductService {
@Autowired
ProductMapper productMapper;
@Autowired
CategoryService categoryService;
@Autowired
ProductImageService productImageService;
@Autowired
OrderItemService orderItemService;
@Autowired
ReviewService reviewService;
@Override
public void add(Product p) {
p.setCreateDate(new Date());
productMapper.insert(p);
}
@Override
public void delete(int id) {
productMapper.deleteByPrimaryKey(id);
}
@Override
public void update(Product p) {
productMapper.updateByPrimaryKeySelective(p);
}
@Override
public Product get(int id) {
Product p = productMapper.selectByPrimaryKey(id);
setFirstProductImage(p);
setCategory(p);
return p;
}
private void setCategory(Product p) {
int categoryId = p.getCategoryId();
Category category = categoryService.get(categoryId);
p.setCategory(category);
}
@Override
public List list(int categoryId) {
ProductExample example = new ProductExample();
example.createCriteria().andCategoryIdEqualTo(categoryId);
example.setOrderByClause("id desc");
List result = productMapper.selectByExample(example);
setFirstProductImage(result);
setCategory(result);
return result;
}
public void setCategory(List<Product> ps) {
for (Product p : ps) {
setCategory(p);
}
}
/**
* 根据productId和图片类型查询出所有的单个图片,然后把第一个取出来放在firstProductImage上。
*
* @param p
* 产品
*/
@Override
public void setFirstProductImage(Product p) {
List<ProductImage> pis = productImageService.list(p.getId(), ProductImageService.type_single);
if (!pis.isEmpty()) {
ProductImage pi = pis.get(0);
p.setFirstProductImage(pi);
}
}
/**
* 给多个产品设置图片
*
* @param ps
* 产品集合
*/
public void setFirstProductImage(List<Product> ps) {
for (Product p : ps) {
setFirstProductImage(p);
}
}
/**
* 为分类填充产品集合
*
* @param categorys
*/
@Override
public void fill(Category c) {
List<Product> ps = list(c.getId());
c.setProducts(ps);
}
/**
* 为多个分类填充产品集合
*
* @param category
*/
@Override
public void fill(List<Category> cs) {
for (Category c : cs) {
fill(c);
}
}
/**
* 为多个分类填充推荐产品集合,即把分类下的产品集合,按照8个为一行,拆成多行,以利于后续页面上进行显示
*
* @param categorys
*/
@Override
public void fillByRow(List<Category> cs) {
// 把分类下的产品集合,按照8个为一行,拆成多行,以利于后续页面上进行显示
int productNumberEachRow = 8;
// 将categorylist中每个category拿出来循环
for (Category c : cs) {
// 获取每个分类中对应的产品,在使用fillByRow(List<Category>
// cs)这个方法前,需要先使用fill方法,注入分类中的所有产品,因此在这里才可以取出产品
List<Product> products = c.getProducts();
// 设置每一页的头图
setFirstProductImage(products);
// 每一行产品的list
List<List<Product>> productsByRow = new ArrayList<>();
for (int i = 0; i < products.size(); i += productNumberEachRow) {
int size = i + productNumberEachRow;
// 界限判断
size = size > products.size() ? products.size() : size;
// 该方法返回的是父list的一个子集合,从fromIndex(包含),到toIndex(不包含)
List<Product> productsOfEachRow = products.subList(i, size);
productsByRow.add(productsOfEachRow);
}
c.setProductsByRow(productsByRow);
}
}
@Override
public void setSaleAndReviewNumber(Product p) {
int saleCount = orderItemService.getSaleCount(p.getId());
p.setSaleCount(saleCount);
int reviewCount = reviewService.getCount(p.getId());
p.setReviewCount(reviewCount);
}
@Override
public void setSaleAndReviewNumber(List<Product> ps) {
for (Product p : ps) {
setSaleAndReviewNumber(p);
}
}
}
~~~
## 步骤 13 :`ForeController.product()`
通过访问地址
`http://127.0.0.1:8080/tmall_ssm/foreproduct?pid=844`
导致`ForeController.product() `方法被调用
1. 获取参数pid
2. 根据pid获取Product 对象p
3. 根据对象p,获取这个产品对应的单个图片集合
4. 根据对象p,获取这个产品对应的详情图片集合
5. 获取产品的所有属性值
6. 获取产品对应的所有的评价
7. 设置产品的销量和评价数量
8. 把上述取值放在request属性上
9. 服务端跳转到 "product.jsp" 页面
~~~
package com.dodoke.tmall.controller;
import java.util.List;
import javax.servlet.http.HttpSession;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.util.HtmlUtils;
import com.dodoke.tmall.pojo.Category;
import com.dodoke.tmall.pojo.Product;
import com.dodoke.tmall.pojo.ProductImage;
import com.dodoke.tmall.pojo.PropertyValue;
import com.dodoke.tmall.pojo.Review;
import com.dodoke.tmall.pojo.User;
import com.dodoke.tmall.service.CategoryService;
import com.dodoke.tmall.service.OrderItemService;
import com.dodoke.tmall.service.OrderService;
import com.dodoke.tmall.service.ProductImageService;
import com.dodoke.tmall.service.ProductService;
import com.dodoke.tmall.service.PropertyValueService;
import com.dodoke.tmall.service.ReviewService;
import com.dodoke.tmall.service.UserService;
@Controller
@RequestMapping("")
public class ForeController {
@Autowired
CategoryService categoryService;
@Autowired
ProductService productService;
@Autowired
UserService userService;
@Autowired
ProductImageService productImageService;
@Autowired
PropertyValueService propertyValueService;
@Autowired
OrderService orderService;
@Autowired
OrderItemService orderItemService;
@Autowired
ReviewService reviewService;
@RequestMapping("forehome")
public String home(Model model) {
List<Category> cs = categoryService.list();
productService.fill(cs);
productService.fillByRow(cs);
model.addAttribute("cs", cs);
return "fore/home";
}
@RequestMapping("foreregister")
public String register(Model model, User user) {
String name = user.getName();
// 把账号里的特殊符号进行转义
name = HtmlUtils.htmlEscape(name);
user.setName(name);
boolean exist = userService.isExist(name);
if (exist) {
String m = "用户名已经被使用,不能使用";
model.addAttribute("msg", m);
model.addAttribute("user", null);
return "fore/register";
}
userService.add(user);
return "redirect:registerSuccessPage";
}
@RequestMapping("forelogin")
public String login(@RequestParam("name") String name, @RequestParam("password") String password, Model model,
HttpSession session) {
name = HtmlUtils.htmlEscape(name);
User user = userService.get(name, password);
if (null == user) {
model.addAttribute("msg", "账号密码错误");
return "fore/login";
}
session.setAttribute("user", user);
return "redirect:forehome";
}
@RequestMapping("forelogout")
public String logout(HttpSession session) {
session.removeAttribute("user");
return "redirect:forehome";
}
@RequestMapping("foreproduct")
public String product(int pid, Model model) {
Product p = productService.get(pid);
// 根据对象p,获取这个产品对应的单个图片集合
List<ProductImage> productSingleImages = productImageService.list(p.getId(), ProductImageService.type_single);
// 根据对象p,获取这个产品对应的详情图片集合
List<ProductImage> productDetailImages = productImageService.list(p.getId(), ProductImageService.type_detail);
p.setProductSingleImages(productSingleImages);
p.setProductDetailImages(productDetailImages);
// 获取产品的所有属性值
List<PropertyValue> pvs = propertyValueService.list(p.getId());
// 获取产品对应的所有的评价
List<Review> reviews = reviewService.list(p.getId());
// 设置产品的销量和评价数量
productService.setSaleAndReviewNumber(p);
model.addAttribute("reviews", reviews);
model.addAttribute("p", p);
model.addAttribute("pvs", pvs);
return "fore/product";
}
}
~~~
## 步骤 14 : product.jsp
与 register.jsp 相仿,product.jsp也包含了header.jsp, top.jsp, simpleSearch.jsp, footer.jsp 等公共页面。
中间是产品业务页面 productPage.jsp
~~~
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@include file="../include/fore/header.jsp"%>
<%@include file="../include/fore/top.jsp"%>
<%@include file="../include/fore/simpleSearch.jsp"%>
<%@include file="../include/fore/product/productPage.jsp"%>
<%@include file="../include/fore/footer.jsp"%>
~~~
## 步骤 15 : productPage.jsp
productPage.jsp 又由3个页面组成
1. imgAndInfo.jsp
单个图片和基本信息
2. productReview.jsp
评价信息
3. productDetail.jsp
详情图片
~~~
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<title>模仿天猫官网 ${p.name}</title>
<div class="categoryPictureInProductPageDiv">
<img class="categoryPictureInProductPage" src="img/category/${p.category.id}.jpg">
</div>
<div class="productPageDiv">
<%@include file="imgAndInfo.jsp" %>
<%@include file="productReview.jsp" %>
<%@include file="productDetail.jsp" %>
</div>
~~~
## 步骤 16 : imgAndInfo.jsp
1. 左侧显示5张单个图片
(1)默认显示第一张图片
~~~
<img src="img/productSingle/${p.firstProductImage.id}.jpg" class="bigImg">
~~~
(2)5张小图片
~~~
<c:forEach items="${p.productSingleImages}" var="pi">
<img src="img/productSingle_small/${pi.id}.jpg" bigImageURL="img/productSingle/${pi.id}.jpg" class="smallImage">
</c:forEach>
~~~
2. 右边显示基本信息
(1)标题和小标题
~~~
<div class="productTitle">
${p.name}
</div>
<div class="productSubTitle">
${p.subTitle}
</div>
~~~
(2)原始价格和促销价
~~~
<fmt:formatNumber type="number" value="${p.originalPrice}" minFractionDigits="2"/>
<fmt:formatNumber type="number" value="${p.promotePrice}" minFractionDigits="2"/>
~~~
(3)销量和累计评价
~~~
<div>销量 <span class="redColor boldWord"> ${p.saleCount }</span></div>
<div>累计评价 <span class="redColor boldWord"> ${p.reviewCount}</span></div>
~~~
(4)库存
`<span>库存${p.stock}件</span>`
![](https://box.kancloud.cn/3cecdadc2860774fdd9aa79d13b6a110_1000x531.png)
> 其中模态框在下一个章节讲解
### 显示缩略图效果js代码讲解
~~~
$("img.smallImage").mouseenter(function(){
var bigImageURL = $(this).attr("bigImageURL");
$("img.bigImg").attr("src",bigImageURL);
});
~~~
首先在小图片上有一个自定义属性bigImageURL,用于存放对应的图片的位置
~~~
<img width="100px" class="smallImage" src="img/productSingle_small/8620.jpg" bigImageURL="img/productSingle/8620.jpg">
~~~
监听小图片的mouseenter事件,获取小图片的bigImageURL属性,把大图片的src修改为该图片
预加载,因为图片比较大,所以需要进行预加载。
~~~
$("img.bigImg").load(
function(){
$("img.smallImage").each(function(){
var bigImageURL = $(this).attr("bigImageURL");
img = new Image();
img.src = bigImageURL;
img.onload = function(){
console.log(bigImageURL);
$("div.img4load").append($(img));
};
});
}
);
~~~
> 当指定的元素(及子元素)已加载时,会发生 `load() `事件。
根据不同的浏览器(Firefox 和 IE),如果图像已被缓存,则也许不会触发 load 事件
在大图片加载好之后,根据每个小图片的bigImageURL ,创建一个Image对象,然后把这个image对象的src属性,设置为bigImageURL。 当这个img对象加载完毕之后,再放到被隐藏的`div.img4load`中,从而达到预加载的效果。
>小图片和大图片的img,分别使用的是不同的图片。
> 小图片本身就是分辨率低的图片,直接使用小图的src,放大显示就会看上去很糊了。
>
> 预加载是这样的:如果不使用预加载,那么当上面的主图片位置显示大图片的时候,浏览器"才"会到服务器去取大图片,而大图片加载比较慢,会影响用户体验。
>
> 使用了预加载,就是用户没有把鼠标移动到小图片之前,浏览器已经去把大图片加载了,当上面的主图片位置显示大图片的时候,直接就显示了,用户不会感受到加载的时间,感觉一下就显示出来了。
>
> 预加载的技术是通过js创建Image对象来实现,创建 Image对象,并设置其src,会导致浏览器去服务端获取图片,从而达到在未使用前,事先加载大图片的缓存效果。 真正需要显示大图片的时候,用户就觉得大图片一下就出来了,不会感觉到大图片加载的卡顿。
### 修改价格
1. 效果:
可以向上调整数量,但是不能超过最大库存
可以向下调整数量,但是不能小于1
输入任何非数字,都会恢复为原来数字
输入的数字超过库存,恢复为最大库存
2. js代码讲解
~~~
$(".productNumberSetting").keyup(function(){
var num= $(".productNumberSetting").val();
num = parseInt(num);
if(isNaN(num))
num= 1;
if(num<=0)
num = 1;
if(num>stock)
num = stock;
$(".productNumberSetting").val(num);
});
~~~
监听keyup键盘弹起事件
获取输入框的值
* 如果是非数字,那么就设置为1。
> 注: parseInt会把文本中的非数字前的数字解析出来,所以如果文本框的内容是22B,那么解析出来是22.
* 如果是负数,那么设置为1。
* 如果大于库存,设置为最大库存。
~~~
$(".increaseNumber").click(function(){
var num= $(".productNumberSetting").val();
num++;
if(num>stock)
num = stock;
$(".productNumberSetting").val(num);
});
~~~
点击增加按钮的时候,获取当前的值,并在当前值的基础上`+1`,如果超过了库存就取库存最大值
~~~
$(".decreaseNumber").click(function(){
var num= $(".productNumberSetting").val();
--num;
if(num<=0)
num=1;
$(".productNumberSetting").val(num);
});
~~~
点击减少按钮的时候,获取当前的值,并在当前值的基础上`-1`,如果`<=0`,则取1。
## 步骤 17 : productReview.jsp
暂时没有数据,所以看不到截图。
这里借助`c:forEach`遍历request中的reviews。
![](https://box.kancloud.cn/b7bb120c60bc587b4296cb8ca1eea5b8_848x139.png)
~~~
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<div class="productReviewDiv" >
<div class="productReviewTopPart">
<a href="#nowhere" class="productReviewTopPartSelectedLink">商品详情</a>
<a href="#nowhere" class="selected">累计评价 <span class="productReviewTopReviewLinkNumber">${p.reviewCount}</span> </a>
</div>
<div class="productReviewContentPart">
<c:forEach items="${reviews}" var="r">
<div class="productReviewItem">
<div class="productReviewItemDesc">
<div class="productReviewItemContent">
${r.content }
</div>
<div class="productReviewItemDate"><fmt:formatDate value="${r.createDate}" pattern="yyyy-MM-dd"/></div>
</div>
<div class="productReviewItemUserInfo">
${r.user.anonymousName}<span class="userInfoGrayPart">(匿名)</span>
</div>
<div style="clear:both"></div>
</div>
</c:forEach>
</div>
</div>
~~~
## 步骤 18 : productDetail.jsp
如图所示,productDetail.jsp做了两件事
1. 显示属性值
~~~
<c:forEach items="${pvs}" var="pv">
<span>${pv.property.name}: ${fn:substring(pv.value, 0, 10)} </span>
</c:forEach>
~~~
2. 显示详情图片
~~~
<c:forEach items="${p.productDetailImages}" var="pi">
<img src="img/productDetail/${pi.id}.jpg">
</c:forEach>
~~~
~~~
<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%@ taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %>
<div class="productDetailDiv" >
<div class="productDetailTopPart">
<a href="#nowhere" class="productDetailTopPartSelectedLink selected">商品详情</a>
<a href="#nowhere" class="productDetailTopReviewLink">累计评价 <span class="productDetailTopReviewLinkNumber">${p.reviewCount}</span> </a>
</div>
<div class="productParamterPart">
<div class="productParamter">产品参数:</div>
<div class="productParamterList">
<c:forEach items="${pvs}" var="pv">
<span>${pv.property.name}: ${fn:substring(pv.value, 0, 10)} </span>
</c:forEach>
</div>
<div style="clear:both"></div>
</div>
<div class="productDetailImagesPart">
<c:forEach items="${p.productDetailImages}" var="pi">
<img src="img/productDetail/${pi.id}.jpg">
</c:forEach>
</div>
</div>
~~~
- 项目简介
- 功能一览
- 前台
- 后台
- 开发流程
- 需求分析-展示
- 首页
- 产品页
- 分类页
- 搜索结果页
- 购物车查看页
- 结算页
- 确认支付页
- 支付成功页
- 我的订单页
- 确认收货页
- 确认收货成功页
- 评价页
- 需求分析-交互
- 分类页排序
- 立即购买
- 加入购物车
- 调整订单项数量
- 删除订单项
- 生成订单
- 订单页功能
- 确认付款
- 确认收货
- 提交评价信息
- 登录
- 注册
- 退出
- 搜索
- 前台需求列表
- 需求分析后台
- 分类管理
- 属性管理
- 产品管理
- 产品图片管理
- 产品属性设置
- 用户管理
- 订单管理
- 后台需求列表
- 表结构设计
- 数据建模
- 表与表之间的关系
- 后台-分类管理
- 可运行的项目
- 静态资源
- JSP包含关系
- 查询
- 分页
- 增加
- 删除
- 编辑
- 修改
- 做一遍
- 重构
- 分页方式
- 分类逆向工程
- 所有逆向工程
- 后台其他页面
- 属性管理实现
- 产品管理实现
- 产品图片管理实现
- 产品属性值设置
- 用户管理实现
- 订单管理实现
- 前端
- 前台-首页
- 可运行的项目
- 静态资源
- ForeController
- home方法
- home.jsp
- homePage.jsp
- 前台-无需登录
- 注册
- 登录
- 退出
- 产品页
- 模态登录
- 分类页
- 搜索
- 前台-需要登录
- 购物流程
- 立即购买
- 结算页面
- 加入购物车
- 查看购物车页面
- 登录状态拦截器
- 其他拦截器
- 购物车页面操作
- 订单状态图
- 生成订单
- 我的订单页
- 我的订单页操作
- 评价产品
- 总结