ss58_registry/token.rs
1// Copyright (C) 2021-2022 Parity Technologies (UK) Ltd.
2// SPDX-License-Identifier: Apache-2.0
3
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15
16#[cfg(feature = "std")]
17use num_format::{Locale, ToFormattedString};
18
19/// Name and decimals of a given token.
20#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
21pub struct Token {
22 /// The short name (ticker) of the token
23 pub name: &'static str,
24 /// The number of decimals the token has (smallest granularity of the token)
25 pub decimals: u8,
26}
27
28impl core::fmt::Debug for Token {
29 fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
30 f.debug_struct("Token")
31 .field("name", &self.name)
32 .field("decimals", &self.decimals)
33 .finish()
34 }
35}
36
37impl Token {
38 /// Creates the specified amount of [`Token`] with its name and decimals filled from the
39 /// [`TokenRegistry`] variant.
40 ///
41 /// ```
42 /// # use ss58_registry::{Token, TokenRegistry};
43 /// # #[cfg(feature = "std")]
44 /// # fn x() {
45 /// let token: Token = TokenRegistry::Dot.into();
46 /// let my_amount = token.amount(100_000_000);
47 /// assert_eq!(format!("{}", my_amount), "0.010 DOT");
48 /// assert_eq!(format!("{:?}", my_amount), "0.010 DOT (100,000,000)");
49 /// # }
50 /// # #[cfg(not(feature = "std"))]
51 /// # fn x() {}
52 /// # x();
53 /// ```
54 pub fn amount(&self, amount: u128) -> TokenAmount {
55 TokenAmount { token: self.clone(), amount }
56 }
57}
58
59/// A given amount of token. Can be used for nicely formatted output and token-aware comparison of
60/// different amounts.
61///
62/// ```
63/// # use ss58_registry::{Token, TokenAmount};
64/// # #[cfg(feature = "std")]
65/// # fn x() {
66/// let token = Token { name: "I❤U", decimals: 8 };
67/// let my_amount = token.amount(100_000_000_000);
68/// assert_eq!(format!("{}", my_amount), "1,000.000 I❤U");
69/// assert_eq!(format!("{:?}", my_amount), "1000.000 I❤U (100,000,000,000)");
70/// # }
71/// # #[cfg(not(feature = "std"))]
72/// # fn x() {}
73/// # x();
74/// ```
75#[derive(Clone, PartialEq, Eq, PartialOrd, Ord, Hash)]
76pub struct TokenAmount {
77 /// The token this amount is from.
78 pub token: Token,
79 /// The amount in the smallest granularity of the token.
80 pub amount: u128,
81}
82
83#[cfg(feature = "std")]
84impl std::fmt::Display for TokenAmount {
85 fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
86 let multiplier = u128::pow(10, self.token.decimals as u32);
87 write!(
88 f,
89 "{}.{:0>3} {}",
90 (self.amount / multiplier).to_formatted_string(&Locale::en),
91 self.amount % multiplier / (multiplier / 1000),
92 self.token.name
93 )
94 }
95}
96
97#[cfg(feature = "std")]
98impl std::fmt::Debug for TokenAmount {
99 fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
100 let multiplier = u128::pow(10, self.token.decimals as u32);
101 write!(
102 f,
103 "{}.{:0>3} {} ({})",
104 self.amount / multiplier,
105 self.amount % multiplier / (multiplier / 1000),
106 self.token.name,
107 self.amount.to_formatted_string(&Locale::en),
108 )
109 }
110}