class Invoice < ApplicationRecord include CustomFields include LocalizeInput belongs_to :supplier belongs_to :created_by, class_name: 'User', foreign_key: 'created_by_user_id' belongs_to :financial_link, optional: true has_many :deliveries, dependent: :nullify has_many :orders, dependent: :nullify validates :supplier_id, presence: true validates :amount, :deposit, :deposit_credit, numericality: true validate :valid_attachment scope :unpaid, -> { where(paid_on: nil) } scope :without_financial_link, -> { where(financial_link: nil) } attr_accessor :delete_attachment # Replace numeric seperator with database format localize_input_of :amount, :deposit, :deposit_credit def attachment=(incoming_file) self.attachment_data = incoming_file.read # allow to soft-fail when FileMagic isn't present and removed from Gemfile (e.g. Heroku) self.attachment_mime = defined?(FileMagic) ? FileMagic.new(FileMagic::MAGIC_MIME).buffer(attachment_data) : 'application/octet-stream' end def delete_attachment=(value) return unless value == '1' self.attachment_data = nil self.attachment_mime = nil end def user_can_edit?(user) user.role_finance? || (user.role_invoices? && !paid_on && created_by.try(:id) == user.id) end # Amount without deposit def net_amount amount - deposit + deposit_credit end def orders_sum orders .joins(order_articles: [:article_price]) .sum('COALESCE(order_articles.units_received, order_articles.units_billed, order_articles.units_to_order)' \ + '* article_prices.unit_quantity' \ + '* ROUND((article_prices.price + article_prices.deposit) * (100 + article_prices.tax) / 100, 2)') end def orders_transport_sum orders.sum(:transport) end def expected_amount return net_amount unless orders.any? orders_sum + orders_transport_sum end protected def valid_attachment return unless attachment_data mime = MIME::Type.simplified(attachment_mime) return if ['application/pdf', 'image/jpeg'].include? mime errors.add :attachment, I18n.t('model.invoice.invalid_mime', mime: mime) end end