2020-03-29 18:59:04 +02:00
|
|
|
import { Component, OnInit, Input, Inject, LOCALE_ID, ChangeDetectionStrategy, OnChanges } from '@angular/core';
|
|
|
|
import { formatDate } from '@angular/common';
|
2020-07-03 18:45:19 +02:00
|
|
|
import { VbytesPipe } from 'src/app/shared/pipes/bytes-pipe/vbytes.pipe';
|
2020-03-29 18:59:04 +02:00
|
|
|
import { OptimizedMempoolStats } from 'src/app/interfaces/node-api.interface';
|
2020-05-09 15:37:50 +02:00
|
|
|
import { StateService } from 'src/app/services/state.service';
|
2020-11-16 13:27:06 +01:00
|
|
|
import { StorageService } from 'src/app/services/storage.service';
|
2021-08-21 06:46:28 +02:00
|
|
|
import { EChartsOption } from 'echarts';
|
|
|
|
import { feeLevels, chartColors } from 'src/app/app.constants';
|
|
|
|
|
2020-03-29 18:59:04 +02:00
|
|
|
@Component({
|
|
|
|
selector: 'app-mempool-graph',
|
|
|
|
templateUrl: './mempool-graph.component.html',
|
|
|
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
|
|
})
|
|
|
|
export class MempoolGraphComponent implements OnInit, OnChanges {
|
2021-08-21 06:46:28 +02:00
|
|
|
@Input() data: any[];
|
2021-08-27 04:30:57 +02:00
|
|
|
@Input() limitFee = 350;
|
2021-08-21 06:46:28 +02:00
|
|
|
@Input() height: number | string = 200;
|
|
|
|
@Input() top: number | string = 20;
|
|
|
|
@Input() right: number | string = 10;
|
|
|
|
@Input() left: number | string = 75;
|
2021-08-27 04:30:57 +02:00
|
|
|
@Input() template: ('widget' | 'advanced') = 'widget';
|
|
|
|
@Input() showZoom = true;
|
2020-03-29 18:59:04 +02:00
|
|
|
|
|
|
|
mempoolVsizeFeesData: any;
|
2021-08-21 06:46:28 +02:00
|
|
|
mempoolVsizeFeesOptions: EChartsOption;
|
2021-09-07 22:48:43 +02:00
|
|
|
mempoolVsizeFeesInitOptions = {
|
|
|
|
renderer: 'svg',
|
|
|
|
};
|
2021-08-25 06:01:35 +02:00
|
|
|
windowPreference: string;
|
|
|
|
hoverIndexSerie: -1;
|
2021-08-27 04:30:57 +02:00
|
|
|
feeLimitIndex: number;
|
2020-09-23 10:49:07 +02:00
|
|
|
|
2020-03-29 18:59:04 +02:00
|
|
|
constructor(
|
|
|
|
private vbytesPipe: VbytesPipe,
|
2020-05-09 15:37:50 +02:00
|
|
|
private stateService: StateService,
|
2021-08-25 06:01:35 +02:00
|
|
|
private storageService: StorageService,
|
2020-03-29 18:59:04 +02:00
|
|
|
@Inject(LOCALE_ID) private locale: string,
|
|
|
|
) { }
|
|
|
|
|
|
|
|
ngOnInit(): void {
|
2021-08-21 06:46:28 +02:00
|
|
|
this.mountFeeChart();
|
2020-03-29 18:59:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ngOnChanges() {
|
2021-08-25 06:01:35 +02:00
|
|
|
this.windowPreference = this.storageService.getValue('graphWindowPreference');
|
2020-03-29 18:59:04 +02:00
|
|
|
this.mempoolVsizeFeesData = this.handleNewMempoolData(this.data.concat([]));
|
2021-08-21 06:46:28 +02:00
|
|
|
this.mountFeeChart();
|
2020-03-29 18:59:04 +02:00
|
|
|
}
|
|
|
|
|
2021-08-25 06:01:35 +02:00
|
|
|
onChartReady(myChart: any) {
|
|
|
|
myChart.on('mouseover', 'series', (serie: any) => {
|
|
|
|
this.hoverIndexSerie = serie.seriesIndex;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-03-29 18:59:04 +02:00
|
|
|
handleNewMempoolData(mempoolStats: OptimizedMempoolStats[]) {
|
|
|
|
mempoolStats.reverse();
|
|
|
|
const labels = mempoolStats.map(stats => stats.added);
|
2021-08-21 06:46:28 +02:00
|
|
|
const finalArrayVByte = this.generateArray(mempoolStats);
|
2020-03-29 18:59:04 +02:00
|
|
|
|
|
|
|
// Only Liquid has lower than 1 sat/vb transactions
|
2020-05-17 12:06:09 +02:00
|
|
|
if (this.stateService.network !== 'liquid') {
|
2021-08-21 06:46:28 +02:00
|
|
|
finalArrayVByte.shift();
|
2020-03-29 18:59:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
labels: labels,
|
2021-08-21 06:46:28 +02:00
|
|
|
series: finalArrayVByte
|
2020-03-29 18:59:04 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
generateArray(mempoolStats: OptimizedMempoolStats[]) {
|
|
|
|
const finalArray: number[][] = [];
|
|
|
|
let feesArray: number[] = [];
|
|
|
|
for (let index = 37; index > -1; index--) {
|
|
|
|
feesArray = [];
|
|
|
|
mempoolStats.forEach((stats) => {
|
2021-09-07 22:48:43 +02:00
|
|
|
feesArray.push(stats.vsizes[index] ? stats.vsizes[index] : 0);
|
2020-03-29 18:59:04 +02:00
|
|
|
});
|
|
|
|
finalArray.push(feesArray);
|
|
|
|
}
|
|
|
|
finalArray.reverse();
|
|
|
|
return finalArray;
|
|
|
|
}
|
2021-08-21 06:46:28 +02:00
|
|
|
|
|
|
|
mountFeeChart(){
|
|
|
|
const { labels, series } = this.mempoolVsizeFeesData;
|
|
|
|
|
2021-08-25 06:01:35 +02:00
|
|
|
const feeLevelsOrdered = feeLevels.map((sat, i, arr) => {
|
2021-08-27 04:30:57 +02:00
|
|
|
if (arr[i] === this.limitFee) { this.feeLimitIndex = i; }
|
|
|
|
if (arr[i] < this.limitFee) {
|
2021-08-25 06:01:35 +02:00
|
|
|
if (i === 0) { return '0 - 1'; }
|
2021-08-27 04:30:57 +02:00
|
|
|
return `${arr[i - 1]} - ${arr[i]}`;
|
|
|
|
} else {
|
|
|
|
return `${this.limitFee}+`;
|
2021-08-25 06:01:35 +02:00
|
|
|
}
|
2021-08-21 06:46:28 +02:00
|
|
|
});
|
|
|
|
|
2021-09-07 22:48:43 +02:00
|
|
|
const seriesGraph = series.map((value: Array<number>, index: number) => {
|
2021-08-27 04:30:57 +02:00
|
|
|
if (index <= this.feeLimitIndex){
|
2021-08-25 06:01:35 +02:00
|
|
|
return {
|
|
|
|
type: 'line',
|
|
|
|
stack: 'total',
|
|
|
|
smooth: false,
|
|
|
|
markPoint: {
|
|
|
|
symbol: 'rect',
|
2021-08-21 06:46:28 +02:00
|
|
|
},
|
|
|
|
lineStyle: {
|
2021-08-25 06:01:35 +02:00
|
|
|
width: 0,
|
|
|
|
opacity: 0,
|
2021-08-21 06:46:28 +02:00
|
|
|
},
|
2021-08-27 04:30:57 +02:00
|
|
|
symbolSize: (this.template === 'advanced') ? 20 : 10,
|
2021-08-25 06:01:35 +02:00
|
|
|
showSymbol: false,
|
|
|
|
areaStyle: {
|
|
|
|
opacity: 1,
|
|
|
|
color: chartColors[index],
|
|
|
|
},
|
|
|
|
emphasis: {
|
|
|
|
focus: 'series',
|
2021-08-27 04:30:57 +02:00
|
|
|
select: {
|
|
|
|
areaStyle: {
|
|
|
|
opacity: 1,
|
|
|
|
}
|
|
|
|
},
|
2021-08-25 06:01:35 +02:00
|
|
|
},
|
|
|
|
itemStyle: {
|
|
|
|
borderWidth: 30,
|
|
|
|
},
|
|
|
|
data: this.vbytesPipe.transform(value, 2, 'vB', 'MvB', true)
|
|
|
|
};
|
|
|
|
}
|
2021-08-21 06:46:28 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
this.mempoolVsizeFeesOptions = {
|
|
|
|
color: chartColors,
|
|
|
|
tooltip: {
|
|
|
|
trigger: 'axis',
|
|
|
|
position: (pos, params, el, elRect, size) => {
|
2021-08-27 04:30:57 +02:00
|
|
|
const positions = { top: (this.template === 'advanced') ? 30 : -30 };
|
|
|
|
positions[['left', 'right'][+(pos[0] < size.viewSize[0] / 2)]] = 60;
|
2021-08-21 06:46:28 +02:00
|
|
|
return positions;
|
|
|
|
},
|
2021-08-27 04:30:57 +02:00
|
|
|
extraCssText: `width: ${(this.template === 'advanced') ? '210px' : '200px'};
|
2021-08-21 06:46:28 +02:00
|
|
|
background: transparent;
|
|
|
|
border: none;
|
|
|
|
box-shadow: none;`,
|
|
|
|
axisPointer: {
|
2021-08-25 06:01:35 +02:00
|
|
|
type: 'line',
|
2021-08-21 06:46:28 +02:00
|
|
|
},
|
|
|
|
formatter: (params: any) => {
|
2021-08-25 06:01:35 +02:00
|
|
|
const legendName = (index: number) => feeLevelsOrdered[index];
|
2021-09-07 22:48:43 +02:00
|
|
|
const colorSpan = (index: number) => `<span class="indicator" style="background-color: ${chartColors[index]}"></span><span>${legendName(index)}`;
|
2021-08-25 06:01:35 +02:00
|
|
|
let total = 0;
|
2021-09-07 22:48:43 +02:00
|
|
|
params.map((item: any) => {
|
2021-08-25 06:01:35 +02:00
|
|
|
total += item.value;
|
2021-08-27 04:30:57 +02:00
|
|
|
});
|
2021-09-07 22:48:43 +02:00
|
|
|
const title = `<div class="title">${params[0].axisValue}
|
|
|
|
<span class="total-value">${this.vbytesPipe.transform(total, 2, 'vB', 'MvB', false)}</span></div>`;
|
2021-08-27 04:30:57 +02:00
|
|
|
const itemFormatted = [];
|
|
|
|
let totalParcial = 0;
|
|
|
|
let progressPercentageText = '';
|
2021-09-07 22:48:43 +02:00
|
|
|
params.map((item: any, index: number) => {
|
2021-08-27 04:30:57 +02:00
|
|
|
totalParcial += item.value;
|
|
|
|
let progressPercentage = 0;
|
|
|
|
if (index <= this.feeLimitIndex) {
|
|
|
|
progressPercentage = (item.value / total) * 100;
|
2021-08-25 06:01:35 +02:00
|
|
|
let activeItemClass = '';
|
2021-08-27 04:30:57 +02:00
|
|
|
if (this.hoverIndexSerie === index) {
|
|
|
|
progressPercentageText = `<div class="total-parcial-active">
|
2021-09-07 22:48:43 +02:00
|
|
|
<span class="progress-percentage">${progressPercentage.toFixed(2)} <span class="symbol">%</span></span><span class="total-parcial-vbytes">${this.vbytesPipe.transform(totalParcial, 2, 'vB', 'MvB', false)}</span>
|
|
|
|
<div class="total-percentage-bar"><span><span style="width: ${progressPercentage}%; background: ${chartColors[index]}"></span></span></div>
|
2021-08-27 04:30:57 +02:00
|
|
|
</div>`;
|
2021-08-25 06:01:35 +02:00
|
|
|
activeItemClass = 'active';
|
|
|
|
}
|
2021-08-27 04:30:57 +02:00
|
|
|
itemFormatted.push(`<tr class="item ${activeItemClass}">
|
|
|
|
<td class="indicator-container">${colorSpan(index)}</td>
|
|
|
|
<td class="value">${this.vbytesPipe.transform(item.value, 2, 'vB', 'MvB', false)}</td>
|
2021-09-07 22:48:43 +02:00
|
|
|
<td class="total-progress-percentage"><span color: ${chartColors[index]}">${progressPercentage.toFixed(1)} <span class="symbol">%</span></td>
|
2021-08-27 04:30:57 +02:00
|
|
|
</tr>`);
|
2021-08-21 06:46:28 +02:00
|
|
|
}
|
|
|
|
});
|
2021-08-27 04:30:57 +02:00
|
|
|
const progressActiveDiv = `<span class="total-value">${progressPercentageText}</span>`;
|
|
|
|
const advancedClass = (this.template === 'advanced') ? 'fees-wrapper-tooltip-chart-advanced' : '';
|
|
|
|
return `<div class="fees-wrapper-tooltip-chart ${advancedClass}">
|
|
|
|
${title}
|
|
|
|
<table>
|
|
|
|
<thead>
|
|
|
|
<tr>
|
|
|
|
<th>Range</th>
|
|
|
|
<th>Size</th>
|
|
|
|
<th>%</th>
|
|
|
|
</tr>
|
|
|
|
</thead>
|
|
|
|
<tbody>${itemFormatted.reverse().join('')}</tbody>
|
|
|
|
</table>
|
|
|
|
${progressActiveDiv}
|
2021-08-25 06:01:35 +02:00
|
|
|
</div>`;
|
2021-08-21 06:46:28 +02:00
|
|
|
}
|
|
|
|
},
|
2021-08-25 06:01:35 +02:00
|
|
|
dataZoom: [{
|
|
|
|
type: 'inside',
|
|
|
|
realtime: true,
|
2021-08-27 04:30:57 +02:00
|
|
|
zoomOnMouseWheel: (this.template === 'advanced') ? true : false,
|
|
|
|
maxSpan: 100,
|
|
|
|
minSpan: 10,
|
2021-08-25 06:01:35 +02:00
|
|
|
}, {
|
2021-08-27 04:30:57 +02:00
|
|
|
show: (this.template === 'advanced' && this.showZoom) ? true : false,
|
2021-08-25 06:01:35 +02:00
|
|
|
type: 'slider',
|
|
|
|
brushSelect: false,
|
|
|
|
realtime: true,
|
|
|
|
selectedDataBackground: {
|
|
|
|
lineStyle: {
|
|
|
|
color: '#fff',
|
|
|
|
opacity: 0.45,
|
|
|
|
},
|
|
|
|
areaStyle: {
|
|
|
|
opacity: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}],
|
2021-08-27 04:30:57 +02:00
|
|
|
animation: false,
|
2021-08-21 06:46:28 +02:00
|
|
|
grid: {
|
|
|
|
height: this.height,
|
|
|
|
right: this.right,
|
|
|
|
top: this.top,
|
|
|
|
left: this.left,
|
|
|
|
},
|
|
|
|
xAxis: [
|
|
|
|
{
|
|
|
|
type: 'category',
|
|
|
|
boundaryGap: false,
|
2021-08-25 06:01:35 +02:00
|
|
|
axisLine: { onZero: false },
|
2021-08-27 04:29:39 +02:00
|
|
|
data: labels.map((value: any) => formatDate(value, 'MM/dd - HH:mm', this.locale)),
|
2021-08-21 06:46:28 +02:00
|
|
|
}
|
|
|
|
],
|
|
|
|
yAxis: {
|
|
|
|
type: 'value',
|
2021-08-25 06:01:35 +02:00
|
|
|
axisLine: { onZero: false },
|
2021-08-21 06:46:28 +02:00
|
|
|
axisLabel: {
|
|
|
|
formatter: (value: number) => (`${this.vbytesPipe.transform(value, 2, 'vB', 'MvB', true)}`),
|
|
|
|
},
|
|
|
|
splitLine: {
|
|
|
|
lineStyle: {
|
|
|
|
type: 'dotted',
|
|
|
|
color: '#ffffff66',
|
|
|
|
opacity: 0.25,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2021-09-07 22:48:43 +02:00
|
|
|
series: seriesGraph
|
2021-08-21 06:46:28 +02:00
|
|
|
};
|
|
|
|
}
|
2020-03-29 18:59:04 +02:00
|
|
|
}
|