§I18N API Migration
There are a number of changes to the I18N API to make working with messages and languages easier to use, particularly with forms and templates.
§Java API
§Refactored Messages API to interfaces
The play.i18n
package has changed to make access to Messages
easier. These changes should be transparent to the user, but are provided here for teams extending the I18N API.
Messages
is now an interface, and there is a MessagesImpl
class that implements that interface.
§Deprecated / Removed Methods
The static deprecated methods in play.i18n.Messages
have been removed in 2.6.x, as there are equivalent methods on the MessagesApi
instance.
§Scala API
§Refactored Messages API to traits
The play.api.i18n
package has changed to make access to Messages
instances easier and reduce the number of implicits in play. These changes should be transparent to the user, but are provided here for teams extending the I18N API.
Messages
is now a trait (rather than a case class). The case class is now MessagesImpl
, which implements Messages
.
§Smoother I18nSupport
Using a form inside a controller is a smoother experience in 2.6.x. ControllerComponents
contains a MessagesApi
instance, which is exposed by AbstractController
. This means that the I18nSupport
trait does not require an explicit val messagesApi: MessagesApi
declaration.
class FormController @Inject()(components: ControllerComponents)
extends AbstractController(components) with I18nSupport {
import play.api.data.validation.Constraints._
val userForm = Form(
mapping(
"name" -> text.verifying(nonEmpty),
"age" -> number.verifying(min(0), max(100))
)(UserData.apply)(UserData.unapply)
)
def index = Action { implicit request =>
// use request2messages implicit conversion method
Ok(views.html.user(userForm))
}
def showMessage = Action { request =>
// uses type enrichment
Ok(request.messages("hello.world"))
}
def userPost = Action { implicit request =>
userForm.bindFromRequest.fold(
formWithErrors => {
BadRequest(views.html.user(formWithErrors))
},
user => {
Redirect(routes.FormController.index()).flashing("success" -> s"User is ${user}!")
}
)
}
}
Note there is now also type enrichment in I18nSupport
which adds request.messages
and request.lang
. This can be added either by extending from I18nSupport
, or by import I18nSupport._
. The import version does not contain the request2messages
implicit conversion.
§MessagesProvider trait
A new MessagesProvider
trait is available, which exposes a Messages
instance.
trait MessagesProvider {
def messages: Messages
}
MessagesImpl
implements Messages
and MessagesProvider
, and returns itself by default.
All the template helpers now take MessagesProvider
as an implicit parameter, rather than a straight Messages
object, i.e. inputText.scala.html
takes the following:
@(field: play.api.data.Field, args: (Symbol,Any)*)(implicit handler: FieldConstructor, messagesProvider: play.api.i18n.MessagesProvider)
The benefit to using a MessagesProvider
is that otherwise, if you used implicit Messages
, you would have to introduce implicit conversions from other types like Request
in places where those implicits could be confusing:
class MessagesRequest[A](request: Request[A], val messages: Messages)
extends WrappedRequest(request) with play.api.i18n.MessagesProvider {
def lang: Lang = messages.lang
}
abstract class AbstractMessagesController(cc: ControllerComponents)
extends AbstractController(cc) {
def MessagesAction: ActionBuilder[MessagesRequest, AnyContent] = {
cc.actionBuilder.andThen(new ActionTransformer[Request, MessagesRequest] {
def transform[A](request: Request[A]) = Future.successful {
val messages = cc.messagesApi.preferred(request)
new MessagesRequest(request, messages)
}
override protected def executionContext: ExecutionContext = cc.executionContext
})
}
}
class MessagesController @Inject() (
addToken: CSRFAddToken,
checkToken: CSRFCheck,
components: ControllerComponents
) extends AbstractMessagesController(components) {
import play.api.data.Form
import play.api.data.Forms._
val userForm = Form(
mapping(
"name" -> text,
"age" -> number
)(UserData.apply)(UserData.unapply)
)
def index = addToken {
MessagesAction { implicit request =>
Ok(views.html.formpage(userForm))
}
}
def userPost = checkToken {
MessagesAction { implicit request =>
userForm.bindFromRequest.fold(
formWithErrors => {
play.api.Logger.info(s"unsuccessful user submission")
BadRequest(views.html.formpage(formWithErrors))
},
user => {
play.api.Logger.info(s"successful user submission ${user}")
Redirect(routes.MessagesController.index()).flashing("success" -> s"User is ${user}!")
}
)
}
}
}
This is also useful for passing around a single implicit request, especially when CSRF checks are involved:
@(userForm: Form[UserData])(implicit request: MessagesRequest[_])
@helper.form(action = routes.MessagesController.userPost()) {
@views.html.helper.CSRF.formField
@helper.inputText(userForm("name"))
@helper.inputText(userForm("age"))
<input type="submit" value="SUBMIT"/>
}
Please see passing messages to form helpers for more details.
§DefaultMessagesApi component
The default implementation of MessagesApi
is DefaultMessagesApi
. DefaultMessagesApi
used to take Configuration
and Environment
directly, which made it awkward to deal with in forms. For unit testing purposes, DefaultMessagesApi
can be instantiated without arguments, and will take a raw map.
import play.api.data.Forms._
import play.api.data._
import play.api.i18n._
val messagesApi = new DefaultMessagesApi(
Map("en" ->
Map("error.min" -> "minimum!")
)
)
implicit val request = {
play.api.test.FakeRequest("POST", "/")
.withFormUrlEncodedBody("name" -> "Play", "age" -> "-1")
}
implicit val messages = messagesApi.preferred(request)
def errorFunc(badForm: Form[UserData]) = {
BadRequest(badForm.errorsAsJson)
}
def successFunc(userData: UserData) = {
Redirect("/").flashing("success" -> "success form!")
}
val result = Future.successful(form.bindFromRequest().fold(errorFunc, successFunc))
Json.parse(contentAsString(result)) must beEqualTo(Json.obj("age" -> Json.arr("minimum!")))
For functional tests that involve configuration, the best option is to use WithApplication
and pull in an injected MessagesApi
:
import play.api.test.{ PlaySpecification, WithApplication }
import play.api.mvc.Controller
import play.api.i18n._
class MessagesSpec extends PlaySpecification with Controller {
sequential
implicit val lang = Lang("en-US")
"Messages" should {
"provide default messages" in new WithApplication(_.requireExplicitBindings()) {
val messagesApi = app.injector.instanceOf[MessagesApi]
val javaMessagesApi = app.injector.instanceOf[play.i18n.MessagesApi]
val msg = messagesApi("constraint.email")
val javaMsg = javaMessagesApi.get(new play.i18n.Lang(lang), "constraint.email")
msg must ===("Email")
msg must ===(javaMsg)
}
"permit default override" in new WithApplication(_.requireExplicitBindings()) {
val messagesApi = app.injector.instanceOf[MessagesApi]
val msg = messagesApi("constraint.required")
msg must ===("Required!")
}
}
}
If you need to customize the configuration, it’s better to add configuration values into the GuiceApplicationBuilder rather than use the DefaultMessagesApiProvider directly.
§Deprecated Methods
play.api.i18n.Messages.Implicits.applicationMessagesApi
and play.api.i18n.Messages.Implicits.applicationMessages
have been deprecated, because they rely on an implicit Application
instance.
The play.api.mvc.Controller.request2lang
method has been deprecated, because it was using a global Application
under the hood.
The play.api.i18n.I18nSupport.request2Messages
implicit conversion method has been moved to I18NSupportLowPriorityImplicits.request2Messages
, and deprecated in favor of request.messages
type enrichment, which is clearer overall.
The I18NSupportLowPriorityImplicits.lang2Messages
implicit conversion has been moved out to LangImplicits.lang2Messages
, because of confusion when both implicit Request and a Lang were in scope. Please extend the play.api.i18n.LangImplicits
trait specifically if you want to create a Messages
from an implicit Lang
.
Next: WS Migration