# frozen_string_literal: true require "mobility/backends/active_record" require "mobility/backends/column" module Mobility module Backends =begin Implements the {Mobility::Backends::Column} backend for ActiveRecord models. You can use the +mobility:translations+ generator to create a migration adding translatable columns to the model table with: rails generate mobility:translations post title:string The generated migration will add columns +title_+ for every locale in +Mobility.available_locales+ (i.e. +I18n.available_locales+). (The generator can be run again to add new attributes or locales.) @example class Post < ActiveRecord::Base extend Mobility translates :title, backend: :column end Mobility.locale = :en post = Post.create(title: "foo") post.title #=> "foo" post.title_en #=> "foo" =end class ActiveRecord::Column include ActiveRecord include Column # @!group Backend Accessors # @!macro backend_reader def read(locale, _ = {}) model.read_attribute(column(locale)) end # @!macro backend_writer def write(locale, value, _ = {}) model.send(:write_attribute, column(locale), value) end # @!endgroup # @!macro backend_iterator def each_locale available_locales.each { |l| yield(l) if present?(l) } end # @param [String] attr Attribute name # @param [Symbol] locale Locale # @return [Arel::Attributes::Attribute] Arel node for translation column # on model table def self.build_node(attr, locale) model_class.arel_table[Column.column_name_for(attr, locale)] .extend(::Mobility::Arel::MobilityExpressions) end private def available_locales @available_locales ||= get_column_locales end def get_column_locales column_name_regex = /\A#{attribute}_([a-z]{2}(_[a-z]{2})?)\z/.freeze model.class.columns.map do |c| (match = c.name.match(column_name_regex)) && match[1].to_sym end.compact end end register_backend(:active_record_column, ActiveRecord::Column) end end