Custom MessageSource

Example code is available on github.com/vaadin-on-grails/custom-messagesource.

We might want to define a custom MessageSource that will return a localized message. If a message is missing, we might want to log information about missing localization (in logs or database) or we might want to provide other way to fetch the localized labels.

In this toturial we will show way to load localization from database.

Step 1

Enable OSIV in VaadinConfig.groovy. Add or comment out the following line.

openSessionInViewFilter = 'org.springframework.orm.hibernate4.support.OpenSessionInViewFilter'

Step 2

Create new domain object that will hold localization data. Run grails create-domain app.Message command.

  • key holds the localization key that we will you in our application to refer a localized string

  • value is localized string, for example a label in English

  • locale defines in what language is value string

package app

class Message {

    String key
    String value
    Locale locale
}

Step 3

Create localized key-value pair for given language in BootStrap.groovy.

import app.Message

class BootStrap {

    def init = { servletContext ->

        Message home = new Message(key: 'home', value: 'Home', locale: Locale.ENGLISH)
        home.save(failOnError: true)
    }
    def destroy = {
    }
}

Step 4

Implement a new message source that will use GORM domain object, Message in our case, to load the localized messages. When a value is not found, we will return the key in brackets.

package app.i18n

import app.Message
import org.codehaus.groovy.grails.context.support.PluginAwareResourceBundleMessageSource
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.context.support.AbstractMessageSource

import java.text.MessageFormat

class JdbcMessageSource extends AbstractMessageSource {

    @Autowired
    PluginAwareResourceBundleMessageSource messageBundleMessageSource

    @Override
    protected MessageFormat resolveCode(String code, Locale l) {
        Message message = Message.findByKeyAndLocale(code, l)

        MessageFormat format
        if (message) {
            format = new MessageFormat(message.value, message.locale)
        } else {
            format = messageBundleMessageSource.resolveCode(code, l)
            if (!format) {
                format = new MessageFormat("[$code]", l)
            }
        }
        return format
    }
}

Step 4

Open grails-app/conf/spring/resources.groovy and add a new bean messageSource and messageBundleMessageSource that we use to get the localization in case the localization is not found in database.

import app.i18n.JdbcMessageSource
import org.codehaus.groovy.grails.context.support.PluginAwareResourceBundleMessageSource

beans = {

    messageSource(JdbcMessageSource)

    messageBundleMessageSource(PluginAwareResourceBundleMessageSource) {
        basenames = "WEB-INF/grails-app/i18n/messages"
    }
}

Step 5

Now we can use standard Grails.i18 method to get localized strings.

package app

import com.vaadin.grails.Grails
import com.vaadin.server.VaadinRequest
import com.vaadin.ui.Label
import com.vaadin.ui.UI
import com.vaadin.ui.VerticalLayout

class MyUI extends UI {

    @Override
    protected void init(VaadinRequest r) {

        VerticalLayout layout = new VerticalLayout()
        layout.setMargin(true)

        Label lbl1 = new Label(Grails.i18n('do.not.exist', Locale.ENGLISH))
        layout.addComponent(new Label(lbl1))

        Label lbl2 = new Label(Grails.i18n('home', Locale.ENGLISH))
        layout.addComponent(new Label(lbl2))

        setContent(layout)
    }
}

Step 6

Run the application grails run-app and open it in the browser http://localhost:8080/custom-messagesource. Then we will see the messages in the console.

Last updated