annotate .svn/pristine/26/266dfa6887b7544baedd932a1d45fffff0382a86.svn-base @ 1519:afce8026aaeb redmine-2.4-integration

Merge from branch "live"
author Chris Cannam
date Tue, 09 Sep 2014 09:34:53 +0100
parents b450a9d58aed
children
rev   line source
Chris@1516 1 # Redmine - project management software
Chris@1516 2 # Copyright (C) 2006-2014 Jean-Philippe Lang
Chris@1516 3 #
Chris@1516 4 # This program is free software; you can redistribute it and/or
Chris@1516 5 # modify it under the terms of the GNU General Public License
Chris@1516 6 # as published by the Free Software Foundation; either version 2
Chris@1516 7 # of the License, or (at your option) any later version.
Chris@1516 8 #
Chris@1516 9 # This program is distributed in the hope that it will be useful,
Chris@1516 10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
Chris@1516 11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Chris@1516 12 # GNU General Public License for more details.
Chris@1516 13 #
Chris@1516 14 # You should have received a copy of the GNU General Public License
Chris@1516 15 # along with this program; if not, write to the Free Software
Chris@1516 16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
Chris@1516 17
Chris@1516 18 class CustomField < ActiveRecord::Base
Chris@1516 19 include Redmine::SubclassFactory
Chris@1516 20
Chris@1516 21 has_many :custom_values, :dependent => :delete_all
Chris@1516 22 has_and_belongs_to_many :roles, :join_table => "#{table_name_prefix}custom_fields_roles#{table_name_suffix}", :foreign_key => "custom_field_id"
Chris@1516 23 acts_as_list :scope => 'type = \'#{self.class}\''
Chris@1516 24 serialize :possible_values
Chris@1516 25
Chris@1516 26 validates_presence_of :name, :field_format
Chris@1516 27 validates_uniqueness_of :name, :scope => :type
Chris@1516 28 validates_length_of :name, :maximum => 30
Chris@1516 29 validates_inclusion_of :field_format, :in => Proc.new { Redmine::CustomFieldFormat.available_formats }
Chris@1516 30 validate :validate_custom_field
Chris@1516 31
Chris@1516 32 before_validation :set_searchable
Chris@1516 33 after_save :handle_multiplicity_change
Chris@1516 34 after_save do |field|
Chris@1516 35 if field.visible_changed? && field.visible
Chris@1516 36 field.roles.clear
Chris@1516 37 end
Chris@1516 38 end
Chris@1516 39
Chris@1516 40 scope :sorted, lambda { order("#{table_name}.position ASC") }
Chris@1516 41 scope :visible, lambda {|*args|
Chris@1516 42 user = args.shift || User.current
Chris@1516 43 if user.admin?
Chris@1516 44 # nop
Chris@1516 45 elsif user.memberships.any?
Chris@1516 46 where("#{table_name}.visible = ? OR #{table_name}.id IN (SELECT DISTINCT cfr.custom_field_id FROM #{Member.table_name} m" +
Chris@1516 47 " INNER JOIN #{MemberRole.table_name} mr ON mr.member_id = m.id" +
Chris@1516 48 " INNER JOIN #{table_name_prefix}custom_fields_roles#{table_name_suffix} cfr ON cfr.role_id = mr.role_id" +
Chris@1516 49 " WHERE m.user_id = ?)",
Chris@1516 50 true, user.id)
Chris@1516 51 else
Chris@1516 52 where(:visible => true)
Chris@1516 53 end
Chris@1516 54 }
Chris@1516 55
Chris@1516 56 def visible_by?(project, user=User.current)
Chris@1516 57 visible? || user.admin?
Chris@1516 58 end
Chris@1516 59
Chris@1516 60 def field_format=(arg)
Chris@1516 61 # cannot change format of a saved custom field
Chris@1516 62 super if new_record?
Chris@1516 63 end
Chris@1516 64
Chris@1516 65 def set_searchable
Chris@1516 66 # make sure these fields are not searchable
Chris@1516 67 self.searchable = false if %w(int float date bool).include?(field_format)
Chris@1516 68 # make sure only these fields can have multiple values
Chris@1516 69 self.multiple = false unless %w(list user version).include?(field_format)
Chris@1516 70 true
Chris@1516 71 end
Chris@1516 72
Chris@1516 73 def validate_custom_field
Chris@1516 74 if self.field_format == "list"
Chris@1516 75 errors.add(:possible_values, :blank) if self.possible_values.nil? || self.possible_values.empty?
Chris@1516 76 errors.add(:possible_values, :invalid) unless self.possible_values.is_a? Array
Chris@1516 77 end
Chris@1516 78
Chris@1516 79 if regexp.present?
Chris@1516 80 begin
Chris@1516 81 Regexp.new(regexp)
Chris@1516 82 rescue
Chris@1516 83 errors.add(:regexp, :invalid)
Chris@1516 84 end
Chris@1516 85 end
Chris@1516 86
Chris@1516 87 if default_value.present? && !valid_field_value?(default_value)
Chris@1516 88 errors.add(:default_value, :invalid)
Chris@1516 89 end
Chris@1516 90 end
Chris@1516 91
Chris@1516 92 def possible_values_options(obj=nil)
Chris@1516 93 case field_format
Chris@1516 94 when 'user', 'version'
Chris@1516 95 if obj.respond_to?(:project) && obj.project
Chris@1516 96 case field_format
Chris@1516 97 when 'user'
Chris@1516 98 obj.project.users.sort.collect {|u| [u.to_s, u.id.to_s]}
Chris@1516 99 when 'version'
Chris@1516 100 obj.project.shared_versions.sort.collect {|u| [u.to_s, u.id.to_s]}
Chris@1516 101 end
Chris@1516 102 elsif obj.is_a?(Array)
Chris@1516 103 obj.collect {|o| possible_values_options(o)}.reduce(:&)
Chris@1516 104 else
Chris@1516 105 []
Chris@1516 106 end
Chris@1516 107 when 'bool'
Chris@1516 108 [[l(:general_text_Yes), '1'], [l(:general_text_No), '0']]
Chris@1516 109 else
Chris@1516 110 possible_values || []
Chris@1516 111 end
Chris@1516 112 end
Chris@1516 113
Chris@1516 114 def possible_values(obj=nil)
Chris@1516 115 case field_format
Chris@1516 116 when 'user', 'version'
Chris@1516 117 possible_values_options(obj).collect(&:last)
Chris@1516 118 when 'bool'
Chris@1516 119 ['1', '0']
Chris@1516 120 else
Chris@1516 121 values = super()
Chris@1516 122 if values.is_a?(Array)
Chris@1516 123 values.each do |value|
Chris@1516 124 value.force_encoding('UTF-8') if value.respond_to?(:force_encoding)
Chris@1516 125 end
Chris@1516 126 values
Chris@1516 127 else
Chris@1516 128 []
Chris@1516 129 end
Chris@1516 130 end
Chris@1516 131 end
Chris@1516 132
Chris@1516 133 # Makes possible_values accept a multiline string
Chris@1516 134 def possible_values=(arg)
Chris@1516 135 if arg.is_a?(Array)
Chris@1516 136 super(arg.compact.collect(&:strip).select {|v| !v.blank?})
Chris@1516 137 else
Chris@1516 138 self.possible_values = arg.to_s.split(/[\n\r]+/)
Chris@1516 139 end
Chris@1516 140 end
Chris@1516 141
Chris@1516 142 def cast_value(value)
Chris@1516 143 casted = nil
Chris@1516 144 unless value.blank?
Chris@1516 145 case field_format
Chris@1516 146 when 'string', 'text', 'list'
Chris@1516 147 casted = value
Chris@1516 148 when 'date'
Chris@1516 149 casted = begin; value.to_date; rescue; nil end
Chris@1516 150 when 'bool'
Chris@1516 151 casted = (value == '1' ? true : false)
Chris@1516 152 when 'int'
Chris@1516 153 casted = value.to_i
Chris@1516 154 when 'float'
Chris@1516 155 casted = value.to_f
Chris@1516 156 when 'user', 'version'
Chris@1516 157 casted = (value.blank? ? nil : field_format.classify.constantize.find_by_id(value.to_i))
Chris@1516 158 end
Chris@1516 159 end
Chris@1516 160 casted
Chris@1516 161 end
Chris@1516 162
Chris@1516 163 def value_from_keyword(keyword, customized)
Chris@1516 164 possible_values_options = possible_values_options(customized)
Chris@1516 165 if possible_values_options.present?
Chris@1516 166 keyword = keyword.to_s.downcase
Chris@1516 167 if v = possible_values_options.detect {|text, id| text.downcase == keyword}
Chris@1516 168 if v.is_a?(Array)
Chris@1516 169 v.last
Chris@1516 170 else
Chris@1516 171 v
Chris@1516 172 end
Chris@1516 173 end
Chris@1516 174 else
Chris@1516 175 keyword
Chris@1516 176 end
Chris@1516 177 end
Chris@1516 178
Chris@1516 179 # Returns a ORDER BY clause that can used to sort customized
Chris@1516 180 # objects by their value of the custom field.
Chris@1516 181 # Returns nil if the custom field can not be used for sorting.
Chris@1516 182 def order_statement
Chris@1516 183 return nil if multiple?
Chris@1516 184 case field_format
Chris@1516 185 when 'string', 'text', 'list', 'date', 'bool'
Chris@1516 186 # COALESCE is here to make sure that blank and NULL values are sorted equally
Chris@1516 187 "COALESCE(#{join_alias}.value, '')"
Chris@1516 188 when 'int', 'float'
Chris@1516 189 # Make the database cast values into numeric
Chris@1516 190 # Postgresql will raise an error if a value can not be casted!
Chris@1516 191 # CustomValue validations should ensure that it doesn't occur
Chris@1516 192 "CAST(CASE #{join_alias}.value WHEN '' THEN '0' ELSE #{join_alias}.value END AS decimal(30,3))"
Chris@1516 193 when 'user', 'version'
Chris@1516 194 value_class.fields_for_order_statement(value_join_alias)
Chris@1516 195 else
Chris@1516 196 nil
Chris@1516 197 end
Chris@1516 198 end
Chris@1516 199
Chris@1516 200 # Returns a GROUP BY clause that can used to group by custom value
Chris@1516 201 # Returns nil if the custom field can not be used for grouping.
Chris@1516 202 def group_statement
Chris@1516 203 return nil if multiple?
Chris@1516 204 case field_format
Chris@1516 205 when 'list', 'date', 'bool', 'int'
Chris@1516 206 order_statement
Chris@1516 207 when 'user', 'version'
Chris@1516 208 "COALESCE(#{join_alias}.value, '')"
Chris@1516 209 else
Chris@1516 210 nil
Chris@1516 211 end
Chris@1516 212 end
Chris@1516 213
Chris@1516 214 def join_for_order_statement
Chris@1516 215 case field_format
Chris@1516 216 when 'user', 'version'
Chris@1516 217 "LEFT OUTER JOIN #{CustomValue.table_name} #{join_alias}" +
Chris@1516 218 " ON #{join_alias}.customized_type = '#{self.class.customized_class.base_class.name}'" +
Chris@1516 219 " AND #{join_alias}.customized_id = #{self.class.customized_class.table_name}.id" +
Chris@1516 220 " AND #{join_alias}.custom_field_id = #{id}" +
Chris@1516 221 " AND (#{visibility_by_project_condition})" +
Chris@1516 222 " AND #{join_alias}.value <> ''" +
Chris@1516 223 " AND #{join_alias}.id = (SELECT max(#{join_alias}_2.id) FROM #{CustomValue.table_name} #{join_alias}_2" +
Chris@1516 224 " WHERE #{join_alias}_2.customized_type = #{join_alias}.customized_type" +
Chris@1516 225 " AND #{join_alias}_2.customized_id = #{join_alias}.customized_id" +
Chris@1516 226 " AND #{join_alias}_2.custom_field_id = #{join_alias}.custom_field_id)" +
Chris@1516 227 " LEFT OUTER JOIN #{value_class.table_name} #{value_join_alias}" +
Chris@1516 228 " ON CAST(CASE #{join_alias}.value WHEN '' THEN '0' ELSE #{join_alias}.value END AS decimal(30,0)) = #{value_join_alias}.id"
Chris@1516 229 when 'int', 'float'
Chris@1516 230 "LEFT OUTER JOIN #{CustomValue.table_name} #{join_alias}" +
Chris@1516 231 " ON #{join_alias}.customized_type = '#{self.class.customized_class.base_class.name}'" +
Chris@1516 232 " AND #{join_alias}.customized_id = #{self.class.customized_class.table_name}.id" +
Chris@1516 233 " AND #{join_alias}.custom_field_id = #{id}" +
Chris@1516 234 " AND (#{visibility_by_project_condition})" +
Chris@1516 235 " AND #{join_alias}.value <> ''" +
Chris@1516 236 " AND #{join_alias}.id = (SELECT max(#{join_alias}_2.id) FROM #{CustomValue.table_name} #{join_alias}_2" +
Chris@1516 237 " WHERE #{join_alias}_2.customized_type = #{join_alias}.customized_type" +
Chris@1516 238 " AND #{join_alias}_2.customized_id = #{join_alias}.customized_id" +
Chris@1516 239 " AND #{join_alias}_2.custom_field_id = #{join_alias}.custom_field_id)"
Chris@1516 240 when 'string', 'text', 'list', 'date', 'bool'
Chris@1516 241 "LEFT OUTER JOIN #{CustomValue.table_name} #{join_alias}" +
Chris@1516 242 " ON #{join_alias}.customized_type = '#{self.class.customized_class.base_class.name}'" +
Chris@1516 243 " AND #{join_alias}.customized_id = #{self.class.customized_class.table_name}.id" +
Chris@1516 244 " AND #{join_alias}.custom_field_id = #{id}" +
Chris@1516 245 " AND (#{visibility_by_project_condition})" +
Chris@1516 246 " AND #{join_alias}.id = (SELECT max(#{join_alias}_2.id) FROM #{CustomValue.table_name} #{join_alias}_2" +
Chris@1516 247 " WHERE #{join_alias}_2.customized_type = #{join_alias}.customized_type" +
Chris@1516 248 " AND #{join_alias}_2.customized_id = #{join_alias}.customized_id" +
Chris@1516 249 " AND #{join_alias}_2.custom_field_id = #{join_alias}.custom_field_id)"
Chris@1516 250 else
Chris@1516 251 nil
Chris@1516 252 end
Chris@1516 253 end
Chris@1516 254
Chris@1516 255 def join_alias
Chris@1516 256 "cf_#{id}"
Chris@1516 257 end
Chris@1516 258
Chris@1516 259 def value_join_alias
Chris@1516 260 join_alias + "_" + field_format
Chris@1516 261 end
Chris@1516 262
Chris@1516 263 def visibility_by_project_condition(project_key=nil, user=User.current)
Chris@1516 264 if visible? || user.admin?
Chris@1516 265 "1=1"
Chris@1516 266 elsif user.anonymous?
Chris@1516 267 "1=0"
Chris@1516 268 else
Chris@1516 269 project_key ||= "#{self.class.customized_class.table_name}.project_id"
Chris@1516 270 "#{project_key} IN (SELECT DISTINCT m.project_id FROM #{Member.table_name} m" +
Chris@1516 271 " INNER JOIN #{MemberRole.table_name} mr ON mr.member_id = m.id" +
Chris@1516 272 " INNER JOIN #{table_name_prefix}custom_fields_roles#{table_name_suffix} cfr ON cfr.role_id = mr.role_id" +
Chris@1516 273 " WHERE m.user_id = #{user.id} AND cfr.custom_field_id = #{id})"
Chris@1516 274 end
Chris@1516 275 end
Chris@1516 276
Chris@1516 277 def self.visibility_condition
Chris@1516 278 if user.admin?
Chris@1516 279 "1=1"
Chris@1516 280 elsif user.anonymous?
Chris@1516 281 "#{table_name}.visible"
Chris@1516 282 else
Chris@1516 283 "#{project_key} IN (SELECT DISTINCT m.project_id FROM #{Member.table_name} m" +
Chris@1516 284 " INNER JOIN #{MemberRole.table_name} mr ON mr.member_id = m.id" +
Chris@1516 285 " INNER JOIN #{table_name_prefix}custom_fields_roles#{table_name_suffix} cfr ON cfr.role_id = mr.role_id" +
Chris@1516 286 " WHERE m.user_id = #{user.id} AND cfr.custom_field_id = #{id})"
Chris@1516 287 end
Chris@1516 288 end
Chris@1516 289
Chris@1516 290 def <=>(field)
Chris@1516 291 position <=> field.position
Chris@1516 292 end
Chris@1516 293
Chris@1516 294 # Returns the class that values represent
Chris@1516 295 def value_class
Chris@1516 296 case field_format
Chris@1516 297 when 'user', 'version'
Chris@1516 298 field_format.classify.constantize
Chris@1516 299 else
Chris@1516 300 nil
Chris@1516 301 end
Chris@1516 302 end
Chris@1516 303
Chris@1516 304 def self.customized_class
Chris@1516 305 self.name =~ /^(.+)CustomField$/
Chris@1516 306 $1.constantize rescue nil
Chris@1516 307 end
Chris@1516 308
Chris@1516 309 # to move in project_custom_field
Chris@1516 310 def self.for_all
Chris@1516 311 where(:is_for_all => true).order('position').all
Chris@1516 312 end
Chris@1516 313
Chris@1516 314 def type_name
Chris@1516 315 nil
Chris@1516 316 end
Chris@1516 317
Chris@1516 318 # Returns the error messages for the given value
Chris@1516 319 # or an empty array if value is a valid value for the custom field
Chris@1516 320 def validate_field_value(value)
Chris@1516 321 errs = []
Chris@1516 322 if value.is_a?(Array)
Chris@1516 323 if !multiple?
Chris@1516 324 errs << ::I18n.t('activerecord.errors.messages.invalid')
Chris@1516 325 end
Chris@1516 326 if is_required? && value.detect(&:present?).nil?
Chris@1516 327 errs << ::I18n.t('activerecord.errors.messages.blank')
Chris@1516 328 end
Chris@1516 329 value.each {|v| errs += validate_field_value_format(v)}
Chris@1516 330 else
Chris@1516 331 if is_required? && value.blank?
Chris@1516 332 errs << ::I18n.t('activerecord.errors.messages.blank')
Chris@1516 333 end
Chris@1516 334 errs += validate_field_value_format(value)
Chris@1516 335 end
Chris@1516 336 errs
Chris@1516 337 end
Chris@1516 338
Chris@1516 339 # Returns true if value is a valid value for the custom field
Chris@1516 340 def valid_field_value?(value)
Chris@1516 341 validate_field_value(value).empty?
Chris@1516 342 end
Chris@1516 343
Chris@1516 344 def format_in?(*args)
Chris@1516 345 args.include?(field_format)
Chris@1516 346 end
Chris@1516 347
Chris@1516 348 protected
Chris@1516 349
Chris@1516 350 # Returns the error message for the given value regarding its format
Chris@1516 351 def validate_field_value_format(value)
Chris@1516 352 errs = []
Chris@1516 353 unless value.to_s == ''
Chris@1516 354 errs << ::I18n.t('activerecord.errors.messages.invalid') unless regexp.blank? or value =~ Regexp.new(regexp)
Chris@1516 355 errs << ::I18n.t('activerecord.errors.messages.too_short', :count => min_length) if min_length && min_length > 0 && value.length < min_length
Chris@1516 356 errs << ::I18n.t('activerecord.errors.messages.too_long', :count => max_length) if max_length && max_length > 0 && value.length > max_length
Chris@1516 357
Chris@1516 358 # Format specific validations
Chris@1516 359 case field_format
Chris@1516 360 when 'int'
Chris@1516 361 errs << ::I18n.t('activerecord.errors.messages.not_a_number') unless value =~ /^[+-]?\d+$/
Chris@1516 362 when 'float'
Chris@1516 363 begin; Kernel.Float(value); rescue; errs << ::I18n.t('activerecord.errors.messages.invalid') end
Chris@1516 364 when 'date'
Chris@1516 365 errs << ::I18n.t('activerecord.errors.messages.not_a_date') unless value =~ /^\d{4}-\d{2}-\d{2}$/ && begin; value.to_date; rescue; false end
Chris@1516 366 when 'list'
Chris@1516 367 errs << ::I18n.t('activerecord.errors.messages.inclusion') unless possible_values.include?(value)
Chris@1516 368 end
Chris@1516 369 end
Chris@1516 370 errs
Chris@1516 371 end
Chris@1516 372
Chris@1516 373 # Removes multiple values for the custom field after setting the multiple attribute to false
Chris@1516 374 # We kepp the value with the highest id for each customized object
Chris@1516 375 def handle_multiplicity_change
Chris@1516 376 if !new_record? && multiple_was && !multiple
Chris@1516 377 ids = custom_values.
Chris@1516 378 where("EXISTS(SELECT 1 FROM #{CustomValue.table_name} cve WHERE cve.custom_field_id = #{CustomValue.table_name}.custom_field_id" +
Chris@1516 379 " AND cve.customized_type = #{CustomValue.table_name}.customized_type AND cve.customized_id = #{CustomValue.table_name}.customized_id" +
Chris@1516 380 " AND cve.id > #{CustomValue.table_name}.id)").
Chris@1516 381 pluck(:id)
Chris@1516 382
Chris@1516 383 if ids.any?
Chris@1516 384 custom_values.where(:id => ids).delete_all
Chris@1516 385 end
Chris@1516 386 end
Chris@1516 387 end
Chris@1516 388 end