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/config/initializers/10-patches.rb

274 lines
9.4 KiB

#-- encoding: UTF-8
#-- copyright
# OpenProject is a project management system.
#
# Copyright (C) 2012-2013 the OpenProject Team
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU General Public License version 3.
#
# See doc/COPYRIGHT.rdoc for more details.
#++
require 'active_record'
module ActiveRecord
class Base
include Redmine::I18n
# Translate attribute names for validation errors display
def self.human_attribute_name(attr, options = {})
begin
options_with_raise = {:raise => true, :default => false}.merge options
attr = attr.to_s.gsub(/_id$/, '')
super(attr, options_with_raise)
rescue I18n::MissingTranslationData => e
included_in_general_attributes = I18n.t('attributes').keys.map(&:to_s).include? attr
included_in_superclasses = ancestors.select { |a| a.ancestors.include? ActiveRecord::Base }.any? { |klass| !(I18n.t("activerecord.attributes.#{klass.name.underscore}.#{attr}").include? 'translation missing:') }
unless included_in_general_attributes or included_in_superclasses
# TODO: remove this method once no warning is displayed when running a server/console/tests/tasks etc.
warn "[DEPRECATION] Relying on Redmine::I18n addition of `field_` to your translation key \"#{attr}\" on the \"#{self}\" model is deprecated. Please use proper ActiveRecord i18n! \n Catched: #{e.message}"
end
super(attr, options) # without raise
end
end
end
end
module ActiveModel
class Errors
def full_messages
full_messages = []
@messages.each_key do |attribute|
@messages[attribute].each do |message|
next unless message
if attribute == :base
full_messages << message
elsif attribute == :custom_values
# Replace the generic "custom values is invalid"
# with the errors on custom values
@base.custom_values.each do |value|
full_messages += value.errors.map do |_, message|
I18n.t(:"errors.format", {
:default => "%{attribute} %{message}",
:attribute => value.custom_field.name,
:message => message
})
end
end
else
attr_name = attribute.to_s.gsub('.', '_').humanize
attr_name = @base.class.human_attribute_name(attribute, :default => attr_name)
full_messages << I18n.t(:"errors.format", {
:default => "%{attribute} %{message}",
:attribute => attr_name,
:message => message
})
end
end
end
full_messages
end
end
end
module ActionView
module Helpers
module AccessibleErrors
def self.included(base)
base.send(:include, InstanceMethods)
base.extend(ClassMethods)
end
module ClassMethods
def wrap_with_error_span(html_tag, object, method)
object_identifier = erroneous_object_identifier(object.object_id.to_s, method)
"<span id='#{object_identifier}' class=\"errorSpan\"><a name=\"#{object_identifier}\"></a>#{html_tag}</span>".html_safe
end
def erroneous_object_identifier(id, method)
# select boxes use name_id whereas the validation uses name
# we have to cut the '_id' of in order for the field to match
id + "_" + method.gsub("_id", "") + "_error"
end
end
module InstanceMethods
def error_message_list(objects)
objects.collect do |object|
error_messages = []
object.errors.each_error do |attr, error|
unless attr == "custom_values"
# Generating unique identifier in order to jump directly to the field with the error
object_identifier = erroneous_object_identifier(object.object_id.to_s, attr)
error_messages << [object.class.human_attribute_name(attr) + " " + error.message, object_identifier]
end
end
# excluding custom_values from the errors.each loop before
# as more than one error can be assigned to custom_values
# which would add to many error messages
if object.errors[:custom_values].any?
object.custom_values.each do |value|
value.errors.collect do |attr, msg|
# Generating unique identifier in order to jump directly to the field with the error
object_identifier = erroneous_object_identifier(value.object_id.to_s, attr)
error_messages << [value.custom_field.name + " " + msg, object_identifier]
end
end
end
error_message_list_elements(error_messages)
end
end
private
def erroneous_object_identifier(id, method)
self.class.erroneous_object_identifier(id, method)
end
def error_message_list_elements(array)
array.collect do |msg, identifier|
content_tag :li do
content_tag :a,
ERB::Util.html_escape(msg),
:href => "#" + identifier,
:class => "afocus"
end
end
end
end
end
module ActiveRecordHelper
def error_messages_for(*params)
options = params.extract_options!.symbolize_keys
if object = options.delete(:object)
objects = Array.wrap(object)
else
objects = params.collect {|object_name| instance_variable_get("@#{object_name}") }.compact
end
count = objects.inject(0) {|sum, object| sum + object.errors.count }
unless count.zero?
html = {}
[:id, :class].each do |key|
if options.include?(key)
value = options[key]
html[key] = value unless value.blank?
else
html[key] = 'errorExplanation'
end
end
options[:object_name] ||= params.first
I18n.with_options :locale => options[:locale], :scope => [:activerecord, :errors, :template] do |locale|
header_message = if options.include?(:header_message)
options[:header_message]
else
object_name = options[:object_name].to_s
object_name = I18n.t(object_name, :default => object_name.gsub('_', ' '), :scope => [:activerecord, :models], :count => 1)
locale.t :header, :count => count, :model => object_name
end
message = options.include?(:message) ? options[:message] : locale.t(:body)
contents = ''
contents << content_tag(options[:header_tag] || :h2, header_message) unless header_message.blank?
contents << content_tag(:p, message) unless message.blank?
contents << content_tag(:ul, error_message_list(objects))
content_tag(:div, contents.html_safe, html)
end
else
''
end
end
end
module DateHelper
# distance_of_time_in_words breaks when difference is greater than 30 years
def distance_of_date_in_words(from_date, to_date = 0, options = {})
from_date = from_date.to_date if from_date.respond_to?(:to_date)
to_date = to_date.to_date if to_date.respond_to?(:to_date)
distance_in_days = (to_date - from_date).abs
I18n.with_options :locale => options[:locale], :scope => :'datetime.distance_in_words' do |locale|
case distance_in_days
when 0..60 then locale.t :x_days, :count => distance_in_days.round
when 61..720 then locale.t :about_x_months, :count => (distance_in_days / 30).round
else locale.t :over_x_years, :count => (distance_in_days / 365).floor
end
end
end
end
end
end
ActionView::Base.send :include, ActionView::Helpers::AccessibleErrors
ActionView::Base.field_error_proc = Proc.new do |html_tag, instance|
if html_tag.include?("<label")
html_tag.to_s
else
ActionView::Base.wrap_with_error_span(html_tag, instance.object, instance.method_name)
end
end
module ActiveRecord
class Base
# active record 2.3 backport, was removed in 3.0, only used in Query
def self.merge_conditions(*conditions)
segments = []
conditions.each do |condition|
unless condition.blank?
sql = sanitize_sql(condition)
segments << sql unless sql.blank?
end
end
"(#{segments.join(') AND (')})" unless segments.empty?
end
end
class Errors
# def on_with_id_handling(attribute)
# attribute = attribute.to_s
# if attribute.ends_with? '_id'
# on_without_id_handling(attribute) || on_without_id_handling(attribute[0..-4])
# else
# on_without_id_handling(attribute)
# end
# end
# alias_method_chain :on, :id_handling
end
end
module CollectiveIdea
module Acts
module NestedSet
module Model
# fixes IssueNestedSetTest#test_destroy_parent_issue_updated_during_children_destroy
def destroy_descendants_with_reload
destroy_descendants_without_reload
# Reload is needed because children may have updated their parent (self) during deletion.
# fixes stale object error in issue_nested_set_test
reload
end
alias_method_chain :destroy_descendants, :reload
end
end
end
end