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.

custom_field.rb 9.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  1. # Redmine - project management software
  2. # Copyright (C) 2006-2017 Jean-Philippe Lang
  3. #
  4. # This program is free software; you can redistribute it and/or
  5. # modify it under the terms of the GNU General Public License
  6. # as published by the Free Software Foundation; either version 2
  7. # of the License, or (at your option) any later version.
  8. #
  9. # This program is distributed in the hope that it will be useful,
  10. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. # GNU General Public License for more details.
  13. #
  14. # You should have received a copy of the GNU General Public License
  15. # along with this program; if not, write to the Free Software
  16. # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
  17. class CustomField < ActiveRecord::Base
  18. include Redmine::SafeAttributes
  19. include Redmine::SubclassFactory
  20. has_many :enumerations,
  21. lambda { order(:position) },
  22. :class_name => 'CustomFieldEnumeration',
  23. :dependent => :delete_all
  24. has_many :custom_values, :dependent => :delete_all
  25. has_and_belongs_to_many :roles, :join_table => "#{table_name_prefix}custom_fields_roles#{table_name_suffix}", :foreign_key => "custom_field_id"
  26. acts_as_positioned
  27. serialize :possible_values
  28. store :format_store
  29. validates_presence_of :name, :field_format
  30. validates_uniqueness_of :name, :scope => :type
  31. validates_length_of :name, :maximum => 30
  32. validates_length_of :regexp, maximum: 255
  33. validates_inclusion_of :field_format, :in => Proc.new { Redmine::FieldFormat.available_formats }
  34. validate :validate_custom_field
  35. before_validation :set_searchable
  36. before_save do |field|
  37. field.format.before_custom_field_save(field)
  38. end
  39. after_save :handle_multiplicity_change
  40. after_save do |field|
  41. if field.saved_change_to_visible? && field.visible
  42. field.roles.clear
  43. end
  44. end
  45. scope :sorted, lambda { order(:position) }
  46. scope :visible, lambda {|*args|
  47. user = args.shift || User.current
  48. if user.admin?
  49. # nop
  50. elsif user.memberships.any?
  51. where("#{table_name}.visible = ? OR #{table_name}.id IN (SELECT DISTINCT cfr.custom_field_id FROM #{Member.table_name} m" +
  52. " INNER JOIN #{MemberRole.table_name} mr ON mr.member_id = m.id" +
  53. " INNER JOIN #{table_name_prefix}custom_fields_roles#{table_name_suffix} cfr ON cfr.role_id = mr.role_id" +
  54. " WHERE m.user_id = ?)",
  55. true, user.id)
  56. else
  57. where(:visible => true)
  58. end
  59. }
  60. def visible_by?(project, user=User.current)
  61. visible? || user.admin?
  62. end
  63. safe_attributes 'name',
  64. 'field_format',
  65. 'possible_values',
  66. 'regexp',
  67. 'min_length',
  68. 'max_length',
  69. 'is_required',
  70. 'is_for_all',
  71. 'is_filter',
  72. 'position',
  73. 'searchable',
  74. 'default_value',
  75. 'editable',
  76. 'visible',
  77. 'multiple',
  78. 'description',
  79. 'role_ids',
  80. 'url_pattern',
  81. 'text_formatting',
  82. 'edit_tag_style',
  83. 'user_role',
  84. 'version_status',
  85. 'extensions_allowed',
  86. 'full_width_layout'
  87. def format
  88. @format ||= Redmine::FieldFormat.find(field_format)
  89. end
  90. def field_format=(arg)
  91. # cannot change format of a saved custom field
  92. if new_record?
  93. @format = nil
  94. super
  95. end
  96. end
  97. def set_searchable
  98. # make sure these fields are not searchable
  99. self.searchable = false unless format.class.searchable_supported
  100. # make sure only these fields can have multiple values
  101. self.multiple = false unless format.class.multiple_supported
  102. true
  103. end
  104. def validate_custom_field
  105. format.validate_custom_field(self).each do |attribute, message|
  106. errors.add attribute, message
  107. end
  108. if regexp.present?
  109. begin
  110. Regexp.new(regexp)
  111. rescue
  112. errors.add(:regexp, :invalid)
  113. end
  114. end
  115. if default_value.present?
  116. validate_field_value(default_value).each do |message|
  117. errors.add :default_value, message
  118. end
  119. end
  120. end
  121. def possible_custom_value_options(custom_value)
  122. format.possible_custom_value_options(custom_value)
  123. end
  124. def possible_values_options(object=nil)
  125. if object.is_a?(Array)
  126. object.map {|o| format.possible_values_options(self, o)}.reduce(:&) || []
  127. else
  128. format.possible_values_options(self, object) || []
  129. end
  130. end
  131. def possible_values
  132. values = read_attribute(:possible_values)
  133. if values.is_a?(Array)
  134. values.each do |value|
  135. value.to_s.force_encoding('UTF-8')
  136. end
  137. values
  138. else
  139. []
  140. end
  141. end
  142. # Makes possible_values accept a multiline string
  143. def possible_values=(arg)
  144. if arg.is_a?(Array)
  145. values = arg.compact.map {|a| a.to_s.strip}.reject(&:blank?)
  146. write_attribute(:possible_values, values)
  147. else
  148. self.possible_values = arg.to_s.split(/[\n\r]+/)
  149. end
  150. end
  151. def set_custom_field_value(custom_field_value, value)
  152. format.set_custom_field_value(self, custom_field_value, value)
  153. end
  154. def cast_value(value)
  155. format.cast_value(self, value)
  156. end
  157. def value_from_keyword(keyword, customized)
  158. format.value_from_keyword(self, keyword, customized)
  159. end
  160. # Returns the options hash used to build a query filter for the field
  161. def query_filter_options(query)
  162. format.query_filter_options(self, query)
  163. end
  164. def totalable?
  165. format.totalable_supported
  166. end
  167. def full_width_layout?
  168. full_width_layout == '1'
  169. end
  170. # Returns a ORDER BY clause that can used to sort customized
  171. # objects by their value of the custom field.
  172. # Returns nil if the custom field can not be used for sorting.
  173. def order_statement
  174. return nil if multiple?
  175. format.order_statement(self)
  176. end
  177. # Returns a GROUP BY clause that can used to group by custom value
  178. # Returns nil if the custom field can not be used for grouping.
  179. def group_statement
  180. return nil if multiple?
  181. format.group_statement(self)
  182. end
  183. def join_for_order_statement
  184. format.join_for_order_statement(self)
  185. end
  186. def visibility_by_project_condition(project_key=nil, user=User.current, id_column=nil)
  187. if visible? || user.admin?
  188. "1=1"
  189. elsif user.anonymous?
  190. "1=0"
  191. else
  192. project_key ||= "#{self.class.customized_class.table_name}.project_id"
  193. id_column ||= id
  194. "#{project_key} IN (SELECT DISTINCT m.project_id FROM #{Member.table_name} m" +
  195. " INNER JOIN #{MemberRole.table_name} mr ON mr.member_id = m.id" +
  196. " INNER JOIN #{table_name_prefix}custom_fields_roles#{table_name_suffix} cfr ON cfr.role_id = mr.role_id" +
  197. " WHERE m.user_id = #{user.id} AND cfr.custom_field_id = #{id_column})"
  198. end
  199. end
  200. def <=>(field)
  201. position <=> field.position
  202. end
  203. # Returns the class that values represent
  204. def value_class
  205. format.target_class if format.respond_to?(:target_class)
  206. end
  207. def self.customized_class
  208. self.name =~ /^(.+)CustomField$/
  209. $1.constantize rescue nil
  210. end
  211. # to move in project_custom_field
  212. def self.for_all
  213. where(:is_for_all => true).order(:position).to_a
  214. end
  215. def type_name
  216. nil
  217. end
  218. # Returns the error messages for the given value
  219. # or an empty array if value is a valid value for the custom field
  220. def validate_custom_value(custom_value)
  221. value = custom_value.value
  222. errs = format.validate_custom_value(custom_value)
  223. unless errs.any?
  224. if value.is_a?(Array)
  225. if !multiple?
  226. errs << ::I18n.t('activerecord.errors.messages.invalid')
  227. end
  228. if is_required? && value.detect(&:present?).nil?
  229. errs << ::I18n.t('activerecord.errors.messages.blank')
  230. end
  231. else
  232. if is_required? && value.blank?
  233. errs << ::I18n.t('activerecord.errors.messages.blank')
  234. end
  235. end
  236. end
  237. errs
  238. end
  239. # Returns the error messages for the default custom field value
  240. def validate_field_value(value)
  241. validate_custom_value(CustomFieldValue.new(:custom_field => self, :value => value))
  242. end
  243. # Returns true if value is a valid value for the custom field
  244. def valid_field_value?(value)
  245. validate_field_value(value).empty?
  246. end
  247. def after_save_custom_value(custom_value)
  248. format.after_save_custom_value(self, custom_value)
  249. end
  250. def format_in?(*args)
  251. args.include?(field_format)
  252. end
  253. def self.human_attribute_name(attribute_key_name, *args)
  254. attr_name = attribute_key_name.to_s
  255. if attr_name == 'url_pattern'
  256. attr_name = "url"
  257. end
  258. super(attr_name, *args)
  259. end
  260. protected
  261. # Removes multiple values for the custom field after setting the multiple attribute to false
  262. # We kepp the value with the highest id for each customized object
  263. def handle_multiplicity_change
  264. if !new_record? && multiple_before_last_save && !multiple
  265. ids = custom_values.
  266. where("EXISTS(SELECT 1 FROM #{CustomValue.table_name} cve WHERE cve.custom_field_id = #{CustomValue.table_name}.custom_field_id" +
  267. " AND cve.customized_type = #{CustomValue.table_name}.customized_type AND cve.customized_id = #{CustomValue.table_name}.customized_id" +
  268. " AND cve.id > #{CustomValue.table_name}.id)").
  269. pluck(:id)
  270. if ids.any?
  271. custom_values.where(:id => ids).delete_all
  272. end
  273. end
  274. end
  275. end
  276. require_dependency 'redmine/field_format'