mirror of
https://github.com/we-promise/sure.git
synced 2026-04-08 06:44:52 +00:00
* Add tax treatment support for accounts, investments, and cryptos * Replace hardcoded region labels with I18n translations * Add I18n support for subtype labels with fallback to hardcoded values * fixed schema --------- Co-authored-by: luckyPipewrench <luckypipewrench@proton.me>
101 lines
2.4 KiB
Ruby
101 lines
2.4 KiB
Ruby
module Accountable
|
|
extend ActiveSupport::Concern
|
|
|
|
TYPES = %w[Depository Investment Crypto Property Vehicle OtherAsset CreditCard Loan OtherLiability]
|
|
|
|
# Define empty hash to ensure all accountables have this defined
|
|
SUBTYPES = {}.freeze
|
|
|
|
def self.from_type(type)
|
|
return nil unless TYPES.include?(type)
|
|
type.constantize
|
|
end
|
|
|
|
included do
|
|
include Enrichable
|
|
|
|
has_one :account, as: :accountable, touch: true
|
|
end
|
|
|
|
class_methods do
|
|
def classification
|
|
raise NotImplementedError, "Accountable must implement #classification"
|
|
end
|
|
|
|
def icon
|
|
raise NotImplementedError, "Accountable must implement #icon"
|
|
end
|
|
|
|
def color
|
|
raise NotImplementedError, "Accountable must implement #color"
|
|
end
|
|
|
|
# Given a subtype, look up the label for this accountable type
|
|
# Uses i18n with fallback to hardcoded SUBTYPES values
|
|
def subtype_label_for(subtype, format: :short)
|
|
return nil if subtype.nil?
|
|
|
|
label_type = format == :long ? :long : :short
|
|
fallback = self::SUBTYPES.dig(subtype, label_type)
|
|
|
|
I18n.t(
|
|
"#{name.underscore.pluralize}.subtypes.#{subtype}.#{label_type}",
|
|
default: fallback
|
|
)
|
|
end
|
|
|
|
# Convenience method for getting the short label
|
|
def short_subtype_label_for(subtype)
|
|
subtype_label_for(subtype, format: :short)
|
|
end
|
|
|
|
# Convenience method for getting the long label
|
|
def long_subtype_label_for(subtype)
|
|
subtype_label_for(subtype, format: :long)
|
|
end
|
|
|
|
def favorable_direction
|
|
classification == "asset" ? "up" : "down"
|
|
end
|
|
|
|
def display_name
|
|
self.name.pluralize.titleize
|
|
end
|
|
|
|
def balance_money(family)
|
|
family.accounts
|
|
.active
|
|
.joins(sanitize_sql_array([
|
|
"LEFT JOIN exchange_rates ON exchange_rates.date = :current_date AND accounts.currency = exchange_rates.from_currency AND exchange_rates.to_currency = :family_currency",
|
|
{ current_date: Date.current.to_s, family_currency: family.currency }
|
|
]))
|
|
.where(accountable_type: self.name)
|
|
.sum("accounts.balance * COALESCE(exchange_rates.rate, 1)")
|
|
end
|
|
end
|
|
|
|
def display_name
|
|
self.class.display_name
|
|
end
|
|
|
|
def balance_display_name
|
|
"account value"
|
|
end
|
|
|
|
def opening_balance_display_name
|
|
"opening balance"
|
|
end
|
|
|
|
def icon
|
|
self.class.icon
|
|
end
|
|
|
|
def color
|
|
self.class.color
|
|
end
|
|
|
|
def classification
|
|
self.class.classification
|
|
end
|
|
end
|