Merge pull request #1512 from mempool/nymkappa/feature/reward-fee-graph
Add block fees graph
This commit is contained in:
		
						commit
						233af87eb4
					
				| @ -5,6 +5,7 @@ import HashratesRepository from '../repositories/HashratesRepository'; | ||||
| import bitcoinClient from './bitcoin/bitcoin-client'; | ||||
| import logger from '../logger'; | ||||
| import blocks from './blocks'; | ||||
| import { Common } from './common'; | ||||
| 
 | ||||
| class Mining { | ||||
|   hashrateIndexingStarted = false; | ||||
| @ -13,6 +14,29 @@ class Mining { | ||||
|   constructor() { | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get historical block reward and total fee | ||||
|    */ | ||||
|   public async $getHistoricalBlockFees(interval: string | null = null): Promise<any> { | ||||
|     let timeRange: number; | ||||
|     switch (interval) { | ||||
|       case '3y': timeRange = 43200; break; // 12h
 | ||||
|       case '2y': timeRange = 28800; break; // 8h
 | ||||
|       case '1y': timeRange = 28800; break; // 8h
 | ||||
|       case '6m': timeRange = 10800; break; // 3h
 | ||||
|       case '3m': timeRange = 7200; break; // 2h
 | ||||
|       case '1m': timeRange = 1800; break; // 30min
 | ||||
|       case '1w': timeRange = 300; break; // 5min
 | ||||
|       case '3d': timeRange = 1; break; | ||||
|       case '24h': timeRange = 1; break; | ||||
|       default: timeRange = 86400; break; // 24h
 | ||||
|     } | ||||
| 
 | ||||
|     interval = Common.getSqlInterval(interval); | ||||
| 
 | ||||
|     return await BlocksRepository.$getHistoricalBlockFees(timeRange, interval); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Generate high level overview of the pool ranks and general stats | ||||
|    */ | ||||
|  | ||||
| @ -316,6 +316,7 @@ class Server { | ||||
|         .get(config.MEMPOOL.API_URL_PREFIX + 'mining/hashrate', routes.$getHistoricalHashrate) | ||||
|         .get(config.MEMPOOL.API_URL_PREFIX + 'mining/hashrate/:interval', routes.$getHistoricalHashrate) | ||||
|         .get(config.MEMPOOL.API_URL_PREFIX + 'mining/reward-stats/:blockCount', routes.$getRewardStats) | ||||
|         .get(config.MEMPOOL.API_URL_PREFIX + 'mining/blocks/fees/:interval', routes.$getHistoricalBlockFees) | ||||
|       ; | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -439,6 +439,35 @@ class BlocksRepository { | ||||
| 
 | ||||
|     connection.release(); | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get the historical averaged block reward and total fees | ||||
|    */ | ||||
|   public async $getHistoricalBlockFees(div: number, interval: string | null): Promise<any> { | ||||
|     let connection; | ||||
|     try { | ||||
|       connection = await DB.getConnection(); | ||||
| 
 | ||||
|       let query = `SELECT CAST(AVG(UNIX_TIMESTAMP(blockTimestamp)) as INT) as timestamp,
 | ||||
|         CAST(AVG(fees) as INT) as avg_fees | ||||
|         FROM blocks`;
 | ||||
| 
 | ||||
|       if (interval !== null) { | ||||
|         query += ` WHERE blockTimestamp BETWEEN DATE_SUB(NOW(), INTERVAL ${interval}) AND NOW()`; | ||||
|       } | ||||
| 
 | ||||
|       query += ` GROUP BY UNIX_TIMESTAMP(blockTimestamp) DIV ${div}`; | ||||
| 
 | ||||
|       const [rows]: any = await connection.query(query); | ||||
|       connection.release(); | ||||
| 
 | ||||
|       return rows; | ||||
|     } catch (e) { | ||||
|       connection.release(); | ||||
|       logger.err('$getHistoricalBlockFees() error: ' + (e instanceof Error ? e.message : e)); | ||||
|       throw e; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| export default new BlocksRepository(); | ||||
|  | ||||
| @ -638,6 +638,22 @@ class Routes { | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public async $getHistoricalBlockFees(req: Request, res: Response) { | ||||
|     try { | ||||
|       const blockFees = await mining.$getHistoricalBlockFees(req.params.interval ?? null); | ||||
|       const oldestIndexedBlockTimestamp = await BlocksRepository.$oldestBlockTimestamp(); | ||||
|       res.header('Pragma', 'public'); | ||||
|       res.header('Cache-control', 'public'); | ||||
|       res.setHeader('Expires', new Date(Date.now() + 1000 * 300).toUTCString()); | ||||
|       res.json({ | ||||
|         oldestIndexedBlockTimestamp: oldestIndexedBlockTimestamp, | ||||
|         blockFees: blockFees, | ||||
|       }); | ||||
|     } catch (e) { | ||||
|       res.status(500).send(e instanceof Error ? e.message : e); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   public async getBlock(req: Request, res: Response) { | ||||
|     try { | ||||
|       const result = await bitcoinApi.$getBlock(req.params.hash); | ||||
|  | ||||
| @ -33,6 +33,7 @@ import { HashrateChartPoolsComponent } from './components/hashrates-chart-pools/ | ||||
| import { MiningStartComponent } from './components/mining-start/mining-start.component'; | ||||
| import { GraphsComponent } from './components/graphs/graphs.component'; | ||||
| import { BlocksList } from './components/blocks-list/blocks-list.component'; | ||||
| import { BlockFeesGraphComponent } from './components/block-fees-graph/block-fees-graph.component'; | ||||
| 
 | ||||
| let routes: Routes = [ | ||||
|   { | ||||
| @ -117,6 +118,10 @@ let routes: Routes = [ | ||||
|             path: 'mining/pools', | ||||
|             component: PoolRankingComponent, | ||||
|           }, | ||||
|           { | ||||
|             path: 'mining/block-fees', | ||||
|             component: BlockFeesGraphComponent, | ||||
|           } | ||||
|         ], | ||||
|       }, | ||||
|       { | ||||
| @ -211,18 +216,6 @@ let routes: Routes = [ | ||||
|                 path: 'blocks', | ||||
|                 component: BlocksList, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'hashrate', | ||||
|                 component: HashrateChartComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'hashrate/pools', | ||||
|                 component: HashrateChartPoolsComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'pools', | ||||
|                 component: PoolRankingComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'pool', | ||||
|                 children: [ | ||||
| @ -259,6 +252,10 @@ let routes: Routes = [ | ||||
|                 path: 'mining/pools', | ||||
|                 component: PoolRankingComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'mining/block-fees', | ||||
|                 component: BlockFeesGraphComponent, | ||||
|               } | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
| @ -347,18 +344,6 @@ let routes: Routes = [ | ||||
|                 path: 'blocks', | ||||
|                 component: BlocksList, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'hashrate', | ||||
|                 component: HashrateChartComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'hashrate/pools', | ||||
|                 component: HashrateChartPoolsComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'pools', | ||||
|                 component: PoolRankingComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'pool', | ||||
|                 children: [ | ||||
| @ -395,6 +380,10 @@ let routes: Routes = [ | ||||
|                 path: 'mining/pools', | ||||
|                 component: PoolRankingComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'mining/block-fees', | ||||
|                 component: BlockFeesGraphComponent, | ||||
|               } | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
| @ -507,19 +496,7 @@ if (browserWindowEnv && browserWindowEnv.BASE_MODULE === 'liquid') { | ||||
|           { | ||||
|             path: 'mempool', | ||||
|             component: StatisticsComponent, | ||||
|           }, | ||||
|           { | ||||
|             path: 'mining/hashrate-difficulty', | ||||
|             component: HashrateChartComponent, | ||||
|           }, | ||||
|           { | ||||
|             path: 'mining/pools-dominance', | ||||
|             component: HashrateChartPoolsComponent, | ||||
|           }, | ||||
|           { | ||||
|             path: 'mining/pools', | ||||
|             component: PoolRankingComponent, | ||||
|           }, | ||||
|           } | ||||
|         ] | ||||
|       }, | ||||
|       { | ||||
| @ -639,19 +616,7 @@ if (browserWindowEnv && browserWindowEnv.BASE_MODULE === 'liquid') { | ||||
|               { | ||||
|                 path: 'mempool', | ||||
|                 component: StatisticsComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'mining/hashrate-difficulty', | ||||
|                 component: HashrateChartComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'mining/pools-dominance', | ||||
|                 component: HashrateChartPoolsComponent, | ||||
|               }, | ||||
|               { | ||||
|                 path: 'mining/pools', | ||||
|                 component: PoolRankingComponent, | ||||
|               }, | ||||
|               } | ||||
|             ] | ||||
|           }, | ||||
|           { | ||||
|  | ||||
| @ -80,6 +80,7 @@ import { DifficultyAdjustmentsTable } from './components/difficulty-adjustments- | ||||
| import { BlocksList } from './components/blocks-list/blocks-list.component'; | ||||
| import { RewardStatsComponent } from './components/reward-stats/reward-stats.component'; | ||||
| import { DataCyDirective } from './data-cy.directive'; | ||||
| import { BlockFeesGraphComponent } from './components/block-fees-graph/block-fees-graph.component'; | ||||
| 
 | ||||
| @NgModule({ | ||||
|   declarations: [ | ||||
| @ -141,6 +142,7 @@ import { DataCyDirective } from './data-cy.directive'; | ||||
|     BlocksList, | ||||
|     DataCyDirective, | ||||
|     RewardStatsComponent, | ||||
|     BlockFeesGraphComponent, | ||||
|   ], | ||||
|   imports: [ | ||||
|     BrowserModule.withServerTransition({ appId: 'serverApp' }), | ||||
|  | ||||
| @ -0,0 +1,63 @@ | ||||
| <div class="full-container"> | ||||
|   <div class="card-header mb-0 mb-md-4"> | ||||
|     <span i18n="mining.block-fees">Block fees</span> | ||||
|     <form [formGroup]="radioGroupForm" class="formRadioGroup" *ngIf="(statsObservable$ | async) as stats"> | ||||
|       <div class="btn-group btn-group-toggle" ngbRadioGroup name="radioBasic" formControlName="dateSpan"> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 1"> | ||||
|           <input ngbButton type="radio" [value]="'24h'" fragment="24h"> 24h | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 3"> | ||||
|           <input ngbButton type="radio" [value]="'3d'" fragment="3d"> 3D | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 7"> | ||||
|           <input ngbButton type="radio" [value]="'1w'" fragment="1w"> 1W | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 30"> | ||||
|           <input ngbButton type="radio" [value]="'1m'" fragment="1m"> 1M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 90"> | ||||
|           <input ngbButton type="radio" [value]="'3m'" fragment="3m"> 3M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 180"> | ||||
|           <input ngbButton type="radio" [value]="'6m'" fragment="6m"> 6M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 365"> | ||||
|           <input ngbButton type="radio" [value]="'1y'" fragment="1y"> 1Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 730"> | ||||
|           <input ngbButton type="radio" [value]="'2y'" fragment="2y"> 2Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 1095"> | ||||
|           <input ngbButton type="radio" [value]="'3y'" fragment="3y"> 3Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay > 1095"> | ||||
|           <input ngbButton type="radio" [value]="'all'" fragment="all"> ALL | ||||
|         </label> | ||||
|       </div> | ||||
|     </form> | ||||
|   </div> | ||||
| 
 | ||||
|   <div class="chart" echarts [initOpts]="chartInitOptions" [options]="chartOptions"> | ||||
|   </div> | ||||
|   <div class="text-center loadingGraphs" *ngIf="isLoading"> | ||||
|     <div class="spinner-border text-light"></div> | ||||
|   </div> | ||||
| 
 | ||||
| </div> | ||||
| 
 | ||||
| <ng-template #loadingStats> | ||||
|   <div class="pool-distribution"> | ||||
|     <div class="item"> | ||||
|       <h5 class="card-title" i18n="mining.miners-luck">Hashrate</h5> | ||||
|       <p class="card-text"> | ||||
|         <span class="skeleton-loader skeleton-loader-big"></span> | ||||
|       </p> | ||||
|     </div> | ||||
|     <div class="item"> | ||||
|       <h5 class="card-title" i18n="master-page.blocks">Difficulty</h5> | ||||
|       <p class="card-text"> | ||||
|         <span class="skeleton-loader skeleton-loader-big"></span> | ||||
|       </p> | ||||
|     </div> | ||||
|   </div> | ||||
| </ng-template> | ||||
| @ -0,0 +1,135 @@ | ||||
| .card-header { | ||||
|   border-bottom: 0; | ||||
|   font-size: 18px; | ||||
|   @media (min-width: 465px) { | ||||
|     font-size: 20px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .main-title { | ||||
|   position: relative; | ||||
|   color: #ffffff91; | ||||
|   margin-top: -13px; | ||||
|   font-size: 10px; | ||||
|   text-transform: uppercase; | ||||
|   font-weight: 500; | ||||
|   text-align: center; | ||||
|   padding-bottom: 3px; | ||||
| } | ||||
| 
 | ||||
| .full-container { | ||||
|   padding: 0px 15px; | ||||
|   width: 100%; | ||||
|   min-height: 500px; | ||||
|   height: calc(100% - 150px); | ||||
|   @media (max-width: 992px) { | ||||
|     height: 100%; | ||||
|     padding-bottom: 100px; | ||||
|   }; | ||||
| } | ||||
| 
 | ||||
| .chart { | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   padding-bottom: 20px; | ||||
|   padding-right: 10px; | ||||
|   @media (max-width: 992px) { | ||||
|     padding-bottom: 25px; | ||||
|   } | ||||
|   @media (max-width: 829px) { | ||||
|     padding-bottom: 50px; | ||||
|   } | ||||
|   @media (max-width: 767px) { | ||||
|     padding-bottom: 25px; | ||||
|   } | ||||
|   @media (max-width: 629px) { | ||||
|     padding-bottom: 55px; | ||||
|   } | ||||
|   @media (max-width: 567px) { | ||||
|     padding-bottom: 55px; | ||||
|   } | ||||
| } | ||||
| .chart-widget { | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   max-height: 270px; | ||||
| } | ||||
| 
 | ||||
| .formRadioGroup { | ||||
|   margin-top: 6px; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   @media (min-width: 1130px) { | ||||
|     position: relative; | ||||
|     top: -65px; | ||||
|   } | ||||
|   @media (min-width: 830px) and (max-width: 1130px) { | ||||
|     position: relative; | ||||
|     top: 0px; | ||||
|   } | ||||
|   @media (min-width: 830px) { | ||||
|     flex-direction: row; | ||||
|     float: right; | ||||
|     margin-top: 0px; | ||||
|   } | ||||
|   .btn-sm { | ||||
|     font-size: 9px; | ||||
|     @media (min-width: 830px) { | ||||
|       font-size: 14px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .pool-distribution { | ||||
|   min-height: 56px; | ||||
|   display: block; | ||||
|   @media (min-width: 485px) { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|   } | ||||
|   h5 { | ||||
|     margin-bottom: 10px; | ||||
|   } | ||||
|   .item { | ||||
|     width: 50%; | ||||
|     display: inline-block; | ||||
|     margin: 0px auto 20px; | ||||
|     &:nth-child(2) { | ||||
|       order: 2; | ||||
|       @media (min-width: 485px) { | ||||
|         order: 3; | ||||
|       } | ||||
|     } | ||||
|     &:nth-child(3) { | ||||
|       order: 3; | ||||
|       @media (min-width: 485px) { | ||||
|         order: 2; | ||||
|         display: block; | ||||
|       } | ||||
|       @media (min-width: 768px) { | ||||
|         display: none; | ||||
|       } | ||||
|       @media (min-width: 992px) { | ||||
|         display: block; | ||||
|       } | ||||
|     } | ||||
|     .card-title { | ||||
|       font-size: 1rem; | ||||
|       color: #4a68b9; | ||||
|     } | ||||
|     .card-text { | ||||
|       font-size: 18px; | ||||
|       span { | ||||
|         color: #ffffff66; | ||||
|         font-size: 12px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .skeleton-loader { | ||||
|   width: 100%; | ||||
|   display: block; | ||||
|   max-width: 80px; | ||||
|   margin: 15px auto 3px; | ||||
| } | ||||
| @ -0,0 +1,201 @@ | ||||
| import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Inject, Input, LOCALE_ID, OnInit } from '@angular/core'; | ||||
| import { EChartsOption, graphic } from 'echarts'; | ||||
| import { Observable } from 'rxjs'; | ||||
| import { map, share, startWith, switchMap, tap } from 'rxjs/operators'; | ||||
| import { ApiService } from 'src/app/services/api.service'; | ||||
| import { SeoService } from 'src/app/services/seo.service'; | ||||
| import { formatNumber } from '@angular/common'; | ||||
| import { FormBuilder, FormGroup } from '@angular/forms'; | ||||
| import { formatterXAxisLabel } from 'src/app/shared/graphs.utils'; | ||||
| import { StorageService } from 'src/app/services/storage.service'; | ||||
| import { MiningService } from 'src/app/services/mining.service'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-block-fees-graph', | ||||
|   templateUrl: './block-fees-graph.component.html', | ||||
|   styleUrls: ['./block-fees-graph.component.scss'], | ||||
|   styles: [` | ||||
|     .loadingGraphs { | ||||
|       position: absolute; | ||||
|       top: 50%; | ||||
|       left: calc(50% - 15px); | ||||
|       z-index: 100; | ||||
|     } | ||||
|   `],
 | ||||
|   changeDetection: ChangeDetectionStrategy.OnPush, | ||||
| }) | ||||
| export class BlockFeesGraphComponent implements OnInit { | ||||
|   @Input() tableOnly = false; | ||||
|   @Input() right: number | string = 45; | ||||
|   @Input() left: number | string = 75; | ||||
| 
 | ||||
|   miningWindowPreference: string; | ||||
|   radioGroupForm: FormGroup; | ||||
| 
 | ||||
|   chartOptions: EChartsOption = {}; | ||||
|   chartInitOptions = { | ||||
|     renderer: 'svg', | ||||
|   }; | ||||
| 
 | ||||
|   statsObservable$: Observable<any>; | ||||
|   isLoading = true; | ||||
|   formatNumber = formatNumber; | ||||
|   timespan = ''; | ||||
| 
 | ||||
|   constructor( | ||||
|     @Inject(LOCALE_ID) public locale: string, | ||||
|     private seoService: SeoService, | ||||
|     private apiService: ApiService, | ||||
|     private formBuilder: FormBuilder, | ||||
|     private storageService: StorageService, | ||||
|     private miningService: MiningService | ||||
|   ) { | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: '1y' }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue('1y'); | ||||
|   } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     this.seoService.setTitle($localize`:@@mining.block-fees:Block Fees`); | ||||
|     this.miningWindowPreference = this.miningService.getDefaultTimespan('24h'); | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: this.miningWindowPreference }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue(this.miningWindowPreference); | ||||
| 
 | ||||
|     this.statsObservable$ = this.radioGroupForm.get('dateSpan').valueChanges | ||||
|       .pipe( | ||||
|         startWith(this.miningWindowPreference), | ||||
|         switchMap((timespan) => { | ||||
|           this.storageService.setValue('miningWindowPreference', timespan); | ||||
|           this.timespan = timespan; | ||||
|           this.isLoading = true; | ||||
|           return this.apiService.getHistoricalBlockFees$(timespan) | ||||
|             .pipe( | ||||
|               tap((data: any) => { | ||||
|                 this.prepareChartOptions({ | ||||
|                   blockFees: data.blockFees.map(val => [val.timestamp * 1000, val.avg_fees / 100000000]), | ||||
|                 }); | ||||
|                 this.isLoading = false; | ||||
|               }), | ||||
|               map((data: any) => { | ||||
|                 const availableTimespanDay = ( | ||||
|                   (new Date().getTime() / 1000) - (data.oldestIndexedBlockTimestamp) | ||||
|                 ) / 3600 / 24; | ||||
| 
 | ||||
|                 return { | ||||
|                   availableTimespanDay: availableTimespanDay, | ||||
|                 }; | ||||
|               }), | ||||
|             ); | ||||
|         }), | ||||
|         share() | ||||
|       ); | ||||
|   } | ||||
| 
 | ||||
|   prepareChartOptions(data) { | ||||
|     this.chartOptions = { | ||||
|       animation: false, | ||||
|       color: [ | ||||
|         new graphic.LinearGradient(0, 0, 0, 0.65, [ | ||||
|           { offset: 0, color: '#F4511E' }, | ||||
|           { offset: 0.25, color: '#FB8C00' }, | ||||
|           { offset: 0.5, color: '#FFB300' }, | ||||
|           { offset: 0.75, color: '#FDD835' }, | ||||
|           { offset: 1, color: '#7CB342' } | ||||
|         ]), | ||||
|       ], | ||||
|       grid: { | ||||
|         top: 30, | ||||
|         bottom: 80, | ||||
|         right: this.right, | ||||
|         left: this.left, | ||||
|       }, | ||||
|       tooltip: { | ||||
|         show: !this.isMobile(), | ||||
|         trigger: 'axis', | ||||
|         axisPointer: { | ||||
|           type: 'line' | ||||
|         }, | ||||
|         backgroundColor: 'rgba(17, 19, 31, 1)', | ||||
|         borderRadius: 4, | ||||
|         shadowColor: 'rgba(0, 0, 0, 0.5)', | ||||
|         textStyle: { | ||||
|           color: '#b1b1b1', | ||||
|           align: 'left', | ||||
|         }, | ||||
|         borderColor: '#000', | ||||
|         formatter: (ticks) => { | ||||
|           const tick = ticks[0]; | ||||
|           const feesString = `${tick.marker} ${tick.seriesName}: ${formatNumber(tick.data[1], this.locale, '1.3-3')} BTC`; | ||||
|           return ` | ||||
|             <b style="color: white; margin-left: 18px">${tick.axisValueLabel}</b><br> | ||||
|             <span>${feesString}</span> | ||||
|           `;
 | ||||
|         } | ||||
|       }, | ||||
|       xAxis: { | ||||
|         name: formatterXAxisLabel(this.locale, this.timespan), | ||||
|         nameLocation: 'middle', | ||||
|         nameTextStyle: { | ||||
|           padding: [10, 0, 0, 0], | ||||
|         }, | ||||
|         type: 'time', | ||||
|         splitNumber: this.isMobile() ? 5 : 10, | ||||
|       }, | ||||
|       yAxis: [ | ||||
|         { | ||||
|           type: 'value', | ||||
|           axisLabel: { | ||||
|             color: 'rgb(110, 112, 121)', | ||||
|             formatter: (val) => { | ||||
|               return `${val} BTC`; | ||||
|             } | ||||
|           }, | ||||
|           splitLine: { | ||||
|             show: false, | ||||
|           } | ||||
|         }, | ||||
|       ], | ||||
|       series: [ | ||||
|         { | ||||
|           zlevel: 0, | ||||
|           name: 'Fees', | ||||
|           showSymbol: false, | ||||
|           symbol: 'none', | ||||
|           data: data.blockFees, | ||||
|           type: 'line', | ||||
|           lineStyle: { | ||||
|             width: 2, | ||||
|           }, | ||||
|         }, | ||||
|       ], | ||||
|       dataZoom: [{ | ||||
|         type: 'inside', | ||||
|         realtime: true, | ||||
|         zoomLock: true, | ||||
|         maxSpan: 100, | ||||
|         minSpan: 10, | ||||
|         moveOnMouseMove: false, | ||||
|       }, { | ||||
|         showDetail: false, | ||||
|         show: true, | ||||
|         type: 'slider', | ||||
|         brushSelect: false, | ||||
|         realtime: true, | ||||
|         left: 20, | ||||
|         right: 15, | ||||
|         selectedDataBackground: { | ||||
|           lineStyle: { | ||||
|             color: '#fff', | ||||
|             opacity: 0.45, | ||||
|           }, | ||||
|           areaStyle: { | ||||
|             opacity: 0, | ||||
|           } | ||||
|         }, | ||||
|       }], | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   isMobile() { | ||||
|     return (window.innerWidth <= 767.98); | ||||
|   } | ||||
| } | ||||
| @ -1,25 +1,27 @@ | ||||
| <ul ngbNav #nav="ngbNav" class="nav-pills mb-3" style="padding: 0px 35px" *ngIf="stateService.env.MINING_DASHBOARD"> | ||||
|   <div class="d-inline-flex flex-wrap menu"> | ||||
|     <li ngbNavItem class="menu-li"> | ||||
|       <a routerLinkActive="active" [routerLink]="['/graphs/mempool' | relativeUrl]" ngbNavLink>Mempool</a> | ||||
|     </li> | ||||
|     <li ngbNavItem class="menu-li"> | ||||
|       <a routerLinkActive="active" [routerLink]="['/graphs/mining/pools' | relativeUrl]" ngbNavLink i18n="mining.pools"> | ||||
| <div class="mb-3 d-flex menu" style="padding: 0px 35px;"> | ||||
|   <a routerLinkActive="active" class="btn btn-primary w-50 mr-1" | ||||
|     [routerLink]="['/graphs/mempool' | relativeUrl]">Mempool</a> | ||||
|   <div ngbDropdown *ngIf="stateService.env.MINING_DASHBOARD" class="w-50"> | ||||
|     <button class="btn btn-primary w-100" id="dropdownBasic1" ngbDropdownToggle i18n="mining">Mining</button> | ||||
|     <div ngbDropdownMenu aria-labelledby="dropdownBasic1"> | ||||
|       <a class="dropdown-item" routerLinkActive="active" [routerLink]="['/graphs/mining/pools' | relativeUrl]" | ||||
|         i18n="mining.pools"> | ||||
|         Pools ranking | ||||
|       </a> | ||||
|     </li> | ||||
|     <li ngbNavItem class="menu-li"> | ||||
|       <a routerLinkActive="active" [routerLink]="['/graphs/mining/pools-dominance' | relativeUrl]" ngbNavLink i18n="mining.pools-dominance"> | ||||
|       <a class="dropdown-item" routerLinkActive="active" [routerLink]="['/graphs/mining/pools-dominance' | relativeUrl]" | ||||
|         i18n="mining.pools-dominance"> | ||||
|         Pools dominance | ||||
|       </a> | ||||
|     </li> | ||||
|     <li ngbNavItem class="menu-li"> | ||||
|       <a routerLinkActive="active" [routerLink]="['/graphs/mining/hashrate-difficulty' | relativeUrl]" ngbNavLink | ||||
|         i18n="mining.hashrate-difficulty"> | ||||
|       <a class="dropdown-item" routerLinkActive="active" | ||||
|         [routerLink]="['/graphs/mining/hashrate-difficulty' | relativeUrl]" i18n="mining.hashrate-difficulty"> | ||||
|         Hashrate & Difficulty | ||||
|       </a> | ||||
|     </li> | ||||
|       <a class="dropdown-item" routerLinkActive="active" | ||||
|         [routerLink]="['/graphs/mining/block-fees' | relativeUrl]" i18n="mining.block-fees"> | ||||
|         Block Fees | ||||
|       </a> | ||||
|     </div> | ||||
| </ul> | ||||
|   </div> | ||||
| </div> | ||||
| 
 | ||||
| <router-outlet></router-outlet> | ||||
| @ -1,9 +1,6 @@ | ||||
| .menu { | ||||
|   flex-grow: 1; | ||||
|   max-width: 600px; | ||||
| } | ||||
| 
 | ||||
| .menu-li { | ||||
|   flex-grow: 1; | ||||
|   text-align: center; | ||||
|   @media (min-width: 576px) { | ||||
|     max-width: 400px; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @ -19,26 +19,29 @@ | ||||
|   </div> | ||||
| 
 | ||||
|   <div class="card-header mb-0 mb-md-4" [style]="widget ? 'display:none' : ''"> | ||||
|     <span i18n="mining.mining-pool-share">Hashrate & Difficulty</span> | ||||
|     <form [formGroup]="radioGroupForm" class="formRadioGroup" *ngIf="(hashrateObservable$ | async) as hashrates"> | ||||
|     <span i18n="mining.hashrate-difficulty">Hashrate & Difficulty</span> | ||||
|     <form [formGroup]="radioGroupForm" class="formRadioGroup" *ngIf="(hashrateObservable$ | async) as stats"> | ||||
|       <div class="btn-group btn-group-toggle" ngbRadioGroup name="radioBasic" formControlName="dateSpan"> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 90"> | ||||
|           <input ngbButton type="radio" [value]="'3m'"> 3M | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 30"> | ||||
|           <input ngbButton type="radio" [value]="'1m'" fragment="1m"> 1M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 180"> | ||||
|           <input ngbButton type="radio" [value]="'6m'"> 6M | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 90"> | ||||
|           <input ngbButton type="radio" [value]="'3m'" fragment="3m"> 3M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 365"> | ||||
|           <input ngbButton type="radio" [value]="'1y'"> 1Y | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 180"> | ||||
|           <input ngbButton type="radio" [value]="'6m'" fragment="6m"> 6M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 730"> | ||||
|           <input ngbButton type="radio" [value]="'2y'"> 2Y | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 365"> | ||||
|           <input ngbButton type="radio" [value]="'1y'" fragment="1y"> 1Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 1095"> | ||||
|           <input ngbButton type="radio" [value]="'3y'"> 3Y | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 730"> | ||||
|           <input ngbButton type="radio" [value]="'2y'" fragment="2y"> 2Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm"> | ||||
|           <input ngbButton type="radio" [value]="'all'"> ALL | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 1095"> | ||||
|           <input ngbButton type="radio" [value]="'3y'" fragment="3y"> 3Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay > 1095"> | ||||
|           <input ngbButton type="radio" [value]="'all'" fragment="all"> ALL | ||||
|         </label> | ||||
|       </div> | ||||
|     </form> | ||||
|  | ||||
| @ -7,6 +7,8 @@ import { SeoService } from 'src/app/services/seo.service'; | ||||
| import { formatNumber } from '@angular/common'; | ||||
| import { FormBuilder, FormGroup } from '@angular/forms'; | ||||
| import { selectPowerOfTen } from 'src/app/bitcoin.utils'; | ||||
| import { StorageService } from 'src/app/services/storage.service'; | ||||
| import { MiningService } from 'src/app/services/mining.service'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-hashrate-chart', | ||||
| @ -28,6 +30,7 @@ export class HashrateChartComponent implements OnInit { | ||||
|   @Input() right: number | string = 45; | ||||
|   @Input() left: number | string = 75; | ||||
| 
 | ||||
|   miningWindowPreference: string; | ||||
|   radioGroupForm: FormGroup; | ||||
| 
 | ||||
|   chartOptions: EChartsOption = {}; | ||||
| @ -47,20 +50,32 @@ export class HashrateChartComponent implements OnInit { | ||||
|     private apiService: ApiService, | ||||
|     private formBuilder: FormBuilder, | ||||
|     private cd: ChangeDetectorRef, | ||||
|     private storageService: StorageService, | ||||
|     private miningService: MiningService | ||||
|   ) { | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: '1y' }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue('1y'); | ||||
|   } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     if (!this.widget) { | ||||
|     let firstRun = true; | ||||
| 
 | ||||
|     if (this.widget) { | ||||
|       this.miningWindowPreference = '1y'; | ||||
|     } else { | ||||
|       this.seoService.setTitle($localize`:@@mining.hashrate-difficulty:Hashrate and Difficulty`); | ||||
|       this.miningWindowPreference = this.miningService.getDefaultTimespan('1m'); | ||||
|     } | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: this.miningWindowPreference }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue(this.miningWindowPreference); | ||||
| 
 | ||||
|     this.hashrateObservable$ = this.radioGroupForm.get('dateSpan').valueChanges | ||||
|       .pipe( | ||||
|         startWith('1y'), | ||||
|         startWith(this.miningWindowPreference), | ||||
|         switchMap((timespan) => { | ||||
|           if (!this.widget && !firstRun) { | ||||
|             this.storageService.setValue('miningWindowPreference', timespan); | ||||
|           } | ||||
|           firstRun = false; | ||||
|           this.miningWindowPreference = timespan; | ||||
|           this.isLoading = true; | ||||
|           return this.apiService.getHistoricalHashrate$(timespan) | ||||
|             .pipe( | ||||
|  | ||||
| @ -1,32 +1,35 @@ | ||||
| <div [class]="widget === false ? 'full-container' : ''"> | ||||
| <div class="full-container"> | ||||
| 
 | ||||
|   <div class="card-header  mb-0 mb-md-4" [style]="widget ? 'display:none' : ''"> | ||||
|     <span *ngIf="!widget" i18n="mining.pools-dominance">Mining pools dominance</span> | ||||
|     <form [formGroup]="radioGroupForm" class="formRadioGroup" *ngIf="(hashrateObservable$ | async) as hashrates"> | ||||
|   <div class="card-header  mb-0 mb-md-4"> | ||||
|     <span i18n="mining.pools-dominance">Mining pools dominance</span> | ||||
|     <form [formGroup]="radioGroupForm" class="formRadioGroup" *ngIf="(hashrateObservable$ | async) as stats"> | ||||
|       <div class="btn-group btn-group-toggle" ngbRadioGroup name="radioBasic" formControlName="dateSpan"> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 90"> | ||||
|           <input ngbButton type="radio" [value]="'3m'"> 3M | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 30"> | ||||
|           <input ngbButton type="radio" [value]="'1m'" fragment="1m"> 1M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 180"> | ||||
|           <input ngbButton type="radio" [value]="'6m'"> 6M | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 90"> | ||||
|           <input ngbButton type="radio" [value]="'3m'" fragment="3m"> 3M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 365"> | ||||
|           <input ngbButton type="radio" [value]="'1y'"> 1Y | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 180"> | ||||
|           <input ngbButton type="radio" [value]="'6m'" fragment="6m"> 6M | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 730"> | ||||
|           <input ngbButton type="radio" [value]="'2y'"> 2Y | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 365"> | ||||
|           <input ngbButton type="radio" [value]="'1y'" fragment="1y"> 1Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="hashrates.availableTimespanDay >= 1095"> | ||||
|           <input ngbButton type="radio" [value]="'3y'"> 3Y | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 730"> | ||||
|           <input ngbButton type="radio" [value]="'2y'" fragment="2y"> 2Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm"> | ||||
|           <input ngbButton type="radio" [value]="'all'"> ALL | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay >= 1095"> | ||||
|           <input ngbButton type="radio" [value]="'3y'" fragment="3y"> 3Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="stats.availableTimespanDay > 1095"> | ||||
|           <input ngbButton type="radio" [value]="'all'" fragment="all"> ALL | ||||
|         </label> | ||||
|       </div> | ||||
|     </form> | ||||
|   </div> | ||||
| 
 | ||||
|   <div [class]="!widget ? 'chart' : 'chart-widget'" | ||||
|   <div class="chart" | ||||
|     echarts [initOpts]="chartInitOptions" [options]="chartOptions"></div> | ||||
|   <div class="text-center loadingGraphs" *ngIf="isLoading"> | ||||
|     <div class="spinner-border text-light"></div> | ||||
|  | ||||
| @ -6,6 +6,8 @@ import { ApiService } from 'src/app/services/api.service'; | ||||
| import { SeoService } from 'src/app/services/seo.service'; | ||||
| import { FormBuilder, FormGroup } from '@angular/forms'; | ||||
| import { poolsColor } from 'src/app/app.constants'; | ||||
| import { StorageService } from 'src/app/services/storage.service'; | ||||
| import { MiningService } from 'src/app/services/mining.service'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-hashrate-chart-pools', | ||||
| @ -22,10 +24,10 @@ import { poolsColor } from 'src/app/app.constants'; | ||||
|   changeDetection: ChangeDetectionStrategy.OnPush, | ||||
| }) | ||||
| export class HashrateChartPoolsComponent implements OnInit { | ||||
|   @Input() widget = false; | ||||
|   @Input() right: number | string = 45; | ||||
|   @Input() left: number | string = 25; | ||||
| 
 | ||||
|   miningWindowPreference: string; | ||||
|   radioGroupForm: FormGroup; | ||||
| 
 | ||||
|   chartOptions: EChartsOption = {}; | ||||
| @ -44,20 +46,29 @@ export class HashrateChartPoolsComponent implements OnInit { | ||||
|     private apiService: ApiService, | ||||
|     private formBuilder: FormBuilder, | ||||
|     private cd: ChangeDetectorRef, | ||||
|     private storageService: StorageService, | ||||
|     private miningService: MiningService | ||||
|   ) { | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: '1y' }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue('1y'); | ||||
|   } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     if (!this.widget) { | ||||
|     let firstRun = true; | ||||
| 
 | ||||
|     this.seoService.setTitle($localize`:@@mining.pools-historical-dominance:Pools Historical Dominance`); | ||||
|     } | ||||
|     this.miningWindowPreference = this.miningService.getDefaultTimespan('1m'); | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: this.miningWindowPreference }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue(this.miningWindowPreference); | ||||
| 
 | ||||
|     this.hashrateObservable$ = this.radioGroupForm.get('dateSpan').valueChanges | ||||
|       .pipe( | ||||
|         startWith('1y'), | ||||
|         startWith(this.miningWindowPreference), | ||||
|         switchMap((timespan) => { | ||||
|           if (!firstRun) { | ||||
|             this.storageService.setValue('miningWindowPreference', timespan); | ||||
|           } | ||||
|           firstRun = false; | ||||
|           this.isLoading = true; | ||||
|           return this.apiService.getHistoricalPoolsHashrate$(timespan) | ||||
|             .pipe( | ||||
| @ -157,11 +168,11 @@ export class HashrateChartPoolsComponent implements OnInit { | ||||
|       grid: { | ||||
|         right: this.right, | ||||
|         left: this.left, | ||||
|         bottom: this.widget ? 30 : 70, | ||||
|         top: this.widget || this.isMobile() ? 10 : 50, | ||||
|         bottom: 70, | ||||
|         top: this.isMobile() ? 10 : 50, | ||||
|       }, | ||||
|       tooltip: { | ||||
|         show: !this.isMobile() || !this.widget, | ||||
|         show: !this.isMobile(), | ||||
|         trigger: 'axis', | ||||
|         axisPointer: { | ||||
|           type: 'line' | ||||
| @ -188,9 +199,9 @@ export class HashrateChartPoolsComponent implements OnInit { | ||||
|       }, | ||||
|       xAxis: data.series.length === 0 ? undefined : { | ||||
|         type: 'time', | ||||
|         splitNumber: (this.isMobile() || this.widget) ? 5 : 10, | ||||
|         splitNumber: (this.isMobile()) ? 5 : 10, | ||||
|       }, | ||||
|       legend: (this.isMobile() || this.widget || data.series.length === 0) ? undefined : { | ||||
|       legend: (this.isMobile() || data.series.length === 0) ? undefined : { | ||||
|         data: data.legends | ||||
|       }, | ||||
|       yAxis: data.series.length === 0 ? undefined : { | ||||
| @ -207,7 +218,7 @@ export class HashrateChartPoolsComponent implements OnInit { | ||||
|         min: 0, | ||||
|       }, | ||||
|       series: data.series, | ||||
|       dataZoom: this.widget ? null : [{ | ||||
|       dataZoom: [{ | ||||
|         type: 'inside', | ||||
|         realtime: true, | ||||
|         zoomLock: true, | ||||
|  | ||||
| @ -55,7 +55,7 @@ | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="miningStats.availableTimespanDay >= 1095"> | ||||
|           <input ngbButton type="radio" [value]="'3y'" fragment="3y"> 3Y | ||||
|         </label> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm"> | ||||
|         <label ngbButtonLabel class="btn-primary btn-sm" *ngIf="miningStats.availableTimespanDay > 1095"> | ||||
|           <input ngbButton type="radio" [value]="'all'" fragment="all"> ALL | ||||
|         </label> | ||||
|       </div> | ||||
| @ -79,7 +79,7 @@ | ||||
|           <th class="d-none d-md-block" i18n="mining.rank">Rank</th> | ||||
|           <th class=""></th> | ||||
|           <th class="" i18n="mining.pool-name">Pool</th> | ||||
|           <th class="" *ngIf="this.poolsWindowPreference === '24h'" i18n="mining.hashrate">Hashrate</th> | ||||
|           <th class="" *ngIf="this.miningWindowPreference === '24h'" i18n="mining.hashrate">Hashrate</th> | ||||
|           <th class="" i18n="master-page.blocks">Blocks</th> | ||||
|           <th class="d-none d-md-block" i18n="mining.empty-blocks">Empty Blocks</th> | ||||
|         </tr> | ||||
| @ -90,7 +90,7 @@ | ||||
|           <td class="text-right"><img width="25" height="25" src="{{ pool.logo }}" | ||||
|               onError="this.src = './resources/mining-pools/default.svg'"></td> | ||||
|           <td class=""><a [routerLink]="[('/mining/pool/' + pool.slug) | relativeUrl]">{{ pool.name }}</a></td> | ||||
|           <td class="" *ngIf="this.poolsWindowPreference === '24h' && !isLoading">{{ pool.lastEstimatedHashrate }} {{ | ||||
|           <td class="" *ngIf="this.miningWindowPreference === '24h' && !isLoading">{{ pool.lastEstimatedHashrate }} {{ | ||||
|             miningStats.miningUnits.hashrateUnit }}</td> | ||||
|           <td class="">{{ pool['blockText'] }}</td> | ||||
|           <td class="d-none d-md-block">{{ pool.emptyBlocks }} ({{ pool.emptyBlockRatio }}%)</td> | ||||
| @ -99,7 +99,7 @@ | ||||
|           <td class="d-none d-md-block"></td> | ||||
|           <td class="text-right"></td> | ||||
|           <td class="" i18n="mining.all-miners"><b>All miners</b></td> | ||||
|           <td class="" *ngIf="this.poolsWindowPreference === '24h'"><b>{{ miningStats.lastEstimatedHashrate}} {{ | ||||
|           <td class="" *ngIf="this.miningWindowPreference === '24h'"><b>{{ miningStats.lastEstimatedHashrate}} {{ | ||||
|               miningStats.miningUnits.hashrateUnit }}</b></td> | ||||
|           <td class=""><b>{{ miningStats.blockCount }}</b></td> | ||||
|           <td class="d-none d-md-block"><b>{{ miningStats.totalEmptyBlock }} ({{ miningStats.totalEmptyBlockRatio | ||||
|  | ||||
| @ -19,9 +19,9 @@ import { RelativeUrlPipe } from 'src/app/shared/pipes/relative-url/relative-url. | ||||
|   changeDetection: ChangeDetectionStrategy.OnPush, | ||||
| }) | ||||
| export class PoolRankingComponent implements OnInit { | ||||
|   @Input() widget: boolean = false; | ||||
|   @Input() widget = false; | ||||
| 
 | ||||
|   poolsWindowPreference: string; | ||||
|   miningWindowPreference: string; | ||||
|   radioGroupForm: FormGroup; | ||||
| 
 | ||||
|   isLoading = true; | ||||
| @ -48,13 +48,13 @@ export class PoolRankingComponent implements OnInit { | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     if (this.widget) { | ||||
|       this.poolsWindowPreference = '1w'; | ||||
|       this.miningWindowPreference = '1w'; | ||||
|     } else { | ||||
|       this.seoService.setTitle($localize`:@@mining.mining-pools:Mining Pools`); | ||||
|       this.poolsWindowPreference = this.storageService.getValue('poolsWindowPreference') ? this.storageService.getValue('poolsWindowPreference') : '1w'; | ||||
|       this.miningWindowPreference = this.miningService.getDefaultTimespan('24h'); | ||||
|     } | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: this.poolsWindowPreference }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue(this.poolsWindowPreference); | ||||
|     this.radioGroupForm = this.formBuilder.group({ dateSpan: this.miningWindowPreference }); | ||||
|     this.radioGroupForm.controls.dateSpan.setValue(this.miningWindowPreference); | ||||
| 
 | ||||
|     // When...
 | ||||
|     this.miningStatsObservable$ = combineLatest([ | ||||
| @ -67,12 +67,12 @@ export class PoolRankingComponent implements OnInit { | ||||
|       // ...or we change the timespan
 | ||||
|       this.radioGroupForm.get('dateSpan').valueChanges | ||||
|         .pipe( | ||||
|           startWith(this.poolsWindowPreference), // (trigger when the page loads)
 | ||||
|           startWith(this.miningWindowPreference), // (trigger when the page loads)
 | ||||
|           tap((value) => { | ||||
|             if (!this.widget) { | ||||
|               this.storageService.setValue('poolsWindowPreference', value); | ||||
|               this.storageService.setValue('miningWindowPreference', value); | ||||
|             } | ||||
|             this.poolsWindowPreference = value; | ||||
|             this.miningWindowPreference = value; | ||||
|           }) | ||||
|         ) | ||||
|     ]) | ||||
| @ -80,7 +80,7 @@ export class PoolRankingComponent implements OnInit { | ||||
|       .pipe( | ||||
|         switchMap(() => { | ||||
|           this.isLoading = true; | ||||
|           return this.miningService.getMiningStats(this.poolsWindowPreference) | ||||
|           return this.miningService.getMiningStats(this.miningWindowPreference) | ||||
|             .pipe( | ||||
|               catchError((e) => of(this.getEmptyMiningStat())) | ||||
|             ); | ||||
| @ -150,7 +150,7 @@ export class PoolRankingComponent implements OnInit { | ||||
|           }, | ||||
|           borderColor: '#000', | ||||
|           formatter: () => { | ||||
|             if (this.poolsWindowPreference === '24h') { | ||||
|             if (this.miningWindowPreference === '24h') { | ||||
|               return `<b style="color: white">${pool.name} (${pool.share}%)</b><br>` + | ||||
|                 pool.lastEstimatedHashrate.toString() + ' PH/s' + | ||||
|                 `<br>` + pool.blockCount.toString() + ` blocks`; | ||||
| @ -186,7 +186,7 @@ export class PoolRankingComponent implements OnInit { | ||||
|         }, | ||||
|         borderColor: '#000', | ||||
|         formatter: () => { | ||||
|           if (this.poolsWindowPreference === '24h') { | ||||
|           if (this.miningWindowPreference === '24h') { | ||||
|             return `<b style="color: white">${'Other'} (${totalShareOther.toFixed(2)}%)</b><br>` + | ||||
|               totalEstimatedHashrateOther.toString() + ' PH/s' + | ||||
|               `<br>` + totalBlockOther.toString() + ` blocks`; | ||||
|  | ||||
| @ -168,6 +168,13 @@ export class ApiService { | ||||
|       ); | ||||
|   } | ||||
| 
 | ||||
|   getHistoricalBlockFees$(interval: string | undefined) : Observable<any> { | ||||
|     return this.httpClient.get<any[]>( | ||||
|       this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/blocks/fees` + | ||||
|       (interval !== undefined ? `/${interval}` : '') | ||||
|     ); | ||||
| } | ||||
| 
 | ||||
|   getRewardStats$(blockCount: number = 144): Observable<RewardStats> { | ||||
|     return this.httpClient.get<RewardStats>(this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/reward-stats/${blockCount}`); | ||||
|   } | ||||
|  | ||||
| @ -4,6 +4,7 @@ import { map } from 'rxjs/operators'; | ||||
| import { PoolsStats, SinglePoolStats } from '../interfaces/node-api.interface'; | ||||
| import { ApiService } from '../services/api.service'; | ||||
| import { StateService } from './state.service'; | ||||
| import { StorageService } from './storage.service'; | ||||
| 
 | ||||
| export interface MiningUnits { | ||||
|   hashrateDivider: number; | ||||
| @ -28,8 +29,12 @@ export class MiningService { | ||||
|   constructor( | ||||
|     private stateService: StateService, | ||||
|     private apiService: ApiService, | ||||
|     private storageService: StorageService, | ||||
|   ) { } | ||||
| 
 | ||||
|   /** | ||||
|    * Generate pool ranking stats | ||||
|    */ | ||||
|   public getMiningStats(interval: string): Observable<MiningStats> { | ||||
|     return this.apiService.listPools$(interval).pipe( | ||||
|       map(pools => this.generateMiningStats(pools)) | ||||
| @ -63,6 +68,20 @@ export class MiningService { | ||||
|     }; | ||||
|   } | ||||
| 
 | ||||
|   /** | ||||
|    * Get the default selection timespan, cap with `min` | ||||
|    */ | ||||
|   public getDefaultTimespan(min: string): string { | ||||
|     const timespans = [ | ||||
|       '24h', '3d', '1w', '1m', '3m', '6m', '1y', '2y', '3y', 'all' | ||||
|     ]; | ||||
|     const preference = this.storageService.getValue('miningWindowPreference') ?? '1w'; | ||||
|     if (timespans.indexOf(preference) < timespans.indexOf(min)) { | ||||
|       return min; | ||||
|     } | ||||
|     return preference; | ||||
|   } | ||||
| 
 | ||||
|   private generateMiningStats(stats: PoolsStats): MiningStats { | ||||
|     const miningUnits = this.getMiningUnits(); | ||||
|     const hashrateDivider = miningUnits.hashrateDivider; | ||||
|  | ||||
| @ -7,21 +7,21 @@ import { Router, ActivatedRoute } from '@angular/router'; | ||||
| export class StorageService { | ||||
|   constructor(private router: Router, private route: ActivatedRoute) { | ||||
|     this.setDefaultValueIfNeeded('graphWindowPreference', '2h'); | ||||
|     this.setDefaultValueIfNeeded('poolsWindowPreference', '1w'); | ||||
|     this.setDefaultValueIfNeeded('miningWindowPreference', '1w'); | ||||
|   } | ||||
| 
 | ||||
|   setDefaultValueIfNeeded(key: string, defaultValue: string) { | ||||
|     let graphWindowPreference: string = this.getValue(key); | ||||
|     const graphWindowPreference: string = this.getValue(key); | ||||
|     if (graphWindowPreference === null) { // First visit to mempool.space
 | ||||
|       if (this.router.url.includes('graphs') && key === 'graphWindowPreference' || | ||||
|         this.router.url.includes('pools') && key === 'poolsWindowPreference' | ||||
|         this.router.url.includes('pools') && key === 'miningWindowPreference' | ||||
|       ) { | ||||
|         this.setValue(key, this.route.snapshot.fragment ? this.route.snapshot.fragment : defaultValue); | ||||
|       } else { | ||||
|         this.setValue(key, defaultValue); | ||||
|       } | ||||
|     } else if (this.router.url.includes('graphs') && key === 'graphWindowPreference' || | ||||
|       this.router.url.includes('pools') && key === 'poolsWindowPreference' | ||||
|       this.router.url.includes('pools') && key === 'miningWindowPreference' | ||||
|     ) { | ||||
|       // Visit a different graphs#fragment from last visit
 | ||||
|       if (this.route.snapshot.fragment !== null && graphWindowPreference !== this.route.snapshot.fragment) { | ||||
|  | ||||
| @ -37,6 +37,16 @@ do for url in / \ | ||||
| 	'/api/v1/mining/hashrate/pools/all' \ | ||||
| 	'/api/v1/mining/reward-stats/144' \ | ||||
| 	'/api/v1/mining/blocks-extras' \ | ||||
| 	'/api/v1/mining/blocks/fees/24h' \ | ||||
| 	'/api/v1/mining/blocks/fees/3d' \ | ||||
| 	'/api/v1/mining/blocks/fees/1w' \ | ||||
| 	'/api/v1/mining/blocks/fees/1m' \ | ||||
| 	'/api/v1/mining/blocks/fees/3m' \ | ||||
| 	'/api/v1/mining/blocks/fees/6m' \ | ||||
| 	'/api/v1/mining/blocks/fees/1y' \ | ||||
| 	'/api/v1/mining/blocks/fees/2y' \ | ||||
| 	'/api/v1/mining/blocks/fees/3y' \ | ||||
| 	'/api/v1/mining/blocks/fees/all' \ | ||||
| 
 | ||||
| 	do | ||||
| 		curl -s "https://${hostname}${url}" >/dev/null | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user