2017-09-13 08:47:34 +02:00
|
|
|
|
using System;
|
2017-10-27 11:58:43 +02:00
|
|
|
|
using NBitcoin;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Reflection;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
2017-10-27 11:58:43 +02:00
|
|
|
|
using System.Globalization;
|
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
|
|
|
|
{
|
2017-10-27 10:53:04 +02:00
|
|
|
|
public string Name
|
|
|
|
|
{
|
|
|
|
|
get;
|
|
|
|
|
internal set;
|
|
|
|
|
}
|
|
|
|
|
public string Code
|
|
|
|
|
{
|
|
|
|
|
get;
|
|
|
|
|
internal set;
|
|
|
|
|
}
|
|
|
|
|
public int Divisibility
|
|
|
|
|
{
|
|
|
|
|
get;
|
|
|
|
|
internal set;
|
|
|
|
|
}
|
|
|
|
|
public string Symbol
|
|
|
|
|
{
|
|
|
|
|
get;
|
|
|
|
|
internal 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
|
|
|
|
|
{
|
|
|
|
|
public CurrencyNameTable()
|
|
|
|
|
{
|
|
|
|
|
_Currencies = LoadCurrency().ToDictionary(k => k.Code);
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-27 11:58:43 +02:00
|
|
|
|
static Dictionary<string, IFormatProvider> _CurrencyProviders = new Dictionary<string, IFormatProvider>();
|
2018-05-15 19:24:59 +02:00
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
foreach (var network in new BTCPayNetworkProvider(NetworkType.Mainnet).GetAll())
|
|
|
|
|
{
|
|
|
|
|
AddCurrency(_CurrencyProviders, network.CryptoCode, 8, network.CryptoCode);
|
|
|
|
|
}
|
2017-10-27 11:58:43 +02:00
|
|
|
|
}
|
|
|
|
|
return _CurrencyProviders.TryGet(currency);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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>
|
|
|
|
|
/// <param name="threeLetterSuffix">Add three letter suffix (like USD)</param>
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
public string DisplayFormatCurrency(decimal value, string currency, bool threeLetterSuffix = true)
|
|
|
|
|
{
|
|
|
|
|
var provider = GetNumberFormatInfo(currency, true);
|
|
|
|
|
var currencyData = GetCurrencyData(currency, true);
|
|
|
|
|
var divisibility = currencyData.Divisibility;
|
|
|
|
|
while (true)
|
|
|
|
|
{
|
|
|
|
|
var rounded = decimal.Round(value, divisibility, MidpointRounding.AwayFromZero);
|
|
|
|
|
if ((Math.Abs(rounded - value) / value) < 0.001m)
|
|
|
|
|
{
|
|
|
|
|
value = rounded;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
divisibility++;
|
|
|
|
|
}
|
|
|
|
|
if (divisibility != provider.CurrencyDecimalDigits)
|
|
|
|
|
{
|
|
|
|
|
provider = (NumberFormatInfo)provider.Clone();
|
|
|
|
|
provider.CurrencyDecimalDigits = divisibility;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (currencyData.Crypto)
|
|
|
|
|
return value.ToString("C", provider);
|
|
|
|
|
else
|
|
|
|
|
return value.ToString("C", provider) + $" ({currency})";
|
|
|
|
|
}
|
|
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
|
Dictionary<string, CurrencyData> _Currencies;
|
2017-09-13 08:47:34 +02:00
|
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
|
static CurrencyData[] LoadCurrency()
|
|
|
|
|
{
|
|
|
|
|
var stream = Assembly.GetExecutingAssembly().GetManifestResourceStream("BTCPayServer.Currencies.txt");
|
|
|
|
|
string content = null;
|
|
|
|
|
using (var reader = new StreamReader(stream, Encoding.UTF8))
|
|
|
|
|
{
|
|
|
|
|
content = reader.ReadToEnd();
|
|
|
|
|
}
|
|
|
|
|
var currencies = content.Split(new[] { "\r\n", "\n" }, StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
|
Dictionary<string, CurrencyData> dico = new Dictionary<string, CurrencyData>();
|
|
|
|
|
foreach (var currency in currencies)
|
|
|
|
|
{
|
|
|
|
|
var splitted = currency.Split(new[] { '\t' }, StringSplitOptions.RemoveEmptyEntries);
|
|
|
|
|
if (splitted.Length < 3)
|
|
|
|
|
continue;
|
|
|
|
|
CurrencyData info = new CurrencyData();
|
|
|
|
|
info.Name = splitted[0];
|
|
|
|
|
info.Code = splitted[1];
|
|
|
|
|
int divisibility;
|
|
|
|
|
if (!int.TryParse(splitted[2], out divisibility))
|
|
|
|
|
continue;
|
|
|
|
|
info.Divisibility = divisibility;
|
|
|
|
|
if (!dico.ContainsKey(info.Code))
|
|
|
|
|
dico.Add(info.Code, info);
|
|
|
|
|
if (splitted.Length >= 4)
|
|
|
|
|
{
|
|
|
|
|
info.Symbol = splitted[3];
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-05-15 19:24:59 +02:00
|
|
|
|
|
|
|
|
|
foreach (var network in new BTCPayNetworkProvider(NetworkType.Mainnet).GetAll())
|
|
|
|
|
{
|
2018-10-09 16:30:06 +02:00
|
|
|
|
if (!dico.TryAdd(network.CryptoCode, new CurrencyData()
|
2018-05-15 19:24:59 +02:00
|
|
|
|
{
|
|
|
|
|
Code = network.CryptoCode,
|
|
|
|
|
Divisibility = 8,
|
2018-05-16 14:19:48 +02:00
|
|
|
|
Name = network.CryptoCode,
|
|
|
|
|
Crypto = true
|
2018-10-09 16:30:06 +02:00
|
|
|
|
}))
|
|
|
|
|
{
|
|
|
|
|
dico[network.CryptoCode].Crypto = true;
|
|
|
|
|
}
|
2018-05-15 19:24:59 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-10-27 10:53:04 +02:00
|
|
|
|
return dico.Values.ToArray();
|
|
|
|
|
}
|
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
|
|
|
|
{
|
|
|
|
|
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
|
|
|
|
}
|