scinote-web/app/models/repository_stock_value.rb
aignatov-bio ca53c3058a
Stock modal bugs [SCI-6667] (#3970)
Co-authored-by: Anton <anton@scinote.net>
2022-03-30 14:03:07 +02:00

168 lines
5.4 KiB
Ruby

# frozen_string_literal: true
class RepositoryStockValue < ApplicationRecord
include RepositoryValueWithReminders
include ActionView::Helpers::NumberHelper
belongs_to :repository_stock_unit_item, optional: true
belongs_to :created_by, class_name: 'User', optional: true, inverse_of: :created_repository_stock_values
belongs_to :last_modified_by, class_name: 'User', optional: true, inverse_of: :modified_repository_stock_values
has_one :repository_cell, as: :value, dependent: :destroy, inverse_of: :value
has_many :repository_ledger_records, dependent: :destroy
accepts_nested_attributes_for :repository_cell
validates :repository_cell, presence: true
SORTABLE_COLUMN_NAME = 'repository_stock_values.amount'
def self.enabled?
ApplicationSettings.instance.values['stock_management_enabled']
end
def formatted
"#{formatted_value} #{repository_stock_unit_item&.data}"
end
def formatted_value
if repository_cell
number_with_precision(
amount,
precision: (repository_cell.repository_column.metadata['decimals'].to_i || 0),
strip_insignificant_zeros: true
)
end
end
def formatted_treshold
if repository_cell && low_stock_threshold
number_with_precision(
low_stock_threshold,
precision: (repository_cell.repository_column.metadata['decimals'].to_i || 0),
strip_insignificant_zeros: true
)
end
end
def low_stock?
return false unless low_stock_threshold
amount <= low_stock_threshold
end
def self.add_filter_condition(repository_rows, join_alias, filter_element)
parameters = filter_element.parameters
if filter_element.operator == 'between'
return repository_rows if parameters['from'].blank? || parameters['to'].blank?
elsif parameters['value'].blank?
return repository_rows
end
repository_rows = case parameters['stock_unit']
when 'all'
repository_rows.where("#{join_alias}.repository_stock_unit_item_id IS NOT NULL")
when 'none'
repository_rows.where("#{join_alias}.repository_stock_unit_item_id IS NULL")
else
repository_rows.where("#{join_alias}.repository_stock_unit_item_id = ?", parameters['stock_unit'])
end
case filter_element.operator
when 'equal_to'
repository_rows.where("#{join_alias}.amount = ?", parameters['value'].to_d)
when 'unequal_to'
repository_rows.where.not("#{join_alias}.amount = ?", parameters['value'].to_d)
when 'greater_than'
repository_rows.where("#{join_alias}.amount > ?", parameters['value'].to_d)
when 'greater_than_or_equal_to'
repository_rows.where("#{join_alias}.amount >= ?", parameters['value'].to_d)
when 'less_than'
repository_rows.where("#{join_alias}.amount < ?", parameters['value'].to_d)
when 'less_than_or_equal_to'
repository_rows.where("#{join_alias}.amount <= ?", parameters['value'].to_d)
when 'between'
repository_rows
.where("#{join_alias}.amount > ? AND #{join_alias}.amount < ?", parameters['from'].to_d, parameters['to'].to_d)
else
raise ArgumentError, 'Wrong operator for RepositoryStockValue!'
end
end
def data_changed?(new_data)
BigDecimal(new_data.to_s) != data
end
def update_data!(new_data, user)
self.amount = BigDecimal(new_data[:amount].to_s)
self.low_stock_threshold = new_data[:low_stock_threshold]
self.last_modified_by = user
save!
end
def snapshot!(cell_snapshot)
value_snapshot = dup
stock_unit_item =
if repository_stock_unit_item.present?
cell_snapshot.repository_column
.repository_stock_unit_items
.find { |item| item.data == repository_stock_unit_item.data }
end
value_snapshot.assign_attributes(
repository_cell: cell_snapshot,
repository_stock_unit_item: stock_unit_item,
created_at: created_at,
updated_at: updated_at
)
value_snapshot.save!
end
def data
amount
end
def self.new_with_payload(payload, attributes)
value = new(attributes)
value.amount = payload[:amount]
value.low_stock_threshold = payload[:low_stock_threshold]
value
end
def self.import_from_text(text, attributes, _options = {})
digit, unit = text.match(/(^\d*\.?\d*)(\D*)/).captures
digit.strip!
unit.strip!
return nil if digit.blank?
value = new(attributes.merge(amount: BigDecimal(digit)))
return value if unit.blank?
column = attributes.dig(:repository_cell_attributes, :repository_column)
stock_unit_item = column.repository_stock_unit_items.find { |item| item.data == unit }
if stock_unit_item.blank?
stock_unit_item = column.repository_stock_unit_items.new(data: unit,
created_by: value.created_by,
last_modified_by: value.last_modified_by)
return value unless stock_unit_item.save
end
value.repository_stock_unit_item = stock_unit_item
value
rescue ArgumentError
nil
end
def update_stock_with_ledger!(amount, reference, comment)
delta = amount.to_d - self.amount.to_d
repository_ledger_records.create!(
user: last_modified_by,
amount: delta,
balance: amount,
reference: reference,
comment: comment
)
end
alias export_formatted formatted
end