scinote-web/app/models/repository_column.rb
2020-11-10 17:24:04 +01:00

123 lines
4.3 KiB
Ruby

# frozen_string_literal: true
class RepositoryColumn < ApplicationRecord
belongs_to :repository, class_name: 'RepositoryBase'
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
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
enum data_type: Extends::REPOSITORY_DATA_TYPES
auto_strip_attributes :name, nullify: false
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
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') }
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 = (
Constants::REPOSITORY_TABLE_DEFAULT_STATE['columns'].length +
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!
end
def delimiter_char
Constants::REPOSITORY_LIST_ITEMS_DELIMITERS_MAP[metadata['delimiter']&.to_sym] || "\n"
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
end