# Article category class ArticleCategory < ApplicationRecord # @!attribute name # @return [String] Title of the category. # @!attrubute description # @return [String] Description (currently unused) # @!attribute articles # @return [Array
] Articles with this category. has_many :articles # @!attribute order_articles # @return [Array] Order articles with this category. has_many :order_articles, through: :articles # @!attribute orders # @return [Array] Orders with articles in this category. has_many :orders, through: :order_articles normalize_attributes :name, :description validates :name, presence: true, uniqueness: true, length: { minimum: 2 } before_destroy :check_for_associated_articles def self.ransackable_attributes(_auth_object = nil) %w[id name] end def self.ransackable_associations(_auth_object = nil) %w[articles order_articles orders] end # Find a category that matches a category name; may return nil. # TODO more intelligence like remembering earlier associations (global and/or per-supplier) def self.find_match(category) return if category.blank? || category.length < 3 c = nil ## exact match - not needed, will be returned by next query as well # c ||= ArticleCategory.where(name: category).first # case-insensitive substring match (take the closest match = shortest) c = ArticleCategory.where('name LIKE ?', "%#{category}%") unless c && c.any? # case-insensitive phrase present in category description unless c && c.any? c = ArticleCategory.where('description LIKE ?', "%#{category}%").select do |s| s.description.match(/(^|,)\s*#{category}\s*(,|$)/i) end end # return closest match if there are multiple c = c.sort_by { |s| s.name.length }.first if c.respond_to? :sort_by c end protected # Deny deleting the category when there are associated articles. def check_for_associated_articles return unless articles.undeleted.exists? raise I18n.t('activerecord.errors.has_many_left', collection: Article.model_name.human) end end