Class: Mint::Money
- Inherits:
-
Object
- Object
- Mint::Money
- Includes:
- Comparable
- Defined in:
- lib/minting/mint/i18n.rb,
lib/minting/money/clamp.rb,
lib/minting/money/money.rb,
lib/minting/money/coercion.rb,
lib/minting/money/comparable.rb,
lib/minting/money/conversion.rb,
lib/minting/money/format/to_s.rb,
lib/minting/money/constructors.rb,
lib/minting/money/allocation/split.rb,
lib/minting/money/format/formatting.rb,
lib/minting/money/arithmetics/methods.rb,
lib/minting/money/allocation/allocation.rb,
lib/minting/money/arithmetics/operators.rb
Overview
:nodoc:
Constant Summary collapse
- DEFAULT_FORMAT =
The default display format pattern for formatting monetary values. Uses ‘%<symbol>s` for the currency symbol and `%<amount>f` for the rounded amount.
'%<symbol>s%<amount>f'- PRESETS =
{ amount: { format: '%<amount>f' }, accounting: { format: { negative: '(%<symbol>s%<amount>f)' } }, european: { format: '%<amount>f %<symbol>s', decimal: ',', thousand: '.' }, currency: { format: '%<currency>s %<amount>f' } }.freeze
Instance Attribute Summary collapse
-
#amount ⇒ Object
readonly
Returns the value of attribute amount.
-
#currency ⇒ Object
readonly
Returns the value of attribute currency.
Class Method Summary collapse
-
.create(amount, currency) ⇒ Object
Backwards-compatible alias for previous API TODO: deprecate in a future major release.
-
.from(amount, currency) ⇒ Money
Creates a new Money immutable object with the specified amount and currency.
-
.from_subunits(subunits, currency) ⇒ Money
Builds a Money from a subunit (smallest-unit) Integer amount.
-
.no_currency(amount) ⇒ Money
Creates a new Money without a currency (ISO 4217 XXX — “No Currency”).
-
.parse(input, currency = nil) ⇒ Money?
Parses a human-readable money string into a Money object.
-
.parse!(input, currency = nil) ⇒ Money
Like Money.parse but raises on failure.
-
.zero(currency) ⇒ Money
Returns a frozen zero Money in the given currency.
Instance Method Summary collapse
-
#*(multiplicand) ⇒ Money
Performs multiplication of the monetary value by a standard scalar Numeric.
-
#**(exponent) ⇒ Money
Performs exponentiation of the monetary value by a standard scalar Numeric.
-
#+(addend) ⇒ Money
Performs addition with another Money instance or standard zero Numeric.
-
#-(subtrahend) ⇒ Money
Performs subtraction with another Money instance or standard zero Numeric.
-
#-@ ⇒ Money
Unary negation operator.
-
#/(divisor) ⇒ Money, Numeric
Performs division of the monetary value by a scalar Numeric or identical currency Money.
- #<=>(other) ⇒ Object
-
#==(other) ⇒ Object
True if both are zero, or both have same amount and same currency.
-
#abs ⇒ Money
Returns the absolute value of the monetary amount as a new Money instance.
-
#allocate(proportions) ⇒ Array<Money>
Proportionally allocates the monetary amount among a list of ratios.
-
#clamp(min_or_range, max = nil) ⇒ Money
Constrains
selfto the inclusive range [min,max]. -
#coerce(other) ⇒ Array(CoercedNumber, Money)
private
Allows Money to interact seamlessly as the right-hand operand in Numeric arithmetic.
-
#copy_with(amount:) ⇒ Money
Returns a new Money object with the specified amount, or self if unchanged.
-
#currency_code ⇒ String
Returns the ISO 3-letter currency code string.
-
#eql?(other) ⇒ Boolean
Strict equality — both amount and currency must match exactly.
-
#fractional ⇒ Object
Returns the fractional part of the amount.
-
#hash ⇒ Integer
Generates a stable hash key for Money instances.
-
#inspect ⇒ String
Returns a standard developer-oriented string inspection of the Money object.
-
#mint(new_amount) ⇒ Money
deprecated
Deprecated.
Use #copy_with instead. Will be removed in v2.
-
#negative? ⇒ Boolean
Returns true if the monetary amount is less than zero.
-
#nonzero? ⇒ self?
Self if amount is non-zero, nil otherwise.
-
#positive? ⇒ Boolean
Returns true if the monetary amount is greater than zero.
-
#same_currency?(other) ⇒ Boolean
Helper method to verify if another Money has the identical currency.
-
#split(slices) ⇒ Array<Money>
Splits the monetary amount into a given quantity of equal parts.
-
#subunits ⇒ Integer
Returns the monetary amount expressed in the currency’s smallest unit (fractional units).
-
#succ ⇒ Money
Returns the successor of the Money instance by adding the minimum possible subunit amount.
-
#to_d ⇒ BigDecimal
Converts the monetary amount to a BigDecimal object.
-
#to_f ⇒ Float
Converts the monetary amount to a standard float.
-
#to_hash ⇒ Hash
Returns a Hash representation of the money instance.
-
#to_html(format = DEFAULT_FORMAT) ⇒ String
Renders a safe HTML5 ‘<data>` element containing the formatted currency.
-
#to_i ⇒ Integer
Truncates and converts the monetary amount to an Integer.
-
#to_json(*_args) ⇒ String
Serializes the money instance to a standard JSON object containing the amount and currency.
-
#to_r ⇒ Rational
Returns the exact internal Rational representation of the monetary amount.
-
#to_s(preset = nil, format: nil, decimal: nil, thousand: nil, width: nil) ⇒ String
Formats money as a string with customizable format, thousand delimiter, and decimal.
-
#zero? ⇒ Boolean
True if amount is zero.
Instance Attribute Details
#amount ⇒ Object (readonly)
Returns the value of attribute amount.
24 25 26 |
# File 'lib/minting/money/money.rb', line 24 def amount @amount end |
#currency ⇒ Object (readonly)
Returns the value of attribute currency.
24 25 26 |
# File 'lib/minting/money/money.rb', line 24 def currency @currency end |
Class Method Details
.create(amount, currency) ⇒ Object
Backwards-compatible alias for previous API TODO: deprecate in a future major release
69 70 71 72 |
# File 'lib/minting/money/constructors.rb', line 69 def self.create(amount, currency) warn 'Money.create is now deprecated. Use Money.from' from(amount, currency) end |
.from(amount, currency) ⇒ Money
Creates a new Money immutable object with the specified amount and currency
13 14 15 16 17 18 19 20 |
# File 'lib/minting/money/constructors.rb', line 13 def self.from(amount, currency) raise ArgumentError, 'amount must be Numeric' unless amount.is_a?(Numeric) currency = Currency.resolve!(currency) amount = currency.normalize_amount(amount) amount.zero? ? currency.zero : new(amount, currency) end |
.from_subunits(subunits, currency) ⇒ Money
Builds a Money from a subunit (smallest-unit) Integer amount. This is the inverse of #subunits: for USD, the subunit is 1 cent; for JPY it is 1 yen; for IQD it is 1 dinar (subunit 3).
92 93 94 95 96 97 98 |
# File 'lib/minting/money/constructors.rb', line 92 def self.from_subunits(subunits, currency) raise ArgumentError, 'subunits must be an Integer' unless subunits.is_a?(Integer) currency = Currency.resolve!(currency) amount = Rational(subunits, currency.fractional_multiplier) amount.zero? ? currency.zero : new(amount, currency) end |
.no_currency(amount) ⇒ Money
Creates a new Money without a currency (ISO 4217 XXX — “No Currency”).
29 |
# File 'lib/minting/money/constructors.rb', line 29 def self.no_currency(amount) = from(amount, 'XXX') |
.parse(input, currency = nil) ⇒ Money?
Parses a human-readable money string into a Mint::Money object.
Returns nil when the input is invalid or currency cannot be determined.
46 |
# File 'lib/minting/money/constructors.rb', line 46 def self.parse(input, currency = nil) = Mint.parse(input, currency) |
Instance Method Details
#*(multiplicand) ⇒ Money
Performs multiplication of the monetary value by a standard scalar Numeric.
42 43 44 45 46 |
# File 'lib/minting/money/arithmetics/operators.rb', line 42 def *(multiplicand) raise TypeError, "#{self} can't be multiplied by #{multiplicand}" unless multiplicand.is_a?(Numeric) copy_with(amount: amount * multiplicand) end |
#**(exponent) ⇒ Money
Performs exponentiation of the monetary value by a standard scalar Numeric.
67 68 69 70 71 |
# File 'lib/minting/money/arithmetics/operators.rb', line 67 def **(exponent) return copy_with(amount: amount**exponent) if exponent.is_a?(Numeric) raise TypeError, "#{self} can't be powered by #{exponent}" end |
#+(addend) ⇒ Money
Performs addition with another Mint::Money instance or standard zero Numeric.
11 12 13 14 15 16 17 |
# File 'lib/minting/money/arithmetics/operators.rb', line 11 def +(addend) case addend in 0 then self in Money if same_currency?(addend) then copy_with(amount: amount + addend.amount) else raise TypeError, "#{addend} can't be added to #{self}" end end |
#-(subtrahend) ⇒ Money
Performs subtraction with another Mint::Money instance or standard zero Numeric.
24 25 26 27 28 29 30 |
# File 'lib/minting/money/arithmetics/operators.rb', line 24 def -(subtrahend) case subtrahend when 0 then return self when Money then return copy_with(amount: amount - subtrahend.amount) if same_currency?(subtrahend) end raise TypeError, "#{subtrahend} can't be subtracted from #{self}" end |
#-@ ⇒ Money
Unary negation operator. Returns a new Mint::Money instance with the inverted sign.
35 |
# File 'lib/minting/money/arithmetics/operators.rb', line 35 def -@ = copy_with(amount: -amount) |
#/(divisor) ⇒ Money, Numeric
Performs division of the monetary value by a scalar Numeric or identical currency Mint::Money.
54 55 56 57 58 59 60 |
# File 'lib/minting/money/arithmetics/operators.rb', line 54 def /(divisor) case divisor when Numeric then return copy_with(amount: amount / divisor) when Money then return amount / divisor.amount if same_currency? divisor end raise TypeError, "#{self} can't be divided by #{divisor}" end |
#<=>(other) ⇒ Object
36 37 38 39 40 41 42 |
# File 'lib/minting/money/comparable.rb', line 36 def <=>(other) case other in 0 then amount <=> other in Mint::Money if same_currency?(other) then amount <=> other.amount else raise TypeError, "#{inspect} can't be compared to #{other.inspect}" end end |
#==(other) ⇒ Object
Returns true if both are zero, or both have same amount and same currency.
9 10 11 12 13 14 15 |
# File 'lib/minting/money/comparable.rb', line 9 def ==(other) case other when 0 then zero? when Mint::Money then amount == other.amount && currency == other.currency else false end end |
#abs ⇒ Money
Returns the absolute value of the monetary amount as a new Mint::Money instance.
9 |
# File 'lib/minting/money/arithmetics/methods.rb', line 9 def abs = copy_with(amount: amount.abs) |
#allocate(proportions) ⇒ Array<Money>
Proportionally allocates the monetary amount among a list of ratios. Disperses any subunit rounding amounts across the initial slots
15 16 17 18 19 20 21 22 |
# File 'lib/minting/money/allocation/allocation.rb', line 15 def allocate(proportions) whole = proportions.sum.to_r raise ArgumentError, 'Need at least 1 proportion element' if proportions.empty? raise ArgumentError, 'Proportions total must not be zero' if whole.zero? amounts = proportions.map { |rate| currency.normalize_amount(Rational(amount * rate, whole)) } allocate_left_over(amounts: amounts, left_over: amount - amounts.sum) end |
#clamp(min_or_range, max = nil) ⇒ Money
Constrains self to the inclusive range [min, max].
Bounds may be:
-
nil meaning no boundary
-
same-currency Mint::Money or Range
-
Numeric amount, or Range
Numeric is interpreted as an amount in self‘s currency, so the common pricing idiom price.clamp(0, 100) reads as “0 to 100 in the same currency as price”.
When self is already in range the receiver is returned (no new object allocated). When out of range, the nearest bound is returned as a new frozen Mint::Money in self‘s currency.
42 43 44 45 46 47 48 49 50 51 |
# File 'lib/minting/money/clamp.rb', line 42 def clamp(min_or_range, max = nil) if min_or_range.is_a?(Range) raise(ArgumentError, "Either amount range alone or two amounts accepted: #{max}") if max min, max = min_or_range.minmax else min = min_or_range end copy_with(amount: amount.clamp(normalize_boundary(min), normalize_boundary(max))) end |
#coerce(other) ⇒ Array(CoercedNumber, Money)
This method is part of a private API. You should avoid using this method if possible, as it may be removed or be changed in the future.
Allows Mint::Money to interact seamlessly as the right-hand operand in Numeric arithmetic. This enables expressions like ‘5 * money` where `5` is a Numeric and `money` is a Money object.
14 15 16 |
# File 'lib/minting/money/coercion.rb', line 14 def coerce(other) [CoercedNumber.new(other), self] end |
#copy_with(amount:) ⇒ Money
Returns a new Money object with the specified amount, or self if unchanged. This is the primary method for creating a modified copy of a Money instance while preserving immutability.
110 111 112 113 114 115 116 117 118 119 120 |
# File 'lib/minting/money/constructors.rb', line 110 def copy_with(amount:) amount = currency.normalize_amount(amount) if amount == self.amount self elsif amount.zero? currency.zero else Money.new(amount, currency) end end |
#currency_code ⇒ String
Returns the ISO 3-letter currency code string.
31 |
# File 'lib/minting/money/money.rb', line 31 def currency_code = currency.code |
#eql?(other) ⇒ Boolean
Strict equality — both amount and currency must match exactly. Unlike ==, does not treat zero as equivalent across currencies.
21 22 23 24 25 |
# File 'lib/minting/money/comparable.rb', line 21 def eql?(other) other.is_a?(Mint::Money) && amount == other.amount && currency == other.currency end |
#fractional ⇒ Object
Returns the fractional part of the amount.
48 |
# File 'lib/minting/money/money.rb', line 48 def fractional = ((amount.abs % 1) * currency.fractional_multiplier).to_i |
#hash ⇒ Integer
Generates a stable hash key for Money instances.
53 |
# File 'lib/minting/money/money.rb', line 53 def hash = [amount, currency_code].hash |
#inspect ⇒ String
Returns a standard developer-oriented string inspection of the Money object.
58 59 60 |
# File 'lib/minting/money/money.rb', line 58 def inspect Kernel.format "[#{currency_code} %0.#{currency.subunit}f]", amount end |
#mint(new_amount) ⇒ Money
Use #copy_with instead. Will be removed in v2.
Returns a new Money with the given amount in the same currency.
129 130 131 132 |
# File 'lib/minting/money/constructors.rb', line 129 def mint(new_amount) warn 'Money#mint is now deprecated and will be removed in v2' copy_with(amount: new_amount) end |
#negative? ⇒ Boolean
Returns true if the monetary amount is less than zero.
14 |
# File 'lib/minting/money/arithmetics/methods.rb', line 14 def negative? = amount.negative? |
#nonzero? ⇒ self?
Returns self if amount is non-zero, nil otherwise.
45 |
# File 'lib/minting/money/comparable.rb', line 45 def nonzero? = amount.nonzero? |
#positive? ⇒ Boolean
Returns true if the monetary amount is greater than zero.
19 |
# File 'lib/minting/money/arithmetics/methods.rb', line 19 def positive? = amount.positive? |
#same_currency?(other) ⇒ Boolean
Helper method to verify if another Money has the identical currency.
51 |
# File 'lib/minting/money/comparable.rb', line 51 def same_currency?(other) = other.currency == currency |
#split(slices) ⇒ Array<Money>
Splits the monetary amount into a given quantity of equal parts. Disperses any fractional subunit rounding differences across the initial slots so that the sum is preserved.
17 18 19 20 21 22 23 |
# File 'lib/minting/money/allocation/split.rb', line 17 def split(slices) raise ArgumentError, 'Slices quantity must be an poitive integer' unless slices.positive? && slices.integer? fraction = currency.normalize_amount(amount / slices) allocate_left_over(amounts: Array.new(slices, fraction), left_over: amount - (fraction * slices)) end |
#subunits ⇒ Integer
Returns the monetary amount expressed in the currency’s smallest unit (fractional units). For example, cents for USD (subunit 2), yen for JPY (subunit 0), fils for IQD (subunit 3).
41 |
# File 'lib/minting/money/money.rb', line 41 def subunits = (amount * currency.fractional_multiplier).to_i |
#succ ⇒ Money
Returns the successor of the Money instance by adding the minimum possible subunit amount. Enables standard ranges and stepping (e.g. ‘1.dollar..10.dollars`).
25 |
# File 'lib/minting/money/arithmetics/methods.rb', line 25 def succ = copy_with(amount: amount + currency.minimum_amount) |
#to_d ⇒ BigDecimal
Converts the monetary amount to a BigDecimal object.
14 |
# File 'lib/minting/money/conversion.rb', line 14 def to_d = amount.to_d 0 |
#to_f ⇒ Float
Converts the monetary amount to a standard float. Note: Using float conversion loses precision guarantees.
20 |
# File 'lib/minting/money/conversion.rb', line 20 def to_f = amount.to_f |
#to_hash ⇒ Hash
Returns a Hash representation of the money instance.
47 48 49 |
# File 'lib/minting/money/conversion.rb', line 47 def to_hash { currency: currency_code, amount: Kernel.format("%0.#{currency.subunit}f", amount) } end |
#to_html(format = DEFAULT_FORMAT) ⇒ String
Renders a safe HTML5 ‘<data>` element containing the formatted currency. Embeds the ISO currency description and raw value as the metadata `title` attribute.
27 28 29 30 31 |
# File 'lib/minting/money/conversion.rb', line 27 def to_html(format = DEFAULT_FORMAT) title = Kernel.format("#{currency_code} %0.#{currency.subunit}f", amount) body = to_s(format: format) %(<data class='money' title='#{title}'>#{ERB::Util.html_escape(body)}</data>) end |
#to_i ⇒ Integer
Truncates and converts the monetary amount to an Integer.
39 |
# File 'lib/minting/money/conversion.rb', line 39 def to_i = amount.to_i |
#to_json(*_args) ⇒ String
Serializes the money instance to a standard JSON object containing the amount and currency. Highly optimized to run without external dependencies.
55 56 57 58 59 |
# File 'lib/minting/money/conversion.rb', line 55 def to_json(*_args) Kernel.format( %({"currency": "#{currency_code}", "amount": "%0.#{currency.subunit}f"}), amount ) end |
#to_r ⇒ Rational
Returns the exact internal Rational representation of the monetary amount.
64 |
# File 'lib/minting/money/conversion.rb', line 64 def to_r = amount |
#to_s(preset = nil, format: nil, decimal: nil, thousand: nil, width: nil) ⇒ String
Formats money as a string with customizable format, thousand delimiter, and decimal
79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 |
# File 'lib/minting/money/format/to_s.rb', line 79 def to_s(preset = nil, format: nil, decimal: nil, thousand: nil, width: nil) if preset config = PRESETS.fetch(preset) { raise ArgumentError, "Unknown format preset: #{preset.inspect}" } format ||= config[:format] decimal ||= config[:decimal] thousand ||= config[:thousand] width ||= config[:width] end validate_separators!(decimal:, thousand:) format, decimal, thousand = resolve_locale_for(format, decimal, thousand) case format when {}, '' then raise ArgumentError, 'format must not be empty' when Hash then validate_format_hash(format) when String then format = { positive: format } else raise ArgumentError, 'Invalid format. Only String or Hash are accepted' end formatted = format_amount(format, decimal: decimal, thousand: thousand) width ? formatted.rjust(width) : formatted end |
#zero? ⇒ Boolean
Returns true if amount is zero.
54 |
# File 'lib/minting/money/comparable.rb', line 54 def zero? = amount.zero? |