注解: CRUD——Create,Retrieve, Update, Delete
文件结构
一、添加控制器(controller和action)——/app/controllers/Products.java
package controllers; import play.mvc.Controller; import play.mvc.Result; public class Products extends Controller { public static Result list() { //列出所有的产品 return TODO; } public static Result newProduct() { //显示一个空的产品Form return TODO; } public static Result details(String ean) { //显示一个编辑产品的Form return TODO; } public static Result save() { //保存产品资料 return TODO; } }
说明: controllers的action方法必须满足以下三个条件:
- 必须是public
- 必须是static
- 返回类型必须是Result和Result的子类
完整代码为
package controllers; import models.Product; import play.data.Form; import play.mvc.Controller; import play.mvc.Result; import java.util.ArrayList; import java.util.List; import views.html.*; public class Products extends Controller { //constant for Form private static final FormproductForm = Form.form(Product.class); public static Result list(){ List products = Product.findAll(); return ok(list.render(products)); } public static Result newProduct(){ return ok(details.render(productForm)); } public static Result details(String ean){ final Product product = Product.findByEan(ean); if(product == null){ return notFound(String.format("Product %s does not exist.", ean)); } Form filledForm = productForm.fill(product); return ok(details.render(filledForm)); } public static Result save(){ Form boundForm = productForm.bindFromRequest(); if(boundForm.hasErrors()){ flash("error","Please correct the form below."); return badRequest(details.render(boundForm)); } Product product = boundForm.get(); product.save(); flash("success",String.format("Successfully added product %s", product)); return redirect(routes.Products.list()); } public static Result delete(String ean){ final Product product = Product.findByEan(ean); if(product == null){ return notFound(String.format("Product %s does not exist.", ean)); } Product.remove(product); return redirect(routes.Products.list()); } }
二. 修改routes文件
GET /products/ controllers.Products.list() GET /products/new controllers.Products.newProduct() GET /products/:ean controllers.Products.details(ean: String) POST /products/ controllers.Products.save()
DELETE /products/:ean controllers.Products.delete(ean: String)
action和route对应关系如下图
三 . 添加Model Class——/app/models/Product.java
3.1) 数据模型
public class Product { public String ean; public String name; public String description; public Product(){} public Product(String ean, String name, String description){ this.ean = ean; this.name = name; this.description = description; } public String toString(){ return String.format("%s - %s", ean, name, description); } }
说明: 不同于Java,play使用public,不用setter和getter方法
3.2)为了模拟练习,在Product.java中加一些模拟数据
// mocking some data private static Listproducts; static { products = new ArrayList (); products.add(new Product("111111111","Paperclips 1","Paperclips description 1")); products.add(new Product("222222222","Paperclips 2","Paperclips description 2")); products.add(new Product("333333333","Paperclips 3","Paperclips description 3")); products.add(new Product("444444444","Paperclips 1","Paperclips description 4")); products.add(new Product("555555555","Paperclips 1","Paperclips description 5")); }
3.3) 数据处理方法——findAll(), findByEan(), findByName(), remove(), save()
public static ListfindAll(){ return new ArrayList (products); } public static Product findByEan(String ean){ for(Product candidate : products){ if(candidate.ean.equals(ean)){ return candidate; } } return null; } public static List findByName(String term){ final List results = new ArrayList (); for(Product candidate : products){ if(candidate.name.toLowerCase().contains(term.toLowerCase())){ results.add(candidate); } } return results; } public static boolean remove(Product product){ return products.remove(product); } public void save(){ products.remove(findByEan(this.ean)); products.add(this); }
完整代码为
四、添加和修改模板文件——list.scala.html, details.scala.html
4.1) list.scala.html
@(products:List[Product]) @main("Product catalogue"){ <h2>All Producth2> <table class="table table-striped"> <thead> <tr> <th>EANth> <th>Nameth> <th>Descriptionth> tr> thead> <tbody> @for(product <- products){> <td><a href="@routes.Products.details(product.ean)">@product.eana>td> <td><a href="@routes.Products.details(product.ean)">@product.namea>td> <td><a href="@routes.Products.details(product.ean)">@product.descriptiona>td> tr> } tbody> table> } 相应的,修改controller用于显示该模板的HTML内容,即修改/app/controllers/Products.java中的list()方法
... import views.html.*; //引入模板 public class Products extends Controller { public static Result list() { Listproducts = Product.findAll(); return ok(list.render(products)); } ... } 在浏览器地址栏输入: http://localhost:9000/products
4.2) details.scala.html
@(productForm:Form[Product]) @import helper._ @import helper.twitterBootstrap._ @main("Product Form"){class="btn btn-primary" value="Save" /> class="btn" href="@routes.Application.index()">Cancel } }Product Form
@helper.form(action = routes.Products.save()){说明: 本模板使用了Play表单Form
- @(productForm:Form[Product]) —>— productForm是play.data.Form.form()的方法,在controller中定义
- @import helper._ —>— 引入生产HTML,@import helper.twitterBootstrap._ 生产的HTML符合Twitter Bootstrap标准
- @helper.form(action =routes.Products.save()){ —>— 生产的 HTML的Form代码, 并将数据提交给controller的save()方法处理,@helper.inputText生产的 HTML的input代码,@helper.textarea生产的 HTML的textarea代码
- Product(@productForm("name").valueOr("New")) —>— productForm.field("name")的缩写,请求表单中名为name的域(field)值,如何该值不存在,则创建一个"New"为默认值。
最后生产的HTML代码如下
HTML Code<form action="/products" method="GET" > <fieldset> <legend>Product(New)legend> <div class="clearfix " id="ean_field"> <label for="ean">EANlabel> <div class="input"> <input type="text" id="ean" name="ean" value="" > <span class="help-inline">span> <span class="help-block">Requiredspan> div> div> <div class="clearfix " id="name_field"> <label for="name">Namelabel> <div class="input"> <input type="text" id="name" name="name" value="" > <span class="help-inline">span> <span class="help-block">Requiredspan> div> div> <div class="clearfix " id="description_field"> <label for="description">descriptionlabel> <div class="input"> <textarea id="description" name="description" >textarea> <span class="help-inline">span> <span class="help-block">span> div> div> fieldset> <input type="submit" class="btn btn-primary" value="Save" /> <a class="btn" href="/products">Cancela> form>相应的,修改controller用于显示该模板的HTML内容,即修改/app/controllers/Products.java中的newProduct()()方法
public static Result newProduct(){ return ok(details.render(productForm)); }在浏览器地址栏输入: http://localhost:9000/products/new
五、处理表单提交和验证——binding values from the request
play可以从HTTP request获取一个包含name/value对的映射Map,实现这个功能的是类Form的bindFromRequest()方法,该方法会返回一个Form对象,故我们可以在保存的时候获取页面表单的数据。
5.1) 在/app/models/Products.java中加入
public class Products extends Controller { ... public static Result save() { FormboundForm = productForm.bindFromRequest(); Product product = boundForm.get(); product.save(); return ok(String.format("Saved product %s", product)); } }5.2) 在/app/models/Product.java中加入验证
... import play.data.validation.Constraints; public class Product { ... @Constraints.Required public String ean; @Constraints.Required public String name; public String description; ... }5.3) 对/app/models/Products.java中save()的改进
对模型加入验证数据后,希望能在controller中显示相应的成功、失败信息,这里可以使用play的validation表单功能,修改save()方法如下:
public static Result save() { FormboundForm = productForm.bindFromRequest(); if(boundForm.hasErrors()) { flash("error", "Please correct the form below."); return badRequest(details.render(boundForm)); } Product product = boundForm.get(); product.save(); flash("success",String.format("Successfully added product %s", product)); return redirect(routes.Products.list()); } 同时还必须在app/views/main.scala.html的
标签的开头中加入实现flash scope的代码:@if(flash.containsKey("success")){ <div class="alert alert-success"> @flash.get("success") div> } @if(flash.containsKey("error")){ <div class="alert alert-error"> @flash.get("error") div> }说明: flash scope是用于在请求(request)的时候存储变量的一个地方,在下一次请求开始之前存储在flash scope的值一直保持着,所以flash 是flash scope用来显示失败和成功是最理想的!
六、在列表中加入删除功能——app/views/products/list.scala.html
6.1) 在routes中加入DELETE路由
DELETE /products/:ean controllers.Products.delete(ean: String)
6.2) 在list.scala.html中加入JS代码,本例用JS处理Ajax请求
All products
...... @for(product <- products){} New product @product.ean @product.name @product.description
说明: 整个案例代码