2020-06-29 04:44:35 +02:00
|
|
|
using System;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.Collections.Generic;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Globalization;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.IO;
|
|
|
|
using System.Linq;
|
2020-06-28 10:55:27 +02:00
|
|
|
using System.Reflection;
|
2017-09-13 08:47:34 +02:00
|
|
|
using System.Text;
|
2020-05-31 12:18:29 +02:00
|
|
|
using BTCPayServer.Rating;
|
2020-06-28 10:55:27 +02:00
|
|
|
using NBitcoin;
|
2020-05-31 12:18:29 +02:00
|
|
|
using Newtonsoft.Json;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2017-09-15 09:06:57 +02:00
|
|
|
namespace BTCPayServer.Services.Rates
|
2017-09-13 08:47:34 +02:00
|
|
|
{
|
2017-10-27 10:53:04 +02:00
|
|
|
public class CurrencyData
|
2017-09-13 08:47:34 +02:00
|
|
|
{
|
2020-05-31 12:18:29 +02:00
|
|
|
public string Name { get; set; }
|
|
|
|
public string Code { get; set; }
|
|
|
|
public int Divisibility { get; set; }
|
|
|
|
public string Symbol { get; set; }
|
2018-05-16 14:19:48 +02:00
|
|
|
public bool Crypto { get; set; }
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
|
|
|
public class CurrencyNameTable
|
|
|
|
{
|
2020-05-31 12:18:29 +02:00
|
|
|
public static CurrencyNameTable Instance = new CurrencyNameTable();
|
2017-10-27 10:53:04 +02:00
|
|
|
public CurrencyNameTable()
|
|
|
|
{
|
|
|
|
_Currencies = LoadCurrency().ToDictionary(k => k.Code);
|
|
|
|
}
|
|
|
|
|
2020-06-29 05:07:48 +02:00
|
|
|
static readonly Dictionary<string, IFormatProvider> _CurrencyProviders = new Dictionary<string, IFormatProvider>();
|
2019-02-17 08:53:41 +01:00
|
|
|
public string FormatCurrency(string price, string currency)
|
|
|
|
{
|
|
|
|
return FormatCurrency(decimal.Parse(price, CultureInfo.InvariantCulture), currency);
|
|
|
|
}
|
|
|
|
public string FormatCurrency(decimal price, string currency)
|
|
|
|
{
|
|
|
|
return price.ToString("C", GetCurrencyProvider(currency));
|
|
|
|
}
|
|
|
|
|
2018-05-20 16:37:18 +02:00
|
|
|
public NumberFormatInfo GetNumberFormatInfo(string currency, bool useFallback)
|
2018-05-15 19:24:59 +02:00
|
|
|
{
|
|
|
|
var data = GetCurrencyProvider(currency);
|
|
|
|
if (data is NumberFormatInfo nfi)
|
|
|
|
return nfi;
|
2018-05-20 16:22:20 +02:00
|
|
|
if (data is CultureInfo ci)
|
|
|
|
return ci.NumberFormat;
|
2018-05-20 16:37:18 +02:00
|
|
|
if (!useFallback)
|
|
|
|
return null;
|
2018-05-20 16:22:20 +02:00
|
|
|
return CreateFallbackCurrencyFormatInfo(currency);
|
2018-05-15 19:24:59 +02:00
|
|
|
}
|
2018-05-20 16:22:20 +02:00
|
|
|
|
|
|
|
private NumberFormatInfo CreateFallbackCurrencyFormatInfo(string currency)
|
|
|
|
{
|
2018-05-20 16:37:18 +02:00
|
|
|
var usd = GetNumberFormatInfo("USD", false);
|
2018-05-20 16:22:20 +02:00
|
|
|
var currencyInfo = (NumberFormatInfo)usd.Clone();
|
|
|
|
currencyInfo.CurrencySymbol = currency;
|
|
|
|
return currencyInfo;
|
|
|
|
}
|
2018-12-04 05:04:26 +01:00
|
|
|
public NumberFormatInfo GetNumberFormatInfo(string currency)
|
|
|
|
{
|
|
|
|
var curr = GetCurrencyProvider(currency);
|
|
|
|
if (curr is CultureInfo cu)
|
|
|
|
return cu.NumberFormat;
|
|
|
|
if (curr is NumberFormatInfo ni)
|
|
|
|
return ni;
|
|
|
|
return null;
|
|
|
|
}
|
2017-10-27 11:58:43 +02:00
|
|
|
public IFormatProvider GetCurrencyProvider(string currency)
|
|
|
|
{
|
|
|
|
lock (_CurrencyProviders)
|
|
|
|
{
|
|
|
|
if (_CurrencyProviders.Count == 0)
|
|
|
|
{
|
|
|
|
foreach (var culture in CultureInfo.GetCultures(CultureTypes.AllCultures).Where(c => !c.IsNeutralCulture))
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
_CurrencyProviders.TryAdd(new RegionInfo(culture.LCID).ISOCurrencySymbol, culture);
|
|
|
|
}
|
|
|
|
catch { }
|
|
|
|
}
|
2018-05-15 19:24:59 +02:00
|
|
|
|
2020-05-31 12:18:29 +02:00
|
|
|
foreach (var curr in _Currencies.Where(pair => pair.Value.Crypto))
|
2018-05-15 19:24:59 +02:00
|
|
|
{
|
2020-06-28 10:55:27 +02:00
|
|
|
AddCurrency(_CurrencyProviders, curr.Key, curr.Value.Divisibility, curr.Value.Symbol ?? curr.Value.Code);
|
2018-05-15 19:24:59 +02:00
|
|
|
}
|
2017-10-27 11:58:43 +02:00
|
|
|
}
|
2019-06-09 17:46:29 +02:00
|
|
|
return _CurrencyProviders.TryGet(currency.ToUpperInvariant());
|
2017-10-27 11:58:43 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void AddCurrency(Dictionary<string, IFormatProvider> currencyProviders, string code, int divisibility, string symbol)
|
|
|
|
{
|
|
|
|
var culture = new CultureInfo("en-US");
|
|
|
|
var number = new NumberFormatInfo();
|
|
|
|
number.CurrencyDecimalDigits = divisibility;
|
|
|
|
number.CurrencySymbol = symbol;
|
|
|
|
number.CurrencyDecimalSeparator = culture.NumberFormat.CurrencyDecimalSeparator;
|
|
|
|
number.CurrencyGroupSeparator = culture.NumberFormat.CurrencyGroupSeparator;
|
|
|
|
number.CurrencyGroupSizes = culture.NumberFormat.CurrencyGroupSizes;
|
|
|
|
number.CurrencyNegativePattern = 8;
|
|
|
|
number.CurrencyPositivePattern = 3;
|
|
|
|
number.NegativeSign = culture.NumberFormat.NegativeSign;
|
|
|
|
currencyProviders.TryAdd(code, number);
|
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2018-10-09 16:30:06 +02:00
|
|
|
/// <summary>
|
|
|
|
/// Format a currency like "0.004 $ (USD)", round to significant divisibility
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="value">The value</param>
|
|
|
|
/// <param name="currency">Currency code</param>
|
|
|
|
/// <returns></returns>
|
2021-08-03 10:03:00 +02:00
|
|
|
public string DisplayFormatCurrency(decimal value, string currency)
|
2018-10-09 16:30:06 +02:00
|
|
|
{
|
|
|
|
var provider = GetNumberFormatInfo(currency, true);
|
|
|
|
var currencyData = GetCurrencyData(currency, true);
|
|
|
|
var divisibility = currencyData.Divisibility;
|
2019-02-22 14:15:25 +01:00
|
|
|
value = value.RoundToSignificant(ref divisibility);
|
2018-10-09 16:30:06 +02:00
|
|
|
if (divisibility != provider.CurrencyDecimalDigits)
|
|
|
|
{
|
|
|
|
provider = (NumberFormatInfo)provider.Clone();
|
|
|
|
provider.CurrencyDecimalDigits = divisibility;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currencyData.Crypto)
|
2018-12-18 16:28:06 +01:00
|
|
|
return value.ToString("C", provider);
|
2018-10-09 16:30:06 +02:00
|
|
|
else
|
2018-12-18 16:28:06 +01:00
|
|
|
return value.ToString("C", provider) + $" ({currency})";
|
2018-10-09 16:30:06 +02:00
|
|
|
}
|
|
|
|
|
2020-06-29 05:07:48 +02:00
|
|
|
readonly Dictionary<string, CurrencyData> _Currencies;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
static CurrencyData[] LoadCurrency()
|
|
|
|
{
|
2020-05-31 12:18:29 +02:00
|
|
|
var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream("BTCPayServer.Rating.Currencies.json");
|
2017-10-27 10:53:04 +02:00
|
|
|
string content = null;
|
|
|
|
using (var reader = new StreamReader(stream, Encoding.UTF8))
|
|
|
|
{
|
|
|
|
content = reader.ReadToEnd();
|
|
|
|
}
|
2019-09-18 17:26:24 +02:00
|
|
|
|
2020-05-31 12:18:29 +02:00
|
|
|
var currencies = JsonConvert.DeserializeObject<CurrencyData[]>(content);
|
|
|
|
return currencies;
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2018-05-20 16:37:18 +02:00
|
|
|
public CurrencyData GetCurrencyData(string currency, bool useFallback)
|
2017-10-27 10:53:04 +02:00
|
|
|
{
|
2020-06-24 03:34:09 +02:00
|
|
|
if (currency == null)
|
|
|
|
throw new ArgumentNullException(nameof(currency));
|
2017-10-27 10:53:04 +02:00
|
|
|
CurrencyData result;
|
2018-10-09 16:30:06 +02:00
|
|
|
if (!_Currencies.TryGetValue(currency.ToUpperInvariant(), out result))
|
2018-05-20 16:37:18 +02:00
|
|
|
{
|
2018-10-09 16:30:06 +02:00
|
|
|
if (useFallback)
|
2018-05-20 16:37:18 +02:00
|
|
|
{
|
|
|
|
var usd = GetCurrencyData("USD", false);
|
|
|
|
result = new CurrencyData()
|
|
|
|
{
|
|
|
|
Code = currency,
|
|
|
|
Crypto = true,
|
|
|
|
Name = currency,
|
|
|
|
Divisibility = usd.Divisibility
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2017-10-27 10:53:04 +02:00
|
|
|
return result;
|
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
}
|
2017-09-13 08:47:34 +02:00
|
|
|
}
|