mirror of
https://github.com/mempool/mempool.git
synced 2024-11-19 09:52:14 +01:00
Improve granularity when zoom in block fees bar graph
This commit is contained in:
parent
42d591bf4c
commit
c2a3ff4b67
@ -24,6 +24,7 @@ class MiningRoutes {
|
|||||||
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/difficulty-adjustments', this.$getDifficultyAdjustments)
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/difficulty-adjustments', this.$getDifficultyAdjustments)
|
||||||
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/reward-stats/:blockCount', this.$getRewardStats)
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/reward-stats/:blockCount', this.$getRewardStats)
|
||||||
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/fees/:interval', this.$getHistoricalBlockFees)
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/fees/:interval', this.$getHistoricalBlockFees)
|
||||||
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/fees', this.$getBlockFeesTimespan)
|
||||||
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/rewards/:interval', this.$getHistoricalBlockRewards)
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/rewards/:interval', this.$getHistoricalBlockRewards)
|
||||||
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/fee-rates/:interval', this.$getHistoricalBlockFeeRates)
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/fee-rates/:interval', this.$getHistoricalBlockFeeRates)
|
||||||
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/sizes-weights/:interval', this.$getHistoricalBlockSizeAndWeight)
|
.get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/sizes-weights/:interval', this.$getHistoricalBlockSizeAndWeight)
|
||||||
@ -217,6 +218,26 @@ class MiningRoutes {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async $getBlockFeesTimespan(req: Request, res: Response) {
|
||||||
|
try {
|
||||||
|
if (!parseInt(req.query.from as string, 10) || !parseInt(req.query.to as string, 10)) {
|
||||||
|
throw new Error('Invalid timestamp range');
|
||||||
|
}
|
||||||
|
if (parseInt(req.query.from as string, 10) > parseInt(req.query.to as string, 10)) {
|
||||||
|
throw new Error('from must be less than to');
|
||||||
|
}
|
||||||
|
const blockFees = await mining.$getBlockFeesTimespan(parseInt(req.query.from as string, 10), parseInt(req.query.to as string, 10));
|
||||||
|
const blockCount = await BlocksRepository.$blockCount(null, null);
|
||||||
|
res.header('Pragma', 'public');
|
||||||
|
res.header('Cache-control', 'public');
|
||||||
|
res.header('X-total-count', blockCount.toString());
|
||||||
|
res.setHeader('Expires', new Date(Date.now() + 1000 * 60).toUTCString());
|
||||||
|
res.json(blockFees);
|
||||||
|
} catch (e) {
|
||||||
|
res.status(500).send(e instanceof Error ? e.message : e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private async $getHistoricalBlockRewards(req: Request, res: Response) {
|
private async $getHistoricalBlockRewards(req: Request, res: Response) {
|
||||||
try {
|
try {
|
||||||
const blockRewards = await mining.$getHistoricalBlockRewards(req.params.interval);
|
const blockRewards = await mining.$getHistoricalBlockRewards(req.params.interval);
|
||||||
|
@ -50,6 +50,17 @@ class Mining {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get timespan block total fees
|
||||||
|
*/
|
||||||
|
public async $getBlockFeesTimespan(from: number, to: number): Promise<number> {
|
||||||
|
return await BlocksRepository.$getHistoricalBlockFees(
|
||||||
|
this.getTimeRangeFromTimespan(from, to),
|
||||||
|
null,
|
||||||
|
{from, to}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get historical block rewards
|
* Get historical block rewards
|
||||||
*/
|
*/
|
||||||
@ -646,6 +657,24 @@ class Mining {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private getTimeRangeFromTimespan(from: number, to: number, scale = 1): number {
|
||||||
|
const timespan = to - from;
|
||||||
|
switch (true) {
|
||||||
|
case timespan > 3600 * 24 * 365 * 4: return 86400 * scale; // 24h
|
||||||
|
case timespan > 3600 * 24 * 365 * 3: return 43200 * scale; // 12h
|
||||||
|
case timespan > 3600 * 24 * 365 * 2: return 43200 * scale; // 12h
|
||||||
|
case timespan > 3600 * 24 * 365: return 28800 * scale; // 8h
|
||||||
|
case timespan > 3600 * 24 * 30 * 6: return 28800 * scale; // 8h
|
||||||
|
case timespan > 3600 * 24 * 30 * 3: return 10800 * scale; // 3h
|
||||||
|
case timespan > 3600 * 24 * 30: return 7200 * scale; // 2h
|
||||||
|
case timespan > 3600 * 24 * 7: return 1800 * scale; // 30min
|
||||||
|
case timespan > 3600 * 24 * 3: return 300 * scale; // 5min
|
||||||
|
case timespan > 3600 * 24: return 1 * scale;
|
||||||
|
default: return 1 * scale;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Finds the oldest block in a consecutive chain back from the tip
|
// Finds the oldest block in a consecutive chain back from the tip
|
||||||
// assumes `blocks` is sorted in ascending height order
|
// assumes `blocks` is sorted in ascending height order
|
||||||
private getOldestConsecutiveBlock(blocks: DifficultyBlock[]): DifficultyBlock {
|
private getOldestConsecutiveBlock(blocks: DifficultyBlock[]): DifficultyBlock {
|
||||||
|
@ -663,7 +663,7 @@ class BlocksRepository {
|
|||||||
/**
|
/**
|
||||||
* Get the historical averaged block fees
|
* Get the historical averaged block fees
|
||||||
*/
|
*/
|
||||||
public async $getHistoricalBlockFees(div: number, interval: string | null): Promise<any> {
|
public async $getHistoricalBlockFees(div: number, interval: string | null, timespan?: {from: number, to: number}): Promise<any> {
|
||||||
try {
|
try {
|
||||||
let query = `SELECT
|
let query = `SELECT
|
||||||
CAST(AVG(blocks.height) as INT) as avgHeight,
|
CAST(AVG(blocks.height) as INT) as avgHeight,
|
||||||
@ -677,6 +677,8 @@ class BlocksRepository {
|
|||||||
|
|
||||||
if (interval !== null) {
|
if (interval !== null) {
|
||||||
query += ` WHERE blockTimestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()`;
|
query += ` WHERE blockTimestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()`;
|
||||||
|
} else if (timespan) {
|
||||||
|
query += ` WHERE blockTimestamp BETWEEN FROM_UNIXTIME(${timespan.from}) AND FROM_UNIXTIME(${timespan.to})`;
|
||||||
}
|
}
|
||||||
|
|
||||||
query += ` GROUP BY UNIX_TIMESTAMP(blockTimestamp) DIV ${div}`;
|
query += ` GROUP BY UNIX_TIMESTAMP(blockTimestamp) DIV ${div}`;
|
||||||
|
@ -45,7 +45,7 @@
|
|||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="chart" *browserOnly echarts [initOpts]="chartInitOptions" [options]="chartOptions"
|
<div class="chart" *browserOnly echarts [initOpts]="chartInitOptions" [options]="chartOptions" [style]="{opacity: isLoading ? 0.5 : 1}"
|
||||||
(chartInit)="onChartInit($event)">
|
(chartInit)="onChartInit($event)">
|
||||||
</div>
|
</div>
|
||||||
<div class="text-center loadingGraphs" *ngIf="!stateService.isBrowser || isLoading">
|
<div class="text-center loadingGraphs" *ngIf="!stateService.isBrowser || isLoading">
|
||||||
|
@ -1,18 +1,19 @@
|
|||||||
import { ChangeDetectionStrategy, Component, Inject, Input, LOCALE_ID, OnInit } from '@angular/core';
|
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, HostListener, Inject, Input, LOCALE_ID, NgZone, OnInit } from '@angular/core';
|
||||||
import { EChartsOption } from '../../graphs/echarts';
|
import { EChartsOption } from '../../graphs/echarts';
|
||||||
import { Observable } from 'rxjs';
|
import { Observable } from 'rxjs';
|
||||||
import { map, share, startWith, switchMap, tap } from 'rxjs/operators';
|
import { catchError, map, share, startWith, switchMap, tap } from 'rxjs/operators';
|
||||||
import { ApiService } from '../../services/api.service';
|
import { ApiService } from '../../services/api.service';
|
||||||
import { SeoService } from '../../services/seo.service';
|
import { SeoService } from '../../services/seo.service';
|
||||||
import { formatNumber } from '@angular/common';
|
import { formatNumber } from '@angular/common';
|
||||||
import { UntypedFormBuilder, UntypedFormGroup } from '@angular/forms';
|
import { UntypedFormBuilder, UntypedFormGroup } from '@angular/forms';
|
||||||
import { download, formatterXAxis } from '../../shared/graphs.utils';
|
import { download, formatterXAxis } from '../../shared/graphs.utils';
|
||||||
import { ActivatedRoute } from '@angular/router';
|
import { ActivatedRoute, Router } from '@angular/router';
|
||||||
import { FiatShortenerPipe } from '../../shared/pipes/fiat-shortener.pipe';
|
import { FiatShortenerPipe } from '../../shared/pipes/fiat-shortener.pipe';
|
||||||
import { FiatCurrencyPipe } from '../../shared/pipes/fiat-currency.pipe';
|
import { FiatCurrencyPipe } from '../../shared/pipes/fiat-currency.pipe';
|
||||||
import { StateService } from '../../services/state.service';
|
import { StateService } from '../../services/state.service';
|
||||||
import { MiningService } from '../../services/mining.service';
|
import { MiningService } from '../../services/mining.service';
|
||||||
import { StorageService } from '../../services/storage.service';
|
import { StorageService } from '../../services/storage.service';
|
||||||
|
import { RelativeUrlPipe } from '../../shared/pipes/relative-url/relative-url.pipe';
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
selector: 'app-block-fees-subsidy-graph',
|
selector: 'app-block-fees-subsidy-graph',
|
||||||
@ -41,11 +42,16 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
};
|
};
|
||||||
|
|
||||||
statsObservable$: Observable<any>;
|
statsObservable$: Observable<any>;
|
||||||
|
data: any;
|
||||||
|
subsidies: { [key: number]: number } = {};
|
||||||
isLoading = true;
|
isLoading = true;
|
||||||
formatNumber = formatNumber;
|
formatNumber = formatNumber;
|
||||||
timespan = '';
|
timespan = '';
|
||||||
chartInstance: any = undefined;
|
chartInstance: any = undefined;
|
||||||
showFiat = false;
|
showFiat = false;
|
||||||
|
updateZoom = false;
|
||||||
|
zoomSpan = 100;
|
||||||
|
zoomTimeSpan = '';
|
||||||
|
|
||||||
constructor(
|
constructor(
|
||||||
@Inject(LOCALE_ID) public locale: string,
|
@Inject(LOCALE_ID) public locale: string,
|
||||||
@ -56,11 +62,16 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
private storageService: StorageService,
|
private storageService: StorageService,
|
||||||
private miningService: MiningService,
|
private miningService: MiningService,
|
||||||
private route: ActivatedRoute,
|
private route: ActivatedRoute,
|
||||||
|
private router: Router,
|
||||||
|
private zone: NgZone,
|
||||||
private fiatShortenerPipe: FiatShortenerPipe,
|
private fiatShortenerPipe: FiatShortenerPipe,
|
||||||
private fiatCurrencyPipe: FiatCurrencyPipe,
|
private fiatCurrencyPipe: FiatCurrencyPipe,
|
||||||
|
private cd: ChangeDetectorRef,
|
||||||
) {
|
) {
|
||||||
this.radioGroupForm = this.formBuilder.group({ dateSpan: '1y' });
|
this.radioGroupForm = this.formBuilder.group({ dateSpan: '1y' });
|
||||||
this.radioGroupForm.controls.dateSpan.setValue('1y');
|
this.radioGroupForm.controls.dateSpan.setValue('1y');
|
||||||
|
|
||||||
|
this.subsidies = this.initSubsidies();
|
||||||
}
|
}
|
||||||
|
|
||||||
ngOnInit(): void {
|
ngOnInit(): void {
|
||||||
@ -86,27 +97,21 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
this.isLoading = true;
|
this.isLoading = true;
|
||||||
this.storageService.setValue('miningWindowPreference', timespan);
|
this.storageService.setValue('miningWindowPreference', timespan);
|
||||||
this.timespan = timespan;
|
this.timespan = timespan;
|
||||||
|
this.zoomTimeSpan = timespan;
|
||||||
this.isLoading = true;
|
this.isLoading = true;
|
||||||
return this.apiService.getHistoricalBlockFees$(timespan)
|
return this.apiService.getHistoricalBlockFees$(timespan)
|
||||||
.pipe(
|
.pipe(
|
||||||
tap((response) => {
|
tap((response) => {
|
||||||
let blockReward = 50 * 100_000_000;
|
this.data = {
|
||||||
const subsidies = {};
|
|
||||||
for (let i = 0; i <= 33; i++) {
|
|
||||||
subsidies[i] = blockReward;
|
|
||||||
blockReward = Math.floor(blockReward / 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = {
|
|
||||||
timestamp: response.body.map(val => val.timestamp * 1000),
|
timestamp: response.body.map(val => val.timestamp * 1000),
|
||||||
blockHeight: response.body.map(val => val.avgHeight),
|
blockHeight: response.body.map(val => val.avgHeight),
|
||||||
blockFees: response.body.map(val => val.avgFees / 100_000_000),
|
blockFees: response.body.map(val => val.avgFees / 100_000_000),
|
||||||
blockFeesFiat: response.body.filter(val => val['USD'] > 0).map(val => val.avgFees / 100_000_000 * val['USD']),
|
blockFeesFiat: response.body.filter(val => val['USD'] > 0).map(val => val.avgFees / 100_000_000 * val['USD']),
|
||||||
blockSubsidy: response.body.map(val => subsidies[Math.floor(Math.min(val.avgHeight / 210000, 33))] / 100_000_000),
|
blockSubsidy: response.body.map(val => this.subsidies[Math.floor(Math.min(val.avgHeight / 210000, 33))] / 100_000_000),
|
||||||
blockSubsidyFiat: response.body.filter(val => val['USD'] > 0).map(val => subsidies[Math.floor(Math.min(val.avgHeight / 210000, 33))] / 100_000_000 * val['USD']),
|
blockSubsidyFiat: response.body.filter(val => val['USD'] > 0).map(val => this.subsidies[Math.floor(Math.min(val.avgHeight / 210000, 33))] / 100_000_000 * val['USD']),
|
||||||
};
|
};
|
||||||
|
|
||||||
this.prepareChartOptions(data);
|
this.prepareChartOptions();
|
||||||
this.isLoading = false;
|
this.isLoading = false;
|
||||||
}),
|
}),
|
||||||
map((response) => {
|
map((response) => {
|
||||||
@ -120,9 +125,9 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
prepareChartOptions(data) {
|
prepareChartOptions() {
|
||||||
let title: object;
|
let title: object;
|
||||||
if (data.blockFees.length === 0) {
|
if (this.data.blockFees.length === 0) {
|
||||||
title = {
|
title = {
|
||||||
textStyle: {
|
textStyle: {
|
||||||
color: 'grey',
|
color: 'grey',
|
||||||
@ -165,13 +170,7 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
if (data.length <= 0) {
|
if (data.length <= 0) {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
|
let tooltip = `<b style="color: white; margin-left: 2px">${formatterXAxis(this.locale, this.zoomTimeSpan, parseInt(this.data.timestamp[data[0].dataIndex], 10))}</b><br>`;
|
||||||
let tooltip = '';
|
|
||||||
if (['24h', '3d'].includes(this.timespan)) {
|
|
||||||
tooltip += $localize`At block <b style="color: white; margin-left: 2px">${data[0].axisValue}</b><br>`;
|
|
||||||
} else {
|
|
||||||
tooltip += $localize`Around block <b style="color: white; margin-left: 2px">${data[0].axisValue}</b><br>`;
|
|
||||||
}
|
|
||||||
for (let i = data.length - 1; i >= 0; i--) {
|
for (let i = data.length - 1; i >= 0; i--) {
|
||||||
const tick = data[i];
|
const tick = data[i];
|
||||||
if (!this.showFiat) tooltip += `${tick.marker} ${tick.seriesName}: ${formatNumber(tick.data, this.locale, '1.0-3')} BTC<br>`;
|
if (!this.showFiat) tooltip += `${tick.marker} ${tick.seriesName}: ${formatNumber(tick.data, this.locale, '1.0-3')} BTC<br>`;
|
||||||
@ -179,13 +178,18 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
}
|
}
|
||||||
if (!this.showFiat) tooltip += `<div style="margin-left: 2px">${formatNumber(data.reduce((acc, val) => acc + val.data, 0), this.locale, '1.0-3')} BTC</div>`;
|
if (!this.showFiat) tooltip += `<div style="margin-left: 2px">${formatNumber(data.reduce((acc, val) => acc + val.data, 0), this.locale, '1.0-3')} BTC</div>`;
|
||||||
else tooltip += `<div style="margin-left: 2px">${this.fiatCurrencyPipe.transform(data.reduce((acc, val) => acc + val.data, 0), null, 'USD')}</div>`;
|
else tooltip += `<div style="margin-left: 2px">${this.fiatCurrencyPipe.transform(data.reduce((acc, val) => acc + val.data, 0), null, 'USD')}</div>`;
|
||||||
|
if (['24h', '3d'].includes(this.zoomTimeSpan)) {
|
||||||
|
tooltip += `<small>` + $localize`At block <b style="color: white; margin-left: 2px">${data[0].axisValue}` + `</small>`;
|
||||||
|
} else {
|
||||||
|
tooltip += `<small>` + $localize`Around block <b style="color: white; margin-left: 2px">${data[0].axisValue}` + `</small>`;
|
||||||
|
}
|
||||||
return tooltip;
|
return tooltip;
|
||||||
}.bind(this)
|
}.bind(this)
|
||||||
},
|
},
|
||||||
xAxis: data.blockFees.length === 0 ? undefined : [
|
xAxis: this.data.blockFees.length === 0 ? undefined : [
|
||||||
{
|
{
|
||||||
type: 'category',
|
type: 'category',
|
||||||
data: data.blockHeight,
|
data: this.data.blockHeight,
|
||||||
show: false,
|
show: false,
|
||||||
axisLabel: {
|
axisLabel: {
|
||||||
hideOverlap: true,
|
hideOverlap: true,
|
||||||
@ -193,7 +197,7 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: 'category',
|
type: 'category',
|
||||||
data: data.timestamp,
|
data: this.data.timestamp,
|
||||||
show: true,
|
show: true,
|
||||||
position: 'bottom',
|
position: 'bottom',
|
||||||
axisLabel: {
|
axisLabel: {
|
||||||
@ -213,7 +217,7 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
legend: data.blockFees.length === 0 ? undefined : {
|
legend: this.data.blockFees.length === 0 ? undefined : {
|
||||||
data: [
|
data: [
|
||||||
{
|
{
|
||||||
name: 'Subsidy',
|
name: 'Subsidy',
|
||||||
@ -255,7 +259,7 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
'Fees': !this.showFiat,
|
'Fees': !this.showFiat,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
yAxis: data.blockFees.length === 0 ? undefined : [
|
yAxis: this.data.blockFees.length === 0 ? undefined : [
|
||||||
{
|
{
|
||||||
type: 'value',
|
type: 'value',
|
||||||
axisLabel: {
|
axisLabel: {
|
||||||
@ -287,37 +291,37 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
series: data.blockFees.length === 0 ? undefined : [
|
series: this.data.blockFees.length === 0 ? undefined : [
|
||||||
{
|
{
|
||||||
name: 'Subsidy',
|
name: 'Subsidy',
|
||||||
yAxisIndex: 0,
|
yAxisIndex: 0,
|
||||||
type: 'bar',
|
type: 'bar',
|
||||||
stack: 'total',
|
stack: 'total',
|
||||||
data: data.blockSubsidy,
|
data: this.data.blockSubsidy,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Fees',
|
name: 'Fees',
|
||||||
yAxisIndex: 0,
|
yAxisIndex: 0,
|
||||||
type: 'bar',
|
type: 'bar',
|
||||||
stack: 'total',
|
stack: 'total',
|
||||||
data: data.blockFees,
|
data: this.data.blockFees,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Subsidy (USD)',
|
name: 'Subsidy (USD)',
|
||||||
yAxisIndex: 1,
|
yAxisIndex: 1,
|
||||||
type: 'bar',
|
type: 'bar',
|
||||||
stack: 'total',
|
stack: 'total',
|
||||||
data: data.blockSubsidyFiat,
|
data: this.data.blockSubsidyFiat,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'Fees (USD)',
|
name: 'Fees (USD)',
|
||||||
yAxisIndex: 1,
|
yAxisIndex: 1,
|
||||||
type: 'bar',
|
type: 'bar',
|
||||||
stack: 'total',
|
stack: 'total',
|
||||||
data: data.blockFeesFiat,
|
data: this.data.blockFeesFiat,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
dataZoom: data.blockFees.length === 0 ? undefined : [{
|
dataZoom: this.data.blockFees.length === 0 ? undefined : [{
|
||||||
type: 'inside',
|
type: 'inside',
|
||||||
realtime: true,
|
realtime: true,
|
||||||
zoomLock: true,
|
zoomLock: true,
|
||||||
@ -364,12 +368,168 @@ export class BlockFeesSubsidyGraphComponent implements OnInit {
|
|||||||
this.chartInstance.dispatchAction({ type: 'legendUnSelect', name: 'Fees (USD)' });
|
this.chartInstance.dispatchAction({ type: 'legendUnSelect', name: 'Fees (USD)' });
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.chartInstance.on('datazoom', (params) => {
|
||||||
|
if (params.silent || this.isLoading) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.updateZoom = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.chartInstance.on('click', (e) => {
|
||||||
|
this.zone.run(() => {
|
||||||
|
if (['24h', '3d'].includes(this.zoomTimeSpan)) {
|
||||||
|
const url = new RelativeUrlPipe(this.stateService).transform(`/block/${e.name}`);
|
||||||
|
if (e.event.event.shiftKey || e.event.event.ctrlKey || e.event.event.metaKey) {
|
||||||
|
window.open(url);
|
||||||
|
} else {
|
||||||
|
this.router.navigate([url]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@HostListener('document:mouseup', ['$event'])
|
||||||
|
onMouseUp(event: MouseEvent) {
|
||||||
|
if (this.updateZoom) {
|
||||||
|
this.onZoom();
|
||||||
|
this.updateZoom = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
isMobile() {
|
isMobile() {
|
||||||
return (window.innerWidth <= 767.98);
|
return (window.innerWidth <= 767.98);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
initSubsidies(): { [key: number]: number } {
|
||||||
|
let blockReward = 50 * 100_000_000;
|
||||||
|
const subsidies = {};
|
||||||
|
for (let i = 0; i <= 33; i++) {
|
||||||
|
subsidies[i] = blockReward;
|
||||||
|
blockReward = Math.floor(blockReward / 2);
|
||||||
|
}
|
||||||
|
return subsidies;
|
||||||
|
}
|
||||||
|
|
||||||
|
onZoom() {
|
||||||
|
const option = this.chartInstance.getOption();
|
||||||
|
const timestamps = option.xAxis[1].data;
|
||||||
|
const startTimestamp = timestamps[option.dataZoom[0].startValue];
|
||||||
|
const endTimestamp = timestamps[option.dataZoom[0].endValue];
|
||||||
|
|
||||||
|
// Avoid to fetch new data if not needed
|
||||||
|
if (option.dataZoom[0].end - option.dataZoom[0].start >= this.zoomSpan + 0.1
|
||||||
|
|| this.getTimeRangeFromTimespan(Math.floor(startTimestamp / 1000), Math.floor(endTimestamp / 1000), false) as number === this.getTimeRange(this.zoomTimeSpan)
|
||||||
|
) {
|
||||||
|
this.zoomSpan = option.dataZoom[0].end - option.dataZoom[0].start;
|
||||||
|
this.zoomTimeSpan = this.getTimeRangeFromTimespan(Math.floor(startTimestamp / 1000), Math.floor(endTimestamp / 1000), true) as string;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isLoading = true;
|
||||||
|
this.cd.detectChanges();
|
||||||
|
|
||||||
|
const subscription = this.apiService.getBlockFeesFromTimespan$(Math.floor(startTimestamp / 1000), Math.floor(endTimestamp / 1000))
|
||||||
|
.pipe(
|
||||||
|
tap((response) => {
|
||||||
|
const startIndex = option.dataZoom[0].startValue;
|
||||||
|
const endIndex = option.dataZoom[0].endValue;
|
||||||
|
|
||||||
|
// Update series with more granular data
|
||||||
|
const lengthBefore = this.data.timestamp.length;
|
||||||
|
this.data.timestamp.splice(startIndex, endIndex - startIndex, ...response.body.map(val => val.timestamp * 1000));
|
||||||
|
this.data.blockHeight.splice(startIndex, endIndex - startIndex, ...response.body.map(val => val.avgHeight));
|
||||||
|
this.data.blockFees.splice(startIndex, endIndex - startIndex, ...response.body.map(val => val.avgFees / 100_000_000));
|
||||||
|
this.data.blockFeesFiat.splice(startIndex, endIndex - startIndex, ...response.body.filter(val => val['USD'] > 0).map(val => val.avgFees / 100_000_000 * val['USD']));
|
||||||
|
this.data.blockSubsidy.splice(startIndex, endIndex - startIndex, ...response.body.map(val => this.subsidies[Math.floor(Math.min(val.avgHeight / 210000, 33))] / 100_000_000));
|
||||||
|
this.data.blockSubsidyFiat.splice(startIndex, endIndex - startIndex, ...response.body.filter(val => val['USD'] > 0).map(val => this.subsidies[Math.floor(Math.min(val.avgHeight / 210000, 33))] / 100_000_000 * val['USD']));
|
||||||
|
option.series[0].data = this.data.blockSubsidy;
|
||||||
|
option.series[1].data = this.data.blockFees;
|
||||||
|
option.series[2].data = this.data.blockSubsidyFiat;
|
||||||
|
option.series[3].data = this.data.blockFeesFiat;
|
||||||
|
option.xAxis[0].data = this.data.blockHeight;
|
||||||
|
option.xAxis[1].data = this.data.timestamp;
|
||||||
|
this.chartInstance.setOption(option, true);
|
||||||
|
const lengthAfter = this.data.timestamp.length;
|
||||||
|
|
||||||
|
// Update the zoom to keep the same range after the update
|
||||||
|
this.chartInstance.dispatchAction({
|
||||||
|
type: 'dataZoom',
|
||||||
|
startValue: startIndex,
|
||||||
|
endValue: endIndex + lengthAfter - lengthBefore,
|
||||||
|
silent: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Update the chart
|
||||||
|
const newOption = this.chartInstance.getOption();
|
||||||
|
this.zoomSpan = newOption.dataZoom[0].end - newOption.dataZoom[0].start;
|
||||||
|
this.zoomTimeSpan = this.getTimeRangeFromTimespan(Math.floor(this.data.timestamp[newOption.dataZoom[0].startValue] / 1000), Math.floor(this.data.timestamp[newOption.dataZoom[0].endValue] / 1000), true) as string;
|
||||||
|
this.isLoading = false;
|
||||||
|
}),
|
||||||
|
catchError(() => {
|
||||||
|
const newOption = this.chartInstance.getOption();
|
||||||
|
this.zoomSpan = newOption.dataZoom[0].end - newOption.dataZoom[0].start;
|
||||||
|
this.zoomTimeSpan = this.getTimeRangeFromTimespan(Math.floor(this.data.timestamp[newOption.dataZoom[0].startValue] / 1000), Math.floor(this.data.timestamp[newOption.dataZoom[0].endValue] / 1000), true) as string;
|
||||||
|
this.isLoading = false;
|
||||||
|
this.cd.detectChanges();
|
||||||
|
return [];
|
||||||
|
})
|
||||||
|
).subscribe(() => {
|
||||||
|
subscription.unsubscribe();
|
||||||
|
this.cd.detectChanges();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
getTimeRange(interval: string, scale = 1): number {
|
||||||
|
switch (interval) {
|
||||||
|
case '4y': return 43200 * scale; // 12h
|
||||||
|
case '3y': return 43200 * scale; // 12h
|
||||||
|
case '2y': return 28800 * scale; // 8h
|
||||||
|
case '1y': return 28800 * scale; // 8h
|
||||||
|
case '6m': return 10800 * scale; // 3h
|
||||||
|
case '3m': return 7200 * scale; // 2h
|
||||||
|
case '1m': return 1800 * scale; // 30min
|
||||||
|
case '1w': return 300 * scale; // 5min
|
||||||
|
case '3d': return 1 * scale;
|
||||||
|
case '24h': return 1 * scale;
|
||||||
|
default: return 86400 * scale;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
getTimeRangeFromTimespan(from: number, to: number, toString: boolean, scale = 1): number | string {
|
||||||
|
const timespan = to - from;
|
||||||
|
if (toString) {
|
||||||
|
switch (true) {
|
||||||
|
case timespan >= 3600 * 24 * 365 * 4: return 'all';
|
||||||
|
case timespan >= 3600 * 24 * 365 * 3: return '4y';
|
||||||
|
case timespan >= 3600 * 24 * 365 * 2: return '3y';
|
||||||
|
case timespan >= 3600 * 24 * 365: return '2y';
|
||||||
|
case timespan >= 3600 * 24 * 30 * 6: return '1y';
|
||||||
|
case timespan >= 3600 * 24 * 30 * 3: return '6m';
|
||||||
|
case timespan >= 3600 * 24 * 30: return '3m';
|
||||||
|
case timespan >= 3600 * 24 * 7: return '1m';
|
||||||
|
case timespan >= 3600 * 24 * 3: return '1w';
|
||||||
|
case timespan >= 3600 * 24: return '3d';
|
||||||
|
default: return '24h';
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
switch (true) {
|
||||||
|
case timespan > 3600 * 24 * 365 * 4: return 86400 * scale; // 24h
|
||||||
|
case timespan > 3600 * 24 * 365 * 3: return 43200 * scale; // 12h
|
||||||
|
case timespan > 3600 * 24 * 365 * 2: return 43200 * scale; // 12h
|
||||||
|
case timespan > 3600 * 24 * 365: return 28800 * scale; // 8h
|
||||||
|
case timespan > 3600 * 24 * 30 * 6: return 28800 * scale; // 8h
|
||||||
|
case timespan > 3600 * 24 * 30 * 3: return 10800 * scale; // 3h
|
||||||
|
case timespan > 3600 * 24 * 30: return 7200 * scale; // 2h
|
||||||
|
case timespan > 3600 * 24 * 7: return 1800 * scale; // 30min
|
||||||
|
case timespan > 3600 * 24 * 3: return 300 * scale; // 5min
|
||||||
|
case timespan > 3600 * 24: return 1 * scale;
|
||||||
|
default: return 1 * scale;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onSaveChart() {
|
onSaveChart() {
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
const prevBottom = this.chartOptions.grid.bottom;
|
const prevBottom = this.chartOptions.grid.bottom;
|
||||||
|
@ -333,6 +333,12 @@ export class ApiService {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getBlockFeesFromTimespan$(from: number, to: number): Observable<any> {
|
||||||
|
return this.httpClient.get<any[]>(
|
||||||
|
this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/blocks/fees?from=${from}&to=${to}`, { observe: 'response' }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
getHistoricalBlockRewards$(interval: string | undefined) : Observable<any> {
|
getHistoricalBlockRewards$(interval: string | undefined) : Observable<any> {
|
||||||
return this.httpClient.get<any[]>(
|
return this.httpClient.get<any[]>(
|
||||||
this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/blocks/rewards` +
|
this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/blocks/rewards` +
|
||||||
|
Loading…
Reference in New Issue
Block a user