2021-04-09 17:01:00 -07:00
|
|
|
import { TOKEN_MINTS } from '@project-serum/serum'
|
2021-03-30 15:47:08 -07:00
|
|
|
import { PublicKey } from '@solana/web3.js'
|
|
|
|
import BN from 'bn.js'
|
|
|
|
|
|
|
|
export function isValidPublicKey(key) {
|
|
|
|
if (!key) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
new PublicKey(key)
|
|
|
|
return true
|
|
|
|
} catch {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function sleep(ms) {
|
|
|
|
return new Promise((resolve) => setTimeout(resolve, ms))
|
|
|
|
}
|
|
|
|
|
|
|
|
export const percentFormat = new Intl.NumberFormat(undefined, {
|
|
|
|
style: 'percent',
|
|
|
|
minimumFractionDigits: 2,
|
|
|
|
maximumFractionDigits: 2,
|
|
|
|
})
|
|
|
|
|
2021-04-29 14:27:13 -07:00
|
|
|
export function displayDepositsForMarginAccount(
|
|
|
|
marginAccount,
|
|
|
|
mangoGroup,
|
|
|
|
tokenIndex
|
|
|
|
) {
|
|
|
|
const deposit = marginAccount.getUiDeposit(mangoGroup, tokenIndex)
|
|
|
|
const decimals = mangoGroup.mintDecimals[tokenIndex]
|
|
|
|
return floorToDecimal(deposit, decimals)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function displayBorrowsForMarginAccount(
|
|
|
|
marginAccount,
|
|
|
|
mangoGroup,
|
|
|
|
tokenIndex
|
|
|
|
) {
|
|
|
|
const borrow = marginAccount.getUiBorrow(mangoGroup, tokenIndex)
|
|
|
|
const decimals = mangoGroup.mintDecimals[tokenIndex]
|
|
|
|
return ceilToDecimal(borrow, decimals)
|
|
|
|
}
|
|
|
|
|
2021-03-30 15:47:08 -07:00
|
|
|
export function floorToDecimal(
|
|
|
|
value: number,
|
|
|
|
decimals: number | undefined | null
|
|
|
|
) {
|
|
|
|
return decimals
|
|
|
|
? Math.floor(value * 10 ** decimals) / 10 ** decimals
|
|
|
|
: Math.floor(value)
|
|
|
|
}
|
|
|
|
|
2021-04-29 14:27:13 -07:00
|
|
|
export function ceilToDecimal(
|
|
|
|
value: number,
|
|
|
|
decimals: number | undefined | null
|
|
|
|
) {
|
|
|
|
return decimals
|
|
|
|
? Math.ceil(value * 10 ** decimals) / 10 ** decimals
|
|
|
|
: Math.ceil(value)
|
|
|
|
}
|
|
|
|
|
2021-03-30 15:47:08 -07:00
|
|
|
export function roundToDecimal(
|
|
|
|
value: number,
|
|
|
|
decimals: number | undefined | null
|
|
|
|
) {
|
|
|
|
return decimals ? Math.round(value * 10 ** decimals) / 10 ** decimals : value
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getDecimalCount(value): number {
|
|
|
|
if (
|
|
|
|
!isNaN(value) &&
|
|
|
|
Math.floor(value) !== value &&
|
|
|
|
value.toString().includes('.')
|
|
|
|
)
|
|
|
|
return value.toString().split('.')[1].length || 0
|
|
|
|
if (
|
|
|
|
!isNaN(value) &&
|
|
|
|
Math.floor(value) !== value &&
|
|
|
|
value.toString().includes('e')
|
|
|
|
)
|
|
|
|
return parseInt(value.toString().split('e-')[1] || '0')
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
export function divideBnToNumber(numerator: BN, denominator: BN): number {
|
|
|
|
const quotient = numerator.div(denominator).toNumber()
|
|
|
|
const rem = numerator.umod(denominator)
|
|
|
|
const gcd = rem.gcd(denominator)
|
|
|
|
return quotient + rem.div(gcd).toNumber() / denominator.div(gcd).toNumber()
|
|
|
|
}
|
|
|
|
|
|
|
|
export const formatBalanceDisplay = (balance, fixedDecimals) => {
|
2021-04-16 14:09:17 -07:00
|
|
|
// Get the decimal part
|
2021-03-30 15:47:08 -07:00
|
|
|
const dPart = balance - Math.trunc(balance)
|
|
|
|
return (
|
|
|
|
Math.trunc(balance) +
|
|
|
|
Math.floor(dPart * Math.pow(10, fixedDecimals)) /
|
|
|
|
Math.pow(10, fixedDecimals)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTokenMultiplierFromDecimals(decimals: number): BN {
|
|
|
|
return new BN(10).pow(new BN(decimals))
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:01:00 -07:00
|
|
|
export function abbreviateAddress(address: PublicKey, size = 5) {
|
2021-03-30 15:47:08 -07:00
|
|
|
const base58 = address.toBase58()
|
|
|
|
return base58.slice(0, size) + '…' + base58.slice(-size)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isEqual(obj1, obj2, keys) {
|
|
|
|
if (!keys && Object.keys(obj1).length !== Object.keys(obj2).length) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
keys = keys || Object.keys(obj1)
|
|
|
|
for (const k of keys) {
|
|
|
|
if (obj1[k] !== obj2[k]) {
|
|
|
|
// shallow comparison
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
export function groupBy(list, keyGetter) {
|
|
|
|
const map = new Map()
|
|
|
|
list.forEach((item) => {
|
|
|
|
const key = keyGetter(item)
|
|
|
|
const collection = map.get(key)
|
|
|
|
if (!collection) {
|
|
|
|
map.set(key, [item])
|
|
|
|
} else {
|
|
|
|
collection.push(item)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return map
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isDefined<T>(argument: T | undefined): argument is T {
|
|
|
|
return argument !== undefined
|
|
|
|
}
|
2021-04-02 11:26:21 -07:00
|
|
|
|
|
|
|
export const calculateMarketPrice = (
|
|
|
|
orderBook: Array<any>,
|
|
|
|
size: number,
|
|
|
|
side: string
|
|
|
|
) => {
|
|
|
|
let acc = 0
|
|
|
|
let selectedOrder
|
|
|
|
for (const order of orderBook) {
|
|
|
|
acc += order[1]
|
|
|
|
if (acc >= size) {
|
|
|
|
selectedOrder = order
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (side === 'buy') {
|
|
|
|
return selectedOrder[0] * 1.05
|
|
|
|
} else {
|
|
|
|
return selectedOrder[0] * 0.95
|
|
|
|
}
|
|
|
|
}
|
2021-04-09 07:27:49 -07:00
|
|
|
|
|
|
|
// Precision for our mango group token
|
|
|
|
export const tokenPrecision = {
|
|
|
|
BTC: 4,
|
|
|
|
ETH: 3,
|
2021-05-20 04:20:39 -07:00
|
|
|
SOL: 2,
|
|
|
|
SRM: 2,
|
2021-04-09 07:27:49 -07:00
|
|
|
USDC: 2,
|
|
|
|
USDT: 2,
|
|
|
|
WUSDT: 2,
|
|
|
|
}
|
2021-04-09 17:01:00 -07:00
|
|
|
|
User account page (#22)
* layout, overview, start on assets, borrows and open orders
* trade history, sortable data hook for tables, borrow page
* handle deposit and withdraw buttons
* borrow modal ui and integration + settle borrow for individual assets
* in orders balance to asset table and totals, responsive css, new connected wallet button + small tweaks
* account switch/creation flow
* accounts modal, update to usebalances hook
* handle settle, deposit before settle, save last account
* disable borrow/withdraw button when no account
2021-06-05 07:11:44 -07:00
|
|
|
// Precision for depositing/withdrawing
|
|
|
|
export const DECIMALS = {
|
|
|
|
BTC: 5,
|
|
|
|
ETH: 4,
|
|
|
|
SOL: 2,
|
|
|
|
SRM: 2,
|
|
|
|
USDC: 2,
|
|
|
|
USDT: 2,
|
|
|
|
WUSDT: 2,
|
|
|
|
}
|
|
|
|
|
2021-04-09 17:01:00 -07:00
|
|
|
export const getSymbolForTokenMintAddress = (address: string): string => {
|
2021-04-13 16:41:04 -07:00
|
|
|
if (address && address.length) {
|
2021-04-26 09:59:01 -07:00
|
|
|
return TOKEN_MINTS.find((m) => m.address.toString() === address)?.name || ''
|
2021-04-13 16:41:04 -07:00
|
|
|
} else {
|
|
|
|
return ''
|
|
|
|
}
|
2021-04-09 17:01:00 -07:00
|
|
|
}
|
2021-04-20 14:00:45 -07:00
|
|
|
|
|
|
|
export const capitalize = (s) => {
|
|
|
|
if (typeof s !== 'string') return ''
|
|
|
|
return s.charAt(0).toUpperCase() + s.slice(1)
|
|
|
|
}
|
2021-05-01 07:02:21 -07:00
|
|
|
|
|
|
|
export const copyToClipboard = (copyThis) => {
|
|
|
|
const el = document.createElement('textarea')
|
|
|
|
el.value = copyThis.toString()
|
|
|
|
document.body.appendChild(el)
|
|
|
|
el.select()
|
|
|
|
document.execCommand('copy')
|
|
|
|
document.body.removeChild(el)
|
|
|
|
}
|
User account page (#22)
* layout, overview, start on assets, borrows and open orders
* trade history, sortable data hook for tables, borrow page
* handle deposit and withdraw buttons
* borrow modal ui and integration + settle borrow for individual assets
* in orders balance to asset table and totals, responsive css, new connected wallet button + small tweaks
* account switch/creation flow
* accounts modal, update to usebalances hook
* handle settle, deposit before settle, save last account
* disable borrow/withdraw button when no account
2021-06-05 07:11:44 -07:00
|
|
|
|
|
|
|
// Truncate decimals without rounding
|
|
|
|
export const trimDecimals = (n, digits) => {
|
2021-06-05 09:14:34 -07:00
|
|
|
const step = Math.pow(10, digits || 0)
|
|
|
|
const temp = Math.trunc(step * n)
|
User account page (#22)
* layout, overview, start on assets, borrows and open orders
* trade history, sortable data hook for tables, borrow page
* handle deposit and withdraw buttons
* borrow modal ui and integration + settle borrow for individual assets
* in orders balance to asset table and totals, responsive css, new connected wallet button + small tweaks
* account switch/creation flow
* accounts modal, update to usebalances hook
* handle settle, deposit before settle, save last account
* disable borrow/withdraw button when no account
2021-06-05 07:11:44 -07:00
|
|
|
|
|
|
|
return temp / step
|
|
|
|
}
|