From 35ff6054157734047eda76a280cae607c5cf39f6 Mon Sep 17 00:00:00 2001
From: mistakia <1823355+mistakia@users.noreply.github.com>
Date: Tue, 30 Apr 2024 03:06:05 -0400
Subject: [PATCH] feat: add charts for address counts within each balance range
---
src/core/constants.js | 32 ++++
.../index.js | 22 +++
.../ledger-chart-addresses-with-balance.js | 174 ++++++++++++++++++
.../ledger-chart-amounts.js | 67 +++----
.../ledger-chart-distribution.js | 61 +++---
src/views/pages/ledger/index.js | 19 +-
src/views/pages/ledger/ledger.js | 129 ++++++++-----
src/views/pages/ledger/ledger.styl | 14 +-
8 files changed, 379 insertions(+), 139 deletions(-)
create mode 100644 src/views/components/ledger-chart-addresses-with-balance/index.js
create mode 100644 src/views/components/ledger-chart-addresses-with-balance/ledger-chart-addresses-with-balance.js
diff --git a/src/core/constants.js b/src/core/constants.js
index c8b1dfb2..f02400d2 100644
--- a/src/core/constants.js
+++ b/src/core/constants.js
@@ -12,3 +12,35 @@ export const WS_URL = IS_DEV ? 'ws://localhost:8080' : 'wss://nano.community'
// 3 Million Nano (3e36)
export const REP_MAX_WEIGHT = BigNumber(3).shiftedBy(36)
+
+export const base_ranges = [
+ '_1000000',
+ '_100000',
+ '_10000',
+ '_1000',
+ '_100',
+ '_10',
+ '_1',
+ '_01',
+ '_001',
+ '_0001',
+ '_00001',
+ '_000001',
+ '_000001_below'
+]
+
+export const base_range_labels = {
+ _1000000: '>1M',
+ _100000: '100k to 1M',
+ _10000: '10k to 100k',
+ _1000: '1k to 10k',
+ _100: '100 to 1k',
+ _10: '10 to 100',
+ _1: '1 to 10',
+ _01: '0.1 to 1',
+ _001: '0.01 to 0.1',
+ _0001: '0.001 to 0.01',
+ _00001: '0.0001 to 0.001',
+ _000001: '0.00001 to 0.0001',
+ _000001_below: '<0.00001'
+}
diff --git a/src/views/components/ledger-chart-addresses-with-balance/index.js b/src/views/components/ledger-chart-addresses-with-balance/index.js
new file mode 100644
index 00000000..a4d47e62
--- /dev/null
+++ b/src/views/components/ledger-chart-addresses-with-balance/index.js
@@ -0,0 +1,22 @@
+import { connect } from 'react-redux'
+import { createSelector } from 'reselect'
+
+import { nanodb_actions } from '@core/nanodb/actions'
+
+import LedgerChartAddressesWithBalance from './ledger-chart-addresses-with-balance'
+
+const map_state_to_props = createSelector(
+ (state) => state.getIn(['nanodb', 'price_history']),
+ (price_history) => ({
+ price_history
+ })
+)
+
+const map_dispatch_to_props = {
+ get_price_history: nanodb_actions.get_price_history
+}
+
+export default connect(
+ map_state_to_props,
+ map_dispatch_to_props
+)(LedgerChartAddressesWithBalance)
diff --git a/src/views/components/ledger-chart-addresses-with-balance/ledger-chart-addresses-with-balance.js b/src/views/components/ledger-chart-addresses-with-balance/ledger-chart-addresses-with-balance.js
new file mode 100644
index 00000000..60c55040
--- /dev/null
+++ b/src/views/components/ledger-chart-addresses-with-balance/ledger-chart-addresses-with-balance.js
@@ -0,0 +1,174 @@
+import React, { useEffect, useMemo } from 'react'
+import PropTypes from 'prop-types'
+import Button from '@mui/material/Button'
+import ReactEChartsCore from 'echarts-for-react/lib/core'
+import * as echarts from 'echarts/core'
+import { LineChart } from 'echarts/charts'
+import { TooltipComponent, GridComponent } from 'echarts/components'
+import { CanvasRenderer } from 'echarts/renderers'
+
+import { download_csv, download_json } from '@core/utils'
+import { base_range_labels } from '@core/constants'
+
+echarts.use([LineChart, TooltipComponent, GridComponent, CanvasRenderer])
+
+export default function LedgerChartAddressesWithBalance({
+ data,
+ isLoading,
+ price_history,
+ get_price_history,
+ range
+}) {
+ useEffect(() => {
+ if (price_history.length === 0) {
+ get_price_history()
+ }
+ }, [])
+
+ const handle_download_csv = () => {
+ const download_data = []
+ data[`${range}_account_count`].forEach((item) => {
+ download_data.push({
+ range: base_range_labels[range],
+ date: item[0],
+ account_count: item[1]
+ })
+ })
+ download_csv({
+ headers: ['Range', 'Date', 'Account Count'],
+ data: download_data,
+ file_name: `nano-community-ledger-daily-addresses-with-balance-${base_range_labels[
+ range
+ ].toLowerCase()}`
+ })
+ }
+
+ const handle_download_json = () => {
+ const download_data = {
+ title: `Nano Community Ledger Daily Addresses With Balance - ${base_range_labels[range]}`,
+ data: data[`${range}_account_count`].map((item) => ({
+ date: item[0],
+ account_count: item[1]
+ }))
+ }
+ download_json({
+ data: download_data,
+ file_name: `nano-community-ledger-daily-addresses-with-balance-${base_range_labels[
+ range
+ ].toLowerCase()}`
+ })
+ }
+
+ const series_data_usd_price = useMemo(() => {
+ const data = []
+ price_history.forEach((item) => {
+ data.push([item.timestamp_utc, item.price])
+ })
+ return data
+ }, [price_history])
+
+ const series_data = useMemo(() => {
+ return {
+ name: base_range_labels[range],
+ type: 'line',
+ lineStyle: {
+ width: 2
+ },
+ showSymbol: false,
+ data: data[`${range}_account_count`].map((item) => [item[0], item[1]])
+ }
+ }, [data, range])
+
+ const option = {
+ tooltip: {
+ trigger: 'axis',
+ axisPointer: {
+ type: 'cross',
+ label: {
+ backgroundColor: '#6a7985'
+ }
+ }
+ },
+ legend: {
+ bottom: 0
+ },
+ xAxis: [
+ {
+ type: 'time',
+ boundaryGap: false
+ }
+ ],
+ yAxis: [
+ {
+ type: 'value'
+ },
+ {
+ type: 'log',
+ position: 'right',
+ axisLabel: {
+ formatter: '{value} USD'
+ }
+ }
+ ],
+ series: [
+ series_data,
+ {
+ name: 'USD Price',
+ type: 'line',
+ yAxisIndex: 1,
+ data: series_data_usd_price,
+ showSymbol: false,
+ lineStyle: {
+ width: 2,
+ color: '#333333'
+ }
+ }
+ ]
+ }
+
+ return (
+ <>
+
+
+
+ Description
+
+
+
+ The number of unique addresses holding {base_range_labels[range]}{' '}
+ Nano.
+
+
+ {!isLoading && (
+
+
+
+
+ )}
+
+ >
+ )
+}
+
+LedgerChartAddressesWithBalance.propTypes = {
+ data: PropTypes.object.isRequired,
+ isLoading: PropTypes.bool.isRequired,
+ price_history: PropTypes.array.isRequired,
+ get_price_history: PropTypes.func.isRequired,
+ range: PropTypes.string
+}
diff --git a/src/views/components/ledger-chart-amounts/ledger-chart-amounts.js b/src/views/components/ledger-chart-amounts/ledger-chart-amounts.js
index b0aa3e06..ddd99076 100644
--- a/src/views/components/ledger-chart-amounts/ledger-chart-amounts.js
+++ b/src/views/components/ledger-chart-amounts/ledger-chart-amounts.js
@@ -13,6 +13,7 @@ import { CanvasRenderer } from 'echarts/renderers'
import Button from '@mui/material/Button'
import { download_csv, download_json } from '@core/utils'
+import { base_ranges, base_range_labels } from '@core/constants'
echarts.use([
TooltipComponent,
@@ -26,49 +27,33 @@ echarts.use([
const color_map = {
// high
- _1000000_count: '#ffd700', // Gold
- _100000_count: '#f9c74f', // Maize Crayola
- _10000_count: '#f4a261', // Sandy Brown
- _1000_count: '#e63946', // Red
+ _1000000: '#ffd700', // Gold
+ _100000: '#f9c74f', // Maize Crayola
+ _10000: '#f4a261', // Sandy Brown
+ _1000: '#e63946', // Red
// medium
- _100_count: '#0077b6', // Star Command Blue
- _10_count: '#023e8a', // Royal Blue Dark
- _1_count: '#0096c7', // Pacific Blue
+ _100: '#0077b6', // Star Command Blue
+ _10: '#023e8a', // Royal Blue Dark
+ _1: '#0096c7', // Pacific Blue
// low
- _01_count: '#6DB65B', // Apple Green
- _001_count: '#57A55A', // Medium Sea Green
- _0001_count: '#3D8B57', // Jungle Green
- _00001_count: '#2C6E49', // Bottle Green
+ _01: '#6DB65B', // Apple Green
+ _001: '#57A55A', // Medium Sea Green
+ _0001: '#3D8B57', // Jungle Green
+ _00001: '#2C6E49', // Bottle Green
// micro
- _000001_count: '#911eb4', // Purple
- _000001_below_count: '#dcbeff' // Lavender Blush
+ _000001: '#911eb4', // Purple
+ _000001_below: '#dcbeff' // Lavender Blush
}
export default function LedgerChartAmounts({ data, isLoading }) {
- const ranges = {
- _1000000_count: '>1M',
- _100000_count: '100k to 1M',
- _10000_count: '10k to 100k',
- _1000_count: '1k to 10k',
- _100_count: '100 to 1k',
- _10_count: '10 to 100',
- _1_count: '1 to 10',
- _01_count: '0.1 to 1',
- _001_count: '0.01 to 0.1',
- _0001_count: '0.001 to 0.01',
- _00001_count: '0.0001 to 0.001',
- _000001_count: '0.00001 to 0.0001',
- _000001_below_count: '<0.00001'
- }
-
- const series_data = Object.entries(ranges).map(([key, value], index) => {
+ const series_data = base_ranges.map((key) => {
const color = color_map[key]
return {
- name: value,
+ name: base_range_labels[key],
type: 'line',
stack: 'total',
color,
@@ -82,7 +67,7 @@ export default function LedgerChartAmounts({ data, isLoading }) {
emphasis: {
focus: 'series'
},
- data: data[key].map((item) => [item[0], item[1]])
+ data: data[`${key}_count`].map((item) => [item[0], item[1]])
}
})
@@ -122,10 +107,10 @@ export default function LedgerChartAmounts({ data, isLoading }) {
const handle_download_csv = () => {
const download_data = []
- Object.entries(ranges).forEach(([key, value]) => {
- data[key].forEach((item) => {
+ base_ranges.forEach((key) => {
+ data[`${key}_count`].forEach((item) => {
download_data.push({
- range: value,
+ range: base_range_labels[key],
date: item[0],
count: item[1]
})
@@ -145,11 +130,13 @@ export default function LedgerChartAmounts({ data, isLoading }) {
data: {}
}
- Object.entries(ranges).forEach(([key, value]) => {
- download_data.data[value] = data[key].map((item) => ({
- date: item[0],
- count: item[1]
- }))
+ base_ranges.forEach((key) => {
+ download_data.data[base_range_labels[key]] = data[`${key}_count`].map(
+ (item) => ({
+ date: item[0],
+ count: item[1]
+ })
+ )
})
download_json({
diff --git a/src/views/components/ledger-chart-distribution/ledger-chart-distribution.js b/src/views/components/ledger-chart-distribution/ledger-chart-distribution.js
index 38999b87..fe123a16 100644
--- a/src/views/components/ledger-chart-distribution/ledger-chart-distribution.js
+++ b/src/views/components/ledger-chart-distribution/ledger-chart-distribution.js
@@ -7,31 +7,32 @@ import { LineChart } from 'echarts/charts'
import { TooltipComponent, GridComponent } from 'echarts/components'
import { CanvasRenderer } from 'echarts/renderers'
+import { base_ranges, base_range_labels } from '@core/constants'
import { download_csv, download_json } from '@core/utils'
echarts.use([LineChart, TooltipComponent, GridComponent, CanvasRenderer])
const color_map = {
// high
- _1000000_relative_supply: '#ffd700', // Gold
- _100000_relative_supply: '#f9c74f', // Maize Crayola
- _10000_relative_supply: '#f4a261', // Sandy Brown
- _1000_relative_supply: '#e63946', // Red
+ _1000000: '#ffd700', // Gold
+ _100000: '#f9c74f', // Maize Crayola
+ _10000: '#f4a261', // Sandy Brown
+ _1000: '#e63946', // Red
// medium
- _100_relative_supply: '#0077b6', // Star Command Blue
- _10_relative_supply: '#023e8a', // Royal Blue Dark
- _1_relative_supply: '#0096c7', // Pacific Blue
+ _100: '#0077b6', // Star Command Blue
+ _10: '#023e8a', // Royal Blue Dark
+ _1: '#0096c7', // Pacific Blue
// low
- _01_relative_supply: '#6DB65B', // Apple Green
- _001_relative_supply: '#57A55A', // Medium Sea Green
- _0001_relative_supply: '#3D8B57', // Jungle Green
- _00001_relative_supply: '#2C6E49', // Bottle Green
+ _01: '#6DB65B', // Apple Green
+ _001: '#57A55A', // Medium Sea Green
+ _0001: '#3D8B57', // Jungle Green
+ _00001: '#2C6E49', // Bottle Green
// micro
- _000001_relative_supply: '#911eb4', // Purple
- _000001_below_relative_supply: '#dcbeff' // Lavender Blush
+ _000001: '#911eb4', // Purple
+ _000001_below: '#dcbeff' // Lavender Blush
}
export default function LedgerChartDistribution({
@@ -54,27 +55,11 @@ export default function LedgerChartDistribution({
return data
}, [price_history])
- const ranges = {
- _1000000_relative_supply: '>1M',
- _100000_relative_supply: '100K to 1M',
- _10000_relative_supply: '10K to 100K',
- _1000_relative_supply: '1K to 10K',
- _100_relative_supply: '100 to 1K',
- _10_relative_supply: '10 to 100',
- _1_relative_supply: '1 to 10',
- _01_relative_supply: '0.1 to 1',
- _001_relative_supply: '0.01 to 0.1',
- _0001_relative_supply: '0.001 to 0.01',
- _00001_relative_supply: '0.0001 to 0.001',
- _000001_relative_supply: '0.00001 to 0.0001',
- _000001_below_relative_supply: '<0.00001'
- }
-
- const series_data = Object.entries(ranges).map(([key, value], index) => {
+ const series_data = base_ranges.map((key) => {
const color = color_map[key]
return {
- name: value,
+ name: base_range_labels[key],
type: 'line',
stack: 'total',
color,
@@ -89,7 +74,7 @@ export default function LedgerChartDistribution({
emphasis: {
focus: 'series'
},
- data: data[key].map((item) => [item[0], item[1]])
+ data: data[`${key}_relative_supply`].map((item) => [item[0], item[1]])
}
})
@@ -147,10 +132,10 @@ export default function LedgerChartDistribution({
const handle_download_csv = () => {
const download_data = []
- Object.entries(ranges).forEach(([key, value]) => {
- data[key].forEach((item) => {
+ base_ranges.forEach((key) => {
+ data[`${key}_relative_supply`].forEach((item) => {
download_data.push({
- range: value,
+ range: base_range_labels[key],
date: item[0],
relative_supply: item[1]
})
@@ -170,8 +155,10 @@ export default function LedgerChartDistribution({
data: {}
}
- Object.entries(ranges).forEach(([key, value]) => {
- download_data.data[value] = data[key].map((item) => ({
+ base_ranges.forEach((key) => {
+ download_data.data[base_range_labels[key]] = data[
+ `${key}_relative_supply`
+ ].map((item) => ({
date: item[0],
relative_supply: item[1]
}))
diff --git a/src/views/pages/ledger/index.js b/src/views/pages/ledger/index.js
index 56c9c7f8..ed448a6c 100644
--- a/src/views/pages/ledger/index.js
+++ b/src/views/pages/ledger/index.js
@@ -3,6 +3,7 @@ import { createSelector } from 'reselect'
import BigNumber from 'bignumber.js'
import { getLedger, ledgerActions } from '@core/ledger'
+import { base_ranges } from '@core/constants'
import LedgerPage from './ledger'
@@ -11,21 +12,6 @@ const mapStateToProps = createSelector(getLedger, (ledger) => {
const data = {
reused_addresses: []
}
- const base_ranges = [
- '_1000000',
- '_100000',
- '_10000',
- '_1000',
- '_100',
- '_10',
- '_1',
- '_01',
- '_001',
- '_0001',
- '_00001',
- '_000001',
- '_000001_below'
- ]
const metrics = [
'active_addresses',
@@ -44,7 +30,8 @@ const mapStateToProps = createSelector(getLedger, (ledger) => {
// TODO open_volume
// TODO receive_volume
- ...base_ranges.map((r) => `${r}_count`)
+ ...base_ranges.map((r) => `${r}_count`),
+ ...base_ranges.map((r) => `${r}_account_count`)
]
for (let i = 0; i < metrics.length; i++) {
diff --git a/src/views/pages/ledger/ledger.js b/src/views/pages/ledger/ledger.js
index bca4cd43..c677e278 100644
--- a/src/views/pages/ledger/ledger.js
+++ b/src/views/pages/ledger/ledger.js
@@ -16,8 +16,10 @@ import LedgerChartVolume from '@components/ledger-chart-volume'
import LedgerChartAmounts from '@components/ledger-chart-amounts'
import LedgerChartUSDTransferred from '@components/ledger-chart-usd-transferred'
import LedgerChartDistribution from '@components/ledger-chart-distribution'
+import LedgerChartAddressesWithBalance from '@components/ledger-chart-addresses-with-balance'
import Seo from '@components/seo'
import Menu from '@components/menu'
+import { base_ranges, base_range_labels } from '@core/constants'
import './ledger.styl'
@@ -36,6 +38,48 @@ export default function LedgerPage({ load, data, isLoading }) {
const [menu_open, set_menu_open] = useState(false)
const filter_input_ref = useRef(null)
+ const categories = {
+ Addresses: {
+ 'Address Counts': (
+
+ )
+ },
+ Blocks: {
+ 'Block Counts':
+ },
+ Transactions: {
+ 'Value Transferred': (
+
+ ),
+ 'Transfer Volume': (
+
+ ),
+ 'Transfer Amounts': (
+
+ )
+ },
+ Distribution: {
+ 'Address Supply Distribution': (
+
+ ),
+ 'Address Balances (Nano)': {
+ ...base_ranges.reduce(
+ (acc, range) => ({
+ ...acc,
+ [`Addresses with Balance ${base_range_labels[range]}`]: (
+
+ )
+ }),
+ {}
+ )
+ }
+ }
+ }
+
useEffect(() => {
load()
}, [])
@@ -78,15 +122,21 @@ export default function LedgerPage({ load, data, isLoading }) {
useEffect(() => {
if (filter_text.length > 0) {
- // Uncollapse all categories when filtering
- set_open_categories(
- Object.keys(filtered_categories).reduce(
- (acc, key) => ({ ...acc, [key]: true }),
- {}
- )
- )
+ // Uncollapse all categories and their children when filtering
+ const all_open_categories = Object.keys(categories).reduce((acc, key) => {
+ const category = categories[key]
+ if (typeof category === 'object') {
+ const sub_keys = Object.keys(category)
+ sub_keys.forEach((sub_key) => {
+ acc[sub_key] = true
+ })
+ }
+ acc[key] = true
+ return acc
+ }, {})
+ set_open_categories(all_open_categories)
}
- }, [filtered_categories])
+ }, [filtered_categories, categories, filter_text])
useEffect(() => {
if (menu_open) {
@@ -135,57 +185,48 @@ export default function LedgerPage({ load, data, isLoading }) {
set_open_categories((prev) => ({ ...prev, [category]: !prev[category] }))
}
- const categories = {
- Addresses: {
- 'Address Counts': (
-
- )
- },
- Blocks: {
- 'Block Counts':
- },
- Transactions: {
- 'Value Transferred': (
-
- ),
- 'Transfer Volume': (
-
- ),
- 'Transfer Amounts': (
-
- )
- },
- Distribution: {
- 'Address Supply Distribution': (
-
- )
- }
+ const count_children = (children) => {
+ return Object.values(children).reduce((acc, child) => {
+ if (React.isValidElement(child)) {
+ return acc + 1
+ } else {
+ return acc + count_children(child)
+ }
+ }, 0)
}
- const render_category = (category, charts, depth) => {
- const is_open = open_categories[category] || false
+ const render_category = (parent_category, children, depth, path = []) => {
+ const is_open = open_categories[parent_category] || false
+ const total_children_count = count_children(children)
return (
<>
toggle_category(category)}
+ onClick={() => toggle_category(parent_category)}
className={`ledger__category ledger__category-depth-${depth}`}>
-
-
+
+
- {Object.entries(charts).map(([key, value]) => {
+ {Object.entries(children).map(([key, value]) => {
+ console.log({
+ path,
+ parent_category,
+ key
+ })
+ const current_path = [...path, parent_category]
const is_selected =
- selected_chart[0] === category && selected_chart[1] === key
+ selected_chart.join(' > ') ===
+ [...current_path, key].join(' > ')
const item_class = is_selected ? 'ledger__chart--selected' : ''
if (React.isValidElement(value)) {
return (
set_selected_chart([category, key])}
+ onClick={() => set_selected_chart([...current_path, key])}
className={`ledger__chart ledger__category-depth-${
depth + 1
} ${item_class}`}>
@@ -193,7 +234,7 @@ export default function LedgerPage({ load, data, isLoading }) {
)
} else {
- return render_category(key, value, depth + 1)
+ return render_category(key, value, depth + 1, current_path)
}
})}
@@ -255,8 +296,8 @@ export default function LedgerPage({ load, data, isLoading }) {
{Object.entries(
filter_text.length > 0 ? filtered_categories : categories
- ).map(([category, charts]) =>
- render_category(category, charts, 0)
+ ).map(([parent_category, children]) =>
+ render_category(parent_category, children, 0)
)}
diff --git a/src/views/pages/ledger/ledger.styl b/src/views/pages/ledger/ledger.styl
index d87a296c..46527cd2 100644
--- a/src/views/pages/ledger/ledger.styl
+++ b/src/views/pages/ledger/ledger.styl
@@ -53,10 +53,20 @@
background hsl(210, 100%, 60%)
.ledger__category-depth-2
- padding-left 49px
+ padding-left 72px
+ &:before
+ content: ""
+ display block
+ position: absolute
+ z-index 1
+ left 44px
+ height 100%
+ width 1px
+ opacity 1
+ background #E5EAF2
.ledger__category-depth-3
- padding-left 62px
+ padding-left 90px
.ledger__footer
margin 32px auto