Globalize builds on the I18n API in Ruby on Rails to add model translations to ActiveRecord models.
- ActiveRecord > 4.0.0 (see below for installation with ActiveRecord 3.x)
- I18n
To install the ActiveRecord 4.x compatible version of Globalize with its default setup, just use:
gem install globalize --pre
When using bundler put this in your Gemfile:
gem 'globalize', '~> 4.0.0.alpha.2'
To use the ActiveRecord 3.x version of globalize, specify the version number 3.0.0
:
gem 'globalize', '~> 3.0.0'
The 3-0-stable
branch of this repository is the ActiveRecord 3.x version of globalize. Note that globalize3
has been deprecated and you are encouraged to update your Gemfile accordingly.
Model translations allow you to translate your models' attribute values. E.g.
class Post < ActiveRecord::Base
translates :title, :text
end
Allows you to translate the attributes :title and :text per locale:
I18n.locale = :en
post.title # => Globalize rocks!
I18n.locale = :he
post.title # => גלובאלייז2 שולט!
In order to make this work, you'll need to add the appropriate translation tables.
Globalize comes with a handy helper method to help you do this.
It's called create_translation_table!
. Here's an example:
_Note that your migrations can use create_translation_table!
and drop_translation_table!
only inside the up
and down
instance methods, respectively. You cannot use create_translation_table!
and drop_translation_table!
inside the change
instance method.
Do not use the change
method, use up
and down
!
class CreatePosts < ActiveRecord::Migration
def up
create_table :posts do |t|
t.timestamps
end
Post.create_translation_table! :title => :string, :text => :text
end
def down
drop_table :posts
Post.drop_translation_table!
end
end
Also, you can pass options for specific columns. Here’s an example:
class CreatePosts < ActiveRecord::Migration
def up
create_table :posts do |t|
t.timestamps
end
Post.create_translation_table! :title => :string,
:text => {:type => :text, :null => false, :default => 'abc'}
end
def down
drop_table :posts
Post.drop_translation_table!
end
end
Note that the ActiveRecord model Post
must already exist and have a translates
directive listing the translated fields.
As well as creating a translation table, you can also use create_translation_table!
to migrate across any existing data to the default locale. This can also operate
in reverse to restore any translations from the default locale back to the model
when you don't want to use a translation table anymore using drop_translation_table!
This feature makes use of untranslated_attributes
which allows access to the
model's attributes as they were before the translation was applied. Here's an
example (which assumes you already have a model called Post
and its table
exists):
class TranslatePosts < ActiveRecord::Migration
def self.up
Post.create_translation_table!({
:title => :string,
:text => :text
}, {
:migrate_data => true
})
end
def self.down
Post.drop_translation_table! :migrate_data => true
end
end
NOTE: Make sure you drop the translated columns from the parent table after all your data is safely migrated.
Globalize nicely integrates with
paper_trail. To add versioning
support to your model, you'll want to add the :versioning => true
option to your call to translates
. An example from our test suite:
translates :title, :content, :published, :published_at, :versioning => true
You will also need to have already generated the versions table that paper_trail expects. See the paper_trail README for more details.
If you are adding globalize to any previously versioned models, please note
that you will need to add a new locale
column to your versioning table.
Also, please see the tests in test/globalize/versioning_test.rb
for some
current gotchas.
It is possible to enable fallbacks for empty translations. It will depend on the configuration setting you have set for I18n translations in your Rails config.
You can enable them by adding the next line to config/application.rb
(or only
config/environments/production.rb
if you only want them in production)
config.i18n.fallbacks = true
By default, globalize will only use fallbacks when your translation model does
not exist or the translation value for the item you've requested is nil
.
However it is possible to also use fallbacks for blank
translations by adding
:fallbacks_for_empty_translations => true
to the translates
method.
class Post < ActiveRecord::Base
translates :title, :name
end
puts post.translations.inspect
# => [#<Post::Translation id: 1, post_id: 1, locale: "en", title: "Globalize rocks!", name: "Globalize">,
#<Post::Translation id: 2, post_id: 1, locale: "nl", title: '', name: nil>]
I18n.locale = :en
post.title # => 'Globalize rocks!'
post.name # => 'Globalize'
I18n.locale = :nl
post.title # => ''
post.name # => 'Globalize'
class Post < ActiveRecord::Base
translates :title, :name, :fallbacks_for_empty_translations => true
end
puts post.translations.inspect
# => [#<Post::Translation id: 1, post_id: 1, locale: "en", title: "Globalize rocks!", name: "Globalize">,
#<Post::Translation id: 2, post_id: 1, locale: "nl", title: '', name: nil>]
I18n.locale = :en
post.title # => 'Globalize rocks!'
post.name # => 'Globalize'
I18n.locale = :nl
post.title # => 'Globalize rocks!'
post.name # => 'Globalize'
It is possible to setup locales to fallback to each other.
class Post < ActiveRecord::Base
translates :title, :name
end
Globalize.fallbacks = {:en => [:en, :pl], :pl => [:pl, :en]}
I18n.locale = :en
en_post = Post.create(:title => 'en_title')
I18n.locale = :pl
pl_post = Post.create(:title => 'pl_title')
en_post.title # => 'en_title'
I18n.locale = :en
en_post.title # => 'en_title'
pl_post.title # => 'pl_title'
To only return objects that have a translation for the given locale we can use
the with_translations
scope. This will only return records that have a
translations for the passed in locale.
Post.with_translations('en')
# => [
#<Post::Translation id: 1, post_id: 1, locale: "en", title: "Globalize rocks!", name: "Globalize">,
#<Post::Translation id: 2, post_id: 1, locale: "nl", title: '', name: nil>
]
Post.with_translations(I18n.locale)
# => [
#<Post::Translation id: 1, post_id: 1, locale: "en", title: "Globalize rocks!", name: "Globalize">,
#<Post::Translation id: 2, post_id: 1, locale: "nl", title: '', name: nil>
]
Post.with_translations('de')
# => []
In views, if there is content from different locales that you wish to display,
you should use the with_locale
option with a block, as below:
<% Globalize.with_locale(:en) do %>
<%= render "my_translated_partial" %>
<% end %>
Your partial will now be rendered with the :en
locale set as the current locale.
Globalize supports interpolation in a similar manner to I18n.
class Post < ActiveRecord::Base
translates :title
end
I18n.locale = :en
post.title = "Globalize %{superlative}!"
post.title
# #=> "Globalize %{superlative}!"
post.title(:foo => "bar")
# SomeError: missing interpolation argument :superlative
post.title(:superlative => "rocks")
# #=> "Globalize rocks!"
- Veger's fork - uses default AR schema for the default locale, delegates to the translations table for other locales only
- TranslatableColumns - have multiple languages of the same attribute in a model (Iain Hecker)
- Traco - A newer take on using multiple columns in the same model (Barsoom)
- localized_record - allows records to have localized attributes without any modifications to the database (Glenn Powell)
- model_translations - Minimal implementation of Globalize2 style model translations (Jan Andersson)
- hstore_translate - Rails I18n library for ActiveRecord model/data translation using PostgreSQL's hstore datatype (Rob Worley)
- globalize2_versioning - acts_as_versioned style versioning for globalize2 (Joshua Harvey)
- i18n_multi_locales_validations - multi-locales attributes validations to validates attributes from globalize2 translations models (Sébastien Grosjean)
- globalize2 Demo App - demo application for globalize2 (Sven Fuchs)
- migrate_from_globalize1 - migrate model translations from Globalize1 to globalize2 (Tomasz Stachewicz)
- easy_globalize2_accessors - easily access (read and write) globalize2-translated fields (astropanic, Tomasz Stachewicz)
- globalize2-easy-translate - adds methods to easily access or set translated attributes to your model (bsamman)
- batch_translations - allow saving multiple globalize2 translations in the same request (Jose Alvarez Rilla)