玩:将表单字段绑定为双?

时间:2012-10-06 11:39:43

标签: scala playframework-2.0

也许我只是忽略了一些显而易见的事情,但我无法弄清楚如何在Play控制器中将表单字段绑定到double。

例如,假设这是我的模型:

case class SavingsGoal(
timeframeInMonths: Option[Int],
amount: Double,
name: String
)

(忽略我使用双倍的钱,我知道这是一个坏主意,这只是一个简化的例子)

我想这样绑定它:

object SavingsGoals extends Controller {

    val savingsForm: Form[SavingsGoal] = Form(

        mapping(
            "timeframeInMonths" -> optional(number.verifying(min(0))),
            "amount" -> of[Double],
            "name" -> nonEmptyText
        )(SavingsGoal.apply)(SavingsGoal.unapply)

    )

}

我意识到number帮助程序仅适用于int,但我认为使用of[]可能允许我绑定一个double。但是,我得到了一个编译器错误:

Cannot find Formatter type class for Double. Perhaps you will need to import
play.api.data.format.Formats._  

这样做没有用,因为API中没有定义双格式化程序。

这只是一个很长的方式,询问在Play中将表单字段绑定到double的规范方法是什么?

谢谢!

编辑:4e6指出了正确的方向。这是我用他的帮助做的事情:

在他的链接中使用片段,我将以下内容添加到app.controllers.Global.scala:

object Global {

    /**
     * Default formatter for the `Double` type.
     */
    implicit def doubleFormat: Formatter[Double] = new Formatter[Double] {

      override val format = Some("format.real", Nil)

      def bind(key: String, data: Map[String, String]) =
        parsing(_.toDouble, "error.real", Nil)(key, data)

      def unbind(key: String, value: Double) = Map(key -> value.toString)
    }

    /**
     * Helper for formatters binders
     * @param parse Function parsing a String value into a T value, throwing an exception in case of failure
     * @param error Error to set in case of parsing failure
     * @param key Key name of the field to parse
     * @param data Field data
     */
    private def parsing[T](parse: String => T, errMsg: String, errArgs: Seq[Any])(key: String, data: Map[String, String]): Either[Seq[FormError], T] = {
      stringFormat.bind(key, data).right.flatMap { s =>
        util.control.Exception.allCatch[T]
          .either(parse(s))
          .left.map(e => Seq(FormError(key, errMsg, errArgs)))
      }
    }

}

然后,在我的表单映射中:

mapping(
    "amount" -> of(Global.doubleFormat)
)

2 个答案:

答案 0 :(得分:11)

如果您的版本为2.1,则不需要使用全局格式。

只需导入:

import play.api.data.format.Formats._

并用作:

mapping(
    "amount" -> of(doubleFormat)
)

答案 1 :(得分:10)

实际上,there is主分支上Double的预定义格式化程序。所以你应该切换到2.1-SNAPSHOT播放版本或者只是复制实现。