【问题标题】:Scala Play: Implement custom QueryStringBindable with an optional fieldScala Play:使用可选字段实现自定义 QueryStringBindable
【发布时间】:2020-06-23 19:04:12
【问题描述】:

我正在使用带有 Play 框架 (2.8.1) 的 Scala,并且有一个 Sort 类用于捕获基于 sort 的查询字符串参数。示例网址:http://myurl:9000?sortBy=name&sortOrder=asc。这里,sortOrder 字段是可选的,如果未提供任何内容,则默认值为"asc"。我已经实现了自己的QueryStringBindable 类如下:

object Sort {
  val asc = "asc"
  implicit def queryStringBinder(implicit stringBinder: QueryStringBindable[String]) = new QueryStringBindable[Sort] {
    override def bind(key: String, params: Map[String, Seq[String]]): Option[Either[String, Sort]] = {
      for {
        sortBy <- stringBinder.bind("sortBy", params)
        if(params.contains("sortOrder")) {sortOrder <- stringBinder.bind("sortOrder", params)}
      } yield {
        (sortBy, sortOrder) match {
          case (Right(sortBy), Right(sortOrder)) => Right(Sort(sortBy, Some(sortOrder)))
          case _ => Left("Unable to bind Sort")
        }
      }
    }
    override def unbind(key: String, sort: Sort): String = {
      stringBinder.unbind("sortBy", sort.sortBy) + "&" + stringBinder.unbind("sortOrder", sort.sortOrder.getOrElse(asc))
    }
  }
}

case class Sort(sortBy: String, sortOrder: Option[String] = Some(Sort.asc))

但是,如果 url 的查询字符串中没有提供任何内容,我将无法使用默认值捕获可选字段 sortOrder。即使未提供&amp;sortOrder,我也希望http://myurl:9000?sortBy=name 仍按升序(默认)排序。

【问题讨论】:

    标签: scala playframework


    【解决方案1】:

    不使用 for 理解可能更容易:

    object Sort {
      val asc = "asc"
    
      implicit def queryStringBinder(implicit stringBinder: QueryStringBindable[String]) = new QueryStringBindable[Sort] {
        override def bind(key: String, params: Map[String, Seq[String]]): Option[Either[String, Sort]] = {
          val sortBy = stringBinder.bind("sortBy", params) // Option[Either[String, String]]
          val sortOrder = stringBinder.bind("sortOrder", params) // Option[Either[String, String]]
    
          val result = // Either[String, Sort]
            (sortBy, sortOrder) match {
              case (Some(Right(field)), Some(Right(order))) => Right(Sort(field, Some(order)))
              case (Some(Right(field)), None) => Right(Sort(field))
              case _ => Left("Unable to bind Sort")
            }
          Option(result)
        }
    
        override def unbind(key: String, sort: Sort): String = {
          stringBinder.unbind("sortBy", sort.sortBy) + "&" + stringBinder.unbind("sortOrder", sort.sortOrder.getOrElse(asc))
        }
      }
    }
    
    case class Sort(sortBy: String, sortOrder: Option[String] = Some(Sort.asc))
    

    【讨论】:

    • 工作就像一个魅力!谢谢。
    猜你喜欢
    • 1970-01-01
    • 2016-10-24
    • 2023-03-24
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-05-10
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多