OpenProject is the leading open source project management software.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
openproject/lib/tabular_form_builder.rb

316 lines
10 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is a project management system.
# Copyright (C) 2012-2015 the OpenProject Foundation (OPF)
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# OpenProject is a fork of ChiliProject, which is a fork of Redmine. The copyright follows:
# Copyright (C) 2006-2013 Jean-Philippe Lang
# Copyright (C) 2010-2013 the ChiliProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License
# as published by the Free Software Foundation; either version 2
# of the License, or (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
#
# See doc/COPYRIGHT.rdoc for more details.
#++
require 'action_view/helpers/form_helper'
class TabularFormBuilder < ActionView::Helpers::FormBuilder
include Redmine::I18n
include ActionView::Helpers::AssetTagHelper
(field_helpers - %w(radio_button hidden_field fields_for label) + %w(date_select)).each do |selector|
define_method selector do |field, options = {}, *args|
if options[:multi_locale] || options[:single_locale]
localize_field(field, options, __method__)
else
options[:class] = Array(options[:class]) + [field_css_class(selector)]
input_options, label_options = extract_from options
label = label_for_field(field, label_options)
input = super(field, input_options, *args)
(label + container_wrap_field(input, selector, options))
end
end
end
def label(method, text = nil, options = {}, &block)
options[:class] = Array(options[:class]) + %w(form--label)
options[:title] = options[:title] || title_from_context(method)
super
end
def radio_button(field, value, options = {}, *args)
options[:class] = Array(options[:class]) + %w(form--radio-button)
input_options, label_options = extract_from options
label_options[:for] = "#{sanitized_object_name}_#{field}_#{value.downcase}"
label = label_for_field(field, label_options)
input = super(field, value, input_options, *args)
(label + container_wrap_field(input, 'radio-button', options))
end
def select(field, choices, options = {}, html_options = {})
html_options[:class] = Array(html_options[:class]) + %w(form--select)
label_for_field(field, options) + container_wrap_field(super, 'select', options)
end
def collection_select(field, collection, value_method, text_method, options = {}, html_options = {})
html_options[:class] = Array(html_options[:class]) + %w(form--select)
label_for_field(field, options) + container_wrap_field(super, 'select', options)
end
def collection_check_box(field,
value,
checked,
text = field.to_s + "_#{value}",
options = {})
label_for = "#{sanitized_object_name}_#{field}_#{value}".to_sym
input_options = options.reverse_merge(multiple: true,
checked: checked,
for: label_for,
label: text)
check_box(field, input_options, value, '')
end
def fields_for_custom_fields(record_name, record_object = nil, options = {}, &block)
options_with_defaults = options.merge(builder: CustomFieldFormBuilder)
fields_for(record_name, record_object, options_with_defaults, &block)
end
private
attr_reader :template
TEXT_LIKE_FIELDS = [
'number_field', 'password_field', 'url_field', 'telephone_field', 'email_field'
].freeze
def container_wrap_field(field_html, selector, options = {})
ret = content_tag(:span, field_html, class: field_container_css_class(selector, options))
prefix, suffix = options.values_at(:prefix, :suffix)
# TODO (Rails 4): switch to SafeBuffer#prepend
ret = content_tag(:span, prefix.html_safe, class: 'form--field-affix').concat(ret) if prefix
ret.concat content_tag(:span, suffix.html_safe, class: 'form--field-affix') if suffix
field_container_wrap_field(ret, options)
end
def field_container_wrap_field(field_html, options = {})
if options[:no_label]
field_html
else
content_tag(:span, field_html, class: 'form--field-container')
end
end
def localize_field(field, options, meth)
localized_field = Proc.new do |translation_form, multiple|
localized_field(translation_form, meth, field, options)
end
ret = nil
translation_objects = translation_objects field, options
fields_for(:translations, translation_objects, builder: TabularFormBuilder) do |translation_form|
ret = label_for_field(field, options, translation_form) unless ret
ret.concat localized_field.call(translation_form)
end
if options[:multi_locale]
ret.concat add_localization_link
end
ret
end
def field_container_css_class(selector, options)
classes = if TEXT_LIKE_FIELDS.include?(selector)
'form--text-field-container'
else
"form--#{selector.tr('_', '-')}-container"
end
classes << ' ' + options.fetch(:container_class, '')
classes.strip
end
def field_css_class(selector)
if TEXT_LIKE_FIELDS.include?(selector)
"form--text-field -#{selector.gsub(/_field$/, '')}"
else
"form--#{selector.tr('_', '-')}"
end
end
# Returns a label tag for the given field
def label_for_field(field, options = {}, translation_form = nil)
options = options.dup
return ''.html_safe if options.delete(:no_label)
text = if options[:label].is_a?(Symbol)
l(options[:label])
elsif options[:label]
options[:label]
elsif @object.is_a?(ActiveRecord::Base)
@object.class.human_attribute_name(field.to_sym)
else
l(field)
end
label_options = { class: '',
title: text }
text += @template.content_tag('span', ' *', class: 'required') if options.delete(:required)
id = element_id(translation_form) if translation_form
# FIXME: reenable the error handling
label_options[:class] << 'error' if false && @object && @object.respond_to?(:errors) && @object.errors[field] # FIXME
label_options[:class] << 'form--label'
label_options[:for] = if options[:for]
options[:for]
elsif options[:multi_locale] && id
id.sub(/\_id$/, "_#{field}")
end
label_options[:lang] = options[:lang]
@template.label(@object_name, field.to_s, text.html_safe, label_options)
end
def element_id(translation_form)
match = /id=\"(?<id>\w+)"/.match(translation_form.hidden_field :id)
match ? match[:id] : nil
end
def localized_field(translation_form, method, field, options)
@template.content_tag :span, class: "form--field-container translation #{field}_translation" do
ret = ''.html_safe
field_options = localized_options options, translation_form.object.locale
ret.safe_concat translation_form.send(method, field, field_options.merge(no_label: true))
ret.safe_concat translation_form.hidden_field :id,
class: 'translation_id'
if options[:multi_locale]
ret.safe_concat translation_form.select :locale,
Setting.available_languages.map { |lang| [ll(lang.to_s, :general_lang_name), lang.to_sym] },
{ no_label: true },
class: 'locale_selector'
ret.safe_concat translation_form.hidden_field '_destroy',
disabled: true,
class: 'destroy_flag',
value: '1'
ret.safe_concat '<a href="#" class="destroy_locale icon icon-delete" title="Delete"></a>'
else
ret.safe_concat translation_form.hidden_field :locale,
class: 'locale_selector'
end
ret
end
end
def translation_objects(field, options)
if options[:multi_locale]
multi_translation_object field, options
elsif options[:single_locale]
single_translation_object field, options
end
end
def single_translation_object(_field, _options)
if object.translations.detect { |t| t.locale == :en }.nil?
object.translations.build locale: :en
end
object.translations.select { |t| t.locale == :en }
end
def multi_translation_object(field, _options)
if object.translations.size == 0
object.translations.build locale: user_locale
object.translations
else
translations = object.translations.select do |t|
t.send(field).present?
end
if translations.size > 0
translations
else
object.translations.detect { |t| t.locale == user_locale } ||
object.translations.first
end
end
end
def add_localization_link
@template.content_tag :a, l(:button_add), href: '#', class: 'form--field-extra-actions add_locale'
end
def localized_options(options, locale = :en)
localized_options = options.clone
localized_options[:value] = localized_options[:value][locale] if options[:value].is_a?(Hash)
localized_options.delete(:single_locale)
localized_options.delete(:multi_locale)
localized_options
end
def user_locale
User.current.language.present? ?
User.current.language.to_sym :
Setting.default_language.to_sym
end
def extract_from(options)
label_options = options.dup
input_options = options.dup.except(:for, :label, :no_label, :prefix, :suffix)
[input_options, label_options]
end
def sanitized_object_name
object_name.to_s.gsub(/\]\[|[^-a-zA-Z0-9:.]/, '_').sub(/_$/, '')
end
def title_from_context(method)
if object.class.respond_to? :human_attribute_name
object.class.human_attribute_name method
else
method.to_s.camelize
end
end
end