6

我是 Play! 的新手,我正在尝试将我现有的网站从 cakePHP 迁移到 Play!。

我面临的问题是关于表单验证。

我定义了一个案例类用户,代表我网站的用户:

case class User(
  val id: Long,
  val username: String,
  val password: String,
  val email: String
  val created: Date)

(还有一些字段,但这些足以解释我的问题)

我希望我的用户能够使用表单在我的网站上创建帐户,并且我希望 Play! 验证此表单。

因此,我创建了以下操作:

def register = Action {
  implicit request =>

    val userForm = Form(
      mapping(
        "id" -> longNumber,
        "username" -> nonEmptyText(8),
        "password" -> nonEmptyText(5),
        "email" -> email,
        "created" -> date)(User.apply)(User.unapply))

    val processedForm = userForm.bindFromRequest
    processedForm.fold(hasErrors => BadRequest("Invalid submission"), success => {
      Ok("Account registered.")
    })
}

显然,我不希望用户在表单中自己填写 id 或创建日期。所以我的问题是:我该怎么办?

我是否应该定义一个新的“转换模型”,只包含表单中实际提供给用户的字段,并将这个中间模型转换为完整的模型,然后再将其插入我的数据库?

也就是说,用类似的东西替换我的操作:

def register = Action {
  implicit request =>

    case class UserRegister(
      username: String,
      password: String,
      email: String)

    val userForm = Form(
      mapping(
        "username" -> nonEmptyText(8),
        "password" -> nonEmptyText(8),
        "email" -> email)(UserRegister.apply)(UserRegister.unapply)

    val processedForm = userForm.bindFromRequest
    processedForm.fold(hasErrors => BadRequest("Invalid submission"), success => {
      val user = User(nextID, success.username, success.password, success.email, new Date())
      // Register the user...
      Ok("Account created")
}

还是有另一种更清洁的方式来做我想做的事?

我已经阅读了许多教程和“Play for Scala”一书,但在我发现的唯一示例中,模型完全被表格填充......我真的很喜欢 Play!到目前为止,但看起来文档经常缺少示例......

非常感谢您的回答!

4

1 回答 1

4

你有几个选择:

首先,您可以分别制作idandcreated字段。然后使用如下映射:Option[Long]Option[Date]

val userForm = Form(
  mapping(
    "id" -> optional(longNumber),
    "username" -> nonEmptyText(8),
    "password" -> nonEmptyText(5),
    "email" -> email,
    "created" -> optional(date)
  )(User.apply)(User.unapply)
)

我认为这是合乎逻辑的,因为User带有Noneid 的 a 表示它尚未保存。当您想使用相同的表单映射来更新现有记录时,这很有效。

或者,您可以使用ignored带有一些任意占位符数据的映射:

val userForm = Form(
  mapping(
    "id" -> ignored(-1L),
    "username" -> nonEmptyText(8),
    "password" -> nonEmptyText(5),
    "email" -> email,
    "created" -> ignored(new Date)
  )(User.apply)(User.unapply)
)

这在重用表单进行更新操作时不太好!

最后,不要忘记您的表单映射是由将元组分别转换为对象和将对象转换为元组的函数绑定/填充的。User.apply使用案例类和User.unapply方法只是一个方便的约定,因为它们就是这样做的。您可以在User对象上编写替代工厂方法来处理表单实例化:

object User {
  def formApply(username: String, password: String, email: String): User = 
    new User(-1L, username, password, email, new Date)

  def formUnapply(user: User): Option[(String,String,String)] = 
    Some((user.username, user.password, user.email))
}

然后使用 Form 对象中的那些:

val userForm = Form(
  mapping(
    "username" -> nonEmptyText(8),
    "password" -> nonEmptyText(5),
    "email" -> email
  )(User.formApply)(User.formUnapply)
)

此外,值得注意的是,Scala 表单文档在 2.2.1 中将变得更好(实际上它可能已经在这里推出了)。

于 2013-10-27T20:06:39.573 回答