kanbanworkflowstimelinescrumrubyroadmapproject-planningproject-managementopenprojectangularissue-trackerifcgantt-chartganttbug-trackerboardsbcf
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.
317 lines
11 KiB
317 lines
11 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
|
|
include ERB::Util
|
|
|
|
(field_helpers - %i(radio_button hidden_field fields_for label) + %i(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,
|
|
checked_value,
|
|
checked,
|
|
text = field.to_s + "_#{checked_value}",
|
|
options = {})
|
|
|
|
label_for = "#{sanitized_object_name}_#{field}_#{checked_value}".to_sym
|
|
unchecked_value = options.delete(:unchecked_value) { '' }
|
|
|
|
input_options = options.reverse_merge(multiple: true,
|
|
checked: checked,
|
|
for: label_for,
|
|
label: text)
|
|
|
|
check_box(field, input_options, checked_value, unchecked_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 = %i(
|
|
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: options[:no_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.to_s.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.to_s.gsub(/_field$/, '')}"
|
|
else
|
|
"form--#{selector.to_s.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)
|
|
else
|
|
l(field)
|
|
end
|
|
|
|
label_options = { class: '',
|
|
title: text }
|
|
|
|
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[:class] << ' -required' if options.delete(:required)
|
|
label_options[:for] = if options[:for]
|
|
options[:for]
|
|
elsif options[:multi_locale] && id
|
|
id.sub(/\_id$/, "_#{field}")
|
|
end
|
|
label_options[:lang] = options[:lang]
|
|
label_options.reject! do |_k, v| v.nil? end
|
|
|
|
@template.label(@object_name, field, h(text), 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 { |t|
|
|
t.send(field).present?
|
|
}
|
|
|
|
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
|
|
|