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