React-Rails:使用具有翻译I18n的组件

时间:2016-06-09 11:10:53

标签: ruby-on-rails-4 coffeescript rails-i18n react-rails

我已添加到我的项目react-rails gem中,我想用于翻译的组件。

我不能放入预编译的资产erb模板,但我仍然试图创建组件,使它们在所有项目中可用,然后在某些部分使用它们进行一些翻译。

工作情景

# app/view/example/_react_component.coffee.erb
DOM = React.DOM

FormInput = React.createClass
  displayName: "FormInput"
  render: ->
    DOM.div
      className: 'row control-group'
      DOM.div
        className: 'form-group col-xs-12 floating-label-form-group controls'
        DOM.label
          htmlFor: @props.id
          @props.label
        DOM.input
          id: @props.id
          className: 'form-control'
          placeholder: @props.placeholder
          type: @props.type
        DOM.p
          className: 'help-block text-danger'

formInput = React.createFactory(FormInput)

window.ValidationFormInput = React.createClass
  displayName: "ValidationFormInput"

  getInitialState: ->
    { }

  render: ->
    formInput
      id: "<%= t('validation_form.id') %>"
      label: "<%= t('validation_form.label') %>"
      placeholder: "<%= t('validation_form.placeholder') %>"
      type: 'text'

validationFormInput = React.createFactory(ValidationFormInput)

# app/view/example/index.html.erb
<%= react_component('ValidationFormInput', {}, {class: "container"}) %>

期望的情景(不工作)

# app/assets/javascripts/components/form_input.coffee
DOM = React.DOM

FormInput = React.createClass
  displayName: "FormInput"
  render: ->
    DOM.div
      className: 'row control-group'
      DOM.div
        className: 'form-group col-xs-12 floating-label-form-group controls'
        DOM.label
          htmlFor: @props.id
          @props.label
        DOM.input
          id: @props.id
          className: 'form-control'
          placeholder: @props.placeholder
          type: @props.type
        DOM.p
          className: 'help-block text-danger'

formInput = React.createFactory(FormInput)

# app/view/example/_react_component.coffee.erb
window.ValidationFormInput = React.createClass
  displayName: "ValidationFormInput"

  getInitialState: ->
    { }

  render: ->
    formInput
      id: "<%= t('validation_form.id') %>"
      label: "<%= t('validation_form.label') %>"
      placeholder: "<%= t('validation_form.placeholder') %>"
      type: 'text'

validationFormInput = React.createFactory(ValidationFormInput)

# app/view/example/index.html.erb
<%= react_component('ValidationFormInput', {}, {class: "container"}) %>

我想这个问题与我的组件定义的范围有关,但我无法弄清楚如何使组件可用于任何部分。

提前谢谢

修改

为了使翻译可用,我找到了gem I18n-js。安装后,我可以轻松运行rake任务来创建我的config/locales/*翻译

的js版本

1 个答案:

答案 0 :(得分:2)

很棒的问题。

有几种方法可以做到这一点。

1-通常,这不仅仅是关于如何将数据从Rails传递到React的问题,而是如何将数据传递给Javascript的问题。您可以将数据存储在标头中的meta中,然后通过Javascript访问它。通过这种方式,您仍然可以快速压缩JS。 (而不是js.erb等)

2-将所有翻译传递给react组件。基本上,您可以将参数传递给react组件,其中一个是翻译。如果是一些翻译,那很好但是如果你的列表增长,你的页面上的负载就会很大。

3-制作自己的Javascript翻译器。这是我创建的CoffeeScript示例;确保在其他文件之前将其添加到资产列表中。 在我的代码中,我从meta中提取区域设置(正如您在代码中看到的那样)。随意编辑它。

class Translator
  @en = {
    internet_connection_lost: "Your internet connection has been lost"
    attempting_to_reconnect: "Attempting to reconnect!"
    failed_to_reconnect: "Failed to reconnect..."
    connection_success: "Connected"
    reconnecting: "Reconnecting..."
    bid_received: "Bid received. New balance $$bid_balance$$"
  }

  @ar = {
    internet_connection_lost: "لقد فقدت الاتصال بالإنترنت"
    attempting_to_reconnect: "نحاول إعادة الاتصال!"
    failed_to_reconnect: "لم تنجح إعادة الاتصال بالشبكة..."
    connection_success: "متصل بشبكة الإنترنت"
    reconnecting: "إعادة الاتصال جارية..."
    bid_received: "تم تلقي العرض. رصيد جديد $$bid_balance$$"
  }

  @get_translations: (locale) ->
    switch (locale)
      when 'en'
        @en
      when 'ar'
        @ar

  @translate: (val, interpolation) ->
    # get locale from meta
    locale = $("meta[name='locale']").attr("content") || "en"
    translation = Translator.get_translations(locale)[val]

    if interpolation
      console.log "#{JSON.stringify(interpolation)}"

    for k,v of interpolation
      console.log "#{translation} : #{k} : #{v}"
      translation = translation.replace(k, v)

    return translation

window.Translator = Translator

这就是你如何使用译者

  message = Translator.translate(
    "bid_received", { "$$bid_balance$$": 10 }
  )