Class: Mint::Money
- Inherits:
-
Object
- Object
- Mint::Money
- Includes:
- Comparable
- Defined in:
- lib/minting/money/money.rb,
lib/minting/money/parse.rb,
lib/minting/money/coercion.rb,
lib/minting/money/allocation.rb,
lib/minting/money/comparable.rb,
lib/minting/money/conversion.rb,
lib/minting/money/formatting.rb,
lib/minting/money/arithmetics.rb,
lib/minting/money/constructors.rb
Overview
Money constructors
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'
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
Creates a new Money immutable object with the specified amount and currency.
-
.from_fractional(fractional, currency) ⇒ Money
Builds a Money from a fractional (smallest-unit) Integer amount.
-
.parse(input, currency = nil) ⇒ Money
Parses a human-readable money string into a Money object.
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)
Allows Money to interact seamlessly as the right-hand operand in Numeric arithmetic.
-
#currency_code ⇒ String
Returns the ISO 3-letter currency code string.
- #eql?(other) ⇒ Boolean
- #fractional ⇒ Object
-
#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
Returns a new Money object with the specified amount, or self if unchanged.
-
#negative? ⇒ Boolean
Returns true if the monetary amount is less than zero.
- #nonzero? ⇒ Boolean
-
#positive? ⇒ Boolean
Returns true if the monetary amount is greater than zero.
-
#same_currency?(other) ⇒ Boolean
Helper method to verify if another object has the identical currency.
-
#split(quantity) ⇒ Array<Money>
Splits the monetary amount into a given quantity of equal parts.
-
#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 ⇒ Object
-
#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(format: '%<symbol>s%<amount>f', decimal: '.', thousand: ',', width: nil) ⇒ String
Formats money as a string with customizable format, thousand delimiter, and decimal.
- #zero? ⇒ Boolean
Instance Attribute Details
#amount ⇒ Object (readonly)
Returns the value of attribute amount.
10 11 12 |
# File 'lib/minting/money/money.rb', line 10 def amount @amount end |
#currency ⇒ Object (readonly)
Returns the value of attribute currency.
10 11 12 |
# File 'lib/minting/money/money.rb', line 10 def currency @currency end |
Class Method Details
.create(amount, currency) ⇒ Object
Creates a new Money immutable object with the specified amount and currency
10 11 12 13 14 15 16 17 |
# File 'lib/minting/money/constructors.rb', line 10 def self.create(amount, currency) raise ArgumentError, 'amount must be Numeric' unless amount.is_a?(Numeric) checked_currency = Mint.currency(currency) raise ArgumentError, "Currency not found (#{currency})" unless checked_currency new(checked_currency.normalize_amount(amount), checked_currency) end |
.from_fractional(fractional, currency) ⇒ Money
Builds a Money from a fractional (smallest-unit) Integer amount. This is the inverse of #fractional: for USD, the fractional unit is 1 cent; for JPY it is 1 yen; for IQD it is 1 dinar (subunit 3).
37 38 39 40 41 42 43 44 45 |
# File 'lib/minting/money/constructors.rb', line 37 def self.from_fractional(fractional, currency) raise ArgumentError, 'fractional must be an Integer' unless fractional.is_a?(Integer) checked_currency = Mint.currency(currency) raise ArgumentError, "Currency not found (#{currency})" unless checked_currency amount = Rational(fractional, checked_currency.fractional_multiplier) new(amount, checked_currency) end |
.parse(input, currency = nil) ⇒ Money
Parses a human-readable money string into a Mint::Money object.
21 22 23 24 25 26 27 28 29 30 31 32 |
# File 'lib/minting/money/parse.rb', line 21 def self.parse(input, currency = nil) raise ArgumentError, 'input must be a String' unless input.is_a?(String) input = input.strip raise ArgumentError, 'input cannot be empty' if input.empty? currency = parse_currency(currency) || parse_currency(input) raise ArgumentError, "Currency [#{currency}] not registered" unless currency amount = currency.normalize_amount(parse_amount(input)) new(amount, currency) end |
Instance Method Details
#*(multiplicand) ⇒ Money
Performs multiplication of the monetary value by a standard scalar Numeric.
65 66 67 68 69 |
# File 'lib/minting/money/arithmetics.rb', line 65 def *(multiplicand) return mint(amount * multiplicand) if multiplicand.is_a?(Numeric) raise TypeError, "#{self} can't be multiplied by #{multiplicand}" end |
#**(exponent) ⇒ Money
Performs exponentiation of the monetary value by a standard scalar Numeric.
90 91 92 93 94 |
# File 'lib/minting/money/arithmetics.rb', line 90 def **(exponent) return mint(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.
32 33 34 35 36 37 38 |
# File 'lib/minting/money/arithmetics.rb', line 32 def +(addend) case addend when 0 then return self when Money then return mint(amount + addend.amount) if same_currency?(addend) end raise TypeError, "#{addend} can't be added to #{self}" end |
#-(subtrahend) ⇒ Money
Performs subtraction with another Mint::Money instance or standard zero Numeric.
45 46 47 48 49 50 51 |
# File 'lib/minting/money/arithmetics.rb', line 45 def -(subtrahend) case subtrahend when 0 then return self when Money then return mint(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.
56 57 58 |
# File 'lib/minting/money/arithmetics.rb', line 56 def -@ mint(-amount) end |
#/(divisor) ⇒ Money, Numeric
Performs division of the monetary value by a scalar Numeric or identical currency Mint::Money.
77 78 79 80 81 82 83 |
# File 'lib/minting/money/arithmetics.rb', line 77 def /(divisor) case divisor when Numeric then return mint(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
30 31 32 33 34 35 36 |
# File 'lib/minting/money/comparable.rb', line 30 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 |
# File 'lib/minting/money/comparable.rb', line 9 def ==(other) return true if zero? && other.respond_to?(:zero?) && other.zero? eql?(other) end |
#abs ⇒ Money
Returns the absolute value of the monetary amount as a new Mint::Money instance.
9 |
# File 'lib/minting/money/arithmetics.rb', line 9 def abs = mint(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 23 |
# File 'lib/minting/money/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? subunit = currency.subunit amounts = proportions.map { |rate| (amount * rate.to_r / whole).round(subunit) } 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.
73 74 75 76 77 78 79 80 81 82 |
# File 'lib/minting/money/money.rb', line 73 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 mint(amount.clamp(normalize_boundary(min), normalize_boundary(max))) end |
#coerce(other) ⇒ Array(CoercedNumber, Money)
Allows Mint::Money to interact seamlessly as the right-hand operand in Numeric arithmetic.
10 11 12 |
# File 'lib/minting/money/coercion.rb', line 10 def coerce(other) [CoercedNumber.new(other), self] end |
#currency_code ⇒ String
Returns the ISO 3-letter currency code string.
15 |
# File 'lib/minting/money/money.rb', line 15 def currency_code = currency.code |
#eql?(other) ⇒ Boolean
15 16 17 18 19 |
# File 'lib/minting/money/comparable.rb', line 15 def eql?(other) other.is_a?(Mint::Money) && amount == other.amount && currency == other.currency end |
#fractional ⇒ Object
17 |
# File 'lib/minting/money/money.rb', line 17 def fractional = (amount * currency.fractional_multiplier).to_i |
#hash ⇒ Integer
Generates a stable hash key for Money instances.
22 |
# File 'lib/minting/money/money.rb', line 22 def hash = [amount, currency_code].hash |
#inspect ⇒ String
Returns a standard developer-oriented string inspection of the Money object.
27 28 29 |
# File 'lib/minting/money/money.rb', line 27 def inspect Kernel.format "[#{currency_code} %0.#{currency.subunit}f]", amount end |
#mint(new_amount) ⇒ Money
Returns a new Money object with the specified amount, or self if unchanged
50 51 52 53 |
# File 'lib/minting/money/constructors.rb', line 50 def mint(new_amount) new_amount = currency.normalize_amount(new_amount) new_amount == amount ? self : Money.new(new_amount, currency) end |
#negative? ⇒ Boolean
Returns true if the monetary amount is less than zero.
14 |
# File 'lib/minting/money/arithmetics.rb', line 14 def negative? = amount.negative? |
#nonzero? ⇒ Boolean
38 |
# File 'lib/minting/money/comparable.rb', line 38 def nonzero? = amount.nonzero? |
#positive? ⇒ Boolean
Returns true if the monetary amount is greater than zero.
19 |
# File 'lib/minting/money/arithmetics.rb', line 19 def positive? = amount.positive? |
#same_currency?(other) ⇒ Boolean
Helper method to verify if another object has the identical currency.
35 |
# File 'lib/minting/money/money.rb', line 35 def same_currency?(other) = other.currency == currency |
#split(quantity) ⇒ 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.
36 37 38 39 40 41 42 43 44 45 |
# File 'lib/minting/money/allocation.rb', line 36 def split(quantity) unless quantity.positive? && quantity.integer? raise ArgumentError, 'quantity must be an integer > 0' end fraction = (amount / quantity).round(currency.subunit) allocate_left_over!(amounts: Array.new(quantity, fraction), left_over: amount - (fraction * quantity)) end |
#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.rb', line 25 def succ = mint(amount + currency.minimum_amount) |
#to_d ⇒ BigDecimal
Converts the monetary amount to a BigDecimal object.
12 |
# File 'lib/minting/money/conversion.rb', line 12 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.
18 |
# File 'lib/minting/money/conversion.rb', line 18 def to_f = amount.to_f |
#to_hash ⇒ Object
36 37 38 |
# File 'lib/minting/money/conversion.rb', line 36 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.
25 26 27 28 29 |
# File 'lib/minting/money/conversion.rb', line 25 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.
34 |
# File 'lib/minting/money/conversion.rb', line 34 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.
44 45 46 47 48 |
# File 'lib/minting/money/conversion.rb', line 44 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.
53 |
# File 'lib/minting/money/conversion.rb', line 53 def to_r = amount |
#to_s(format: '%<symbol>s%<amount>f', decimal: '.', thousand: ',', width: nil) ⇒ String
Formats money as a string with customizable format, thousand delimiter, and decimal
46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 |
# File 'lib/minting/money/formatting.rb', line 46 def to_s(format: '%<symbol>s%<amount>f', decimal: '.', thousand: ',', width: nil) case format when {}, '', nil then raise ArgumentError, 'format must not be empty or null' when Hash then validate_format_hash!(format) when String # noop else raise ArgumentError, 'Invalid format' end formatted = format_amount(format) formatted.tr!('.', decimal) if decimal != '.' unless thousand.empty? # Regular expression courtesy of Money gem # Matches digits followed by groups of 3 digits until non-digit or end formatted.gsub!(/(\d)(?=(?:\d{3})+(?:[^\d]{1}|$))/, "\\1#{thousand}") end formatted = formatted.rjust(width) if width formatted end |
#zero? ⇒ Boolean
40 |
# File 'lib/minting/money/comparable.rb', line 40 def zero? = amount.zero? |