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 10KB

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