# frozen_string_literal: true class RepositoryColumn < ApplicationRecord belongs_to :repository, class_name: 'RepositoryBase' belongs_to :repository_snapshot, foreign_key: :repository_id, optional: true belongs_to :created_by, foreign_key: :created_by_id, class_name: 'User' has_many :repository_cells, dependent: :destroy has_many :repository_rows, through: :repository_cells has_many :repository_list_items, -> { order('data ASC') }, dependent: :destroy, index_errors: true, inverse_of: :repository_column has_many :repository_status_items, -> { order('status ASC') }, dependent: :destroy, index_errors: true, inverse_of: :repository_column has_many :repository_checklist_items, -> { order('data ASC') }, dependent: :destroy, index_errors: true, inverse_of: :repository_column has_many :repository_stock_unit_items, -> { order('data ASC') }, dependent: :destroy, index_errors: true, inverse_of: :repository_column has_many :repository_table_filter_elements, dependent: :destroy accepts_nested_attributes_for :repository_status_items, allow_destroy: true accepts_nested_attributes_for :repository_list_items, allow_destroy: true accepts_nested_attributes_for :repository_checklist_items, allow_destroy: true accepts_nested_attributes_for :repository_stock_unit_items, allow_destroy: true enum data_type: Extends::REPOSITORY_DATA_TYPES store_accessor :metadata, %i(reminder_delta reminder_value reminder_unit reminder_message) validates :data_type, uniqueness: { if: :repository_stock_value?, scope: :repository_id } validates :data_type, uniqueness: { if: :repository_stock_consumption_value?, scope: :repository_id } validates :name, length: { maximum: Constants::NAME_MAX_LENGTH }, uniqueness: { scope: :repository_id, case_sensitive: true } validates :name, :data_type, :repository, :created_by, presence: true after_create :update_repository_table_states_with_new_column after_update :clear_hidden_repository_cell_reminders before_destroy :prevent_stock_column_destroy around_destroy :update_repository_table_states_with_removed_column scope :list_type, -> { where(data_type: 'RepositoryListValue') } scope :asset_type, -> { where(data_type: 'RepositoryAssetValue') } scope :status_type, -> { where(data_type: 'RepositoryStatusValue') } scope :checkbox_type, -> { where(data_type: 'RepositoryChecklistValue') } scope :stock_type, -> { where(data_type: 'RepositoryStockValue') } scope :stock_consumption_type, -> { where(data_type: 'RepositoryStockConsumptionValue') } def self.name_like(query) where('repository_columns.name ILIKE ?', "%#{query}%") end # Add enum check method with underscores (eg repository_list_value) data_types.each do |k, _| define_method "#{k.underscore}?" do public_send "#{k}?" end end def update_repository_table_states_with_new_column service = RepositoryTableStateColumnUpdateService.new service.update_states_with_new_column(repository) end def update_repository_table_states_with_removed_column # Calculate old_column_index - this can only be done before # record is deleted when we still have its index old_column_index = ( repository.default_columns_count + repository.repository_columns .order(id: :asc) .pluck(:id) .index(id) ) # Perform the destroy itself yield # Update repository table states service = RepositoryTableStateColumnUpdateService.new service.update_states_with_removed_column( repository, old_column_index ) end def importable? Extends::REPOSITORY_IMPORTABLE_TYPES.include?(data_type.to_sym) end def deep_dup new_column = super extra_method_name = "#{data_type.underscore}_deep_dup" __send__(extra_method_name, new_column) if respond_to?(extra_method_name, true) new_column end def snapshot!(repository_snapshot) column_snapshot = deep_dup column_snapshot.assign_attributes( repository: repository_snapshot, parent_id: id, created_at: created_at, updated_at: updated_at ) column_snapshot.save! snapshot_stock_consumption!(repository_snapshot) if repository_stock_value? column_snapshot end def delimiter_char Constants::REPOSITORY_LIST_ITEMS_DELIMITERS_MAP[metadata['delimiter']&.to_sym] || "\n" end def deletable? data_type != 'RepositoryStockValue' end def items items_method_name = "#{data_type.chomp('Value').underscore}_items" items_method_name = 'repository_stock_unit_items' if data_type == 'RepositoryStockValue' __send__(items_method_name) if respond_to?(items_method_name, true) end private def repository_list_value_deep_dup(new_column) repository_list_items.each do |item| new_column.repository_list_items << item.deep_dup end end def repository_checklist_value_deep_dup(new_column) repository_checklist_items.each do |item| new_column.repository_checklist_items << item.deep_dup end end def repository_status_value_deep_dup(new_column) repository_status_items.each do |item| new_column.repository_status_items << item.deep_dup end end def repository_stock_value_deep_dup(new_column) repository_stock_unit_items.each do |item| new_column.repository_stock_unit_items << item.deep_dup end end def snapshot_stock_consumption!(repository_snapshot) column_snapshot = deep_dup column_snapshot.assign_attributes( name: I18n.t('repositories.table.row_consumption'), repository_snapshot: repository_snapshot, data_type: 'RepositoryStockConsumptionValue', parent_id: nil ) column_snapshot.save! end def clear_hidden_repository_cell_reminders return unless reminder_delta_changed? HiddenRepositoryCellReminder.joins(repository_cell: :repository_column) .where(repository_columns: { id: id }) .delete_all end def prevent_stock_column_destroy raise NotImplementedError unless deletable? end end