Add balance graph to address page
This commit is contained in:
		
							parent
							
								
									48d852fd49
								
							
						
					
					
						commit
						82f1fa5110
					
				@ -121,8 +121,10 @@ class BitcoinRoutes {
 | 
				
			|||||||
          .get(config.MEMPOOL.API_URL_PREFIX + 'block-height/:height', this.getBlockHeight)
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'block-height/:height', this.getBlockHeight)
 | 
				
			||||||
          .get(config.MEMPOOL.API_URL_PREFIX + 'address/:address', this.getAddress)
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'address/:address', this.getAddress)
 | 
				
			||||||
          .get(config.MEMPOOL.API_URL_PREFIX + 'address/:address/txs', this.getAddressTransactions)
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'address/:address/txs', this.getAddressTransactions)
 | 
				
			||||||
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'address/:address/txs/summary', this.getAddressTransactionSummary)
 | 
				
			||||||
          .get(config.MEMPOOL.API_URL_PREFIX + 'scripthash/:scripthash', this.getScriptHash)
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'scripthash/:scripthash', this.getScriptHash)
 | 
				
			||||||
          .get(config.MEMPOOL.API_URL_PREFIX + 'scripthash/:scripthash/txs', this.getScriptHashTransactions)
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'scripthash/:scripthash/txs', this.getScriptHashTransactions)
 | 
				
			||||||
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'scripthash/:scripthash/txs/summary', this.getScriptHashTransactionSummary)
 | 
				
			||||||
          .get(config.MEMPOOL.API_URL_PREFIX + 'address-prefix/:prefix', this.getAddressPrefix)
 | 
					          .get(config.MEMPOOL.API_URL_PREFIX + 'address-prefix/:prefix', this.getAddressPrefix)
 | 
				
			||||||
          ;
 | 
					          ;
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
@ -566,6 +568,13 @@ class BitcoinRoutes {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  private async getAddressTransactionSummary(req: Request, res: Response): Promise<void> {
 | 
				
			||||||
 | 
					    if (config.MEMPOOL.BACKEND !== 'esplora') {
 | 
				
			||||||
 | 
					      res.status(405).send('Address summary lookups require mempool/electrs backend.');
 | 
				
			||||||
 | 
					      return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  private async getScriptHash(req: Request, res: Response) {
 | 
					  private async getScriptHash(req: Request, res: Response) {
 | 
				
			||||||
    if (config.MEMPOOL.BACKEND === 'none') {
 | 
					    if (config.MEMPOOL.BACKEND === 'none') {
 | 
				
			||||||
      res.status(405).send('Address lookups cannot be used with bitcoind as backend.');
 | 
					      res.status(405).send('Address lookups cannot be used with bitcoind as backend.');
 | 
				
			||||||
@ -609,6 +618,13 @@ class BitcoinRoutes {
 | 
				
			|||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  private async getScriptHashTransactionSummary(req: Request, res: Response): Promise<void> {
 | 
				
			||||||
 | 
					    if (config.MEMPOOL.BACKEND !== 'esplora') {
 | 
				
			||||||
 | 
					      res.status(405).send('Scripthash summary lookups require mempool/electrs backend.');
 | 
				
			||||||
 | 
					      return;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  private async getAddressPrefix(req: Request, res: Response) {
 | 
					  private async getAddressPrefix(req: Request, res: Response) {
 | 
				
			||||||
    try {
 | 
					    try {
 | 
				
			||||||
      const blockHash = await bitcoinApi.$getAddressPrefix(req.params.prefix);
 | 
					      const blockHash = await bitcoinApi.$getAddressPrefix(req.params.prefix);
 | 
				
			||||||
 | 
				
			|||||||
@ -0,0 +1,23 @@
 | 
				
			|||||||
 | 
					<app-indexing-progress></app-indexing-progress>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					<div class="full-container">
 | 
				
			||||||
 | 
					  <div class="card-header mb-0 mb-md-2">
 | 
				
			||||||
 | 
					    <div class="d-flex d-md-block align-items-baseline">
 | 
				
			||||||
 | 
					      <span i18n="address.balance-history">Balance History</span>
 | 
				
			||||||
 | 
					    </div>  
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  <ng-container *ngIf="!error">
 | 
				
			||||||
 | 
					    <div class="chart" echarts [initOpts]="chartInitOptions" [options]="chartOptions"
 | 
				
			||||||
 | 
					      (chartInit)="onChartInit($event)">
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					    <div class="text-center loadingGraphs" *ngIf="isLoading">
 | 
				
			||||||
 | 
					      <div class="spinner-border text-light"></div>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					  </ng-container>
 | 
				
			||||||
 | 
					  <ng-container *ngIf="error">
 | 
				
			||||||
 | 
					    <div class="error-wrapper">
 | 
				
			||||||
 | 
					      <p class="error">{{ error }}</p>
 | 
				
			||||||
 | 
					    </div>
 | 
				
			||||||
 | 
					  </ng-container>
 | 
				
			||||||
 | 
					</div>
 | 
				
			||||||
@ -0,0 +1,75 @@
 | 
				
			|||||||
 | 
					.card-header {
 | 
				
			||||||
 | 
					  border-bottom: 0;
 | 
				
			||||||
 | 
					  font-size: 18px;
 | 
				
			||||||
 | 
					  @media (min-width: 465px) {
 | 
				
			||||||
 | 
					    font-size: 20px;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					  @media (min-width: 992px) {
 | 
				
			||||||
 | 
					    height: 40px;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.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 {
 | 
				
			||||||
 | 
					  display: flex;
 | 
				
			||||||
 | 
					  flex-direction: column;
 | 
				
			||||||
 | 
					  padding: 0px;
 | 
				
			||||||
 | 
					  width: 100%;
 | 
				
			||||||
 | 
					  height: 400px;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.error-wrapper {
 | 
				
			||||||
 | 
					  display: flex;
 | 
				
			||||||
 | 
					  flex-direction: column;
 | 
				
			||||||
 | 
					  width: 100%;
 | 
				
			||||||
 | 
					  height: 100%;
 | 
				
			||||||
 | 
					  align-items: center;
 | 
				
			||||||
 | 
					  justify-content: center;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  font-size: 15px;
 | 
				
			||||||
 | 
					  color: grey;
 | 
				
			||||||
 | 
					  font-weight: bold;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.chart {
 | 
				
			||||||
 | 
					  display: flex;
 | 
				
			||||||
 | 
					  flex: 1;
 | 
				
			||||||
 | 
					  width: 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;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					.disabled {
 | 
				
			||||||
 | 
					  pointer-events: none;
 | 
				
			||||||
 | 
					  opacity: 0.5;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -0,0 +1,180 @@
 | 
				
			|||||||
 | 
					import { ChangeDetectionStrategy, ChangeDetectorRef, Component, Inject, Input, LOCALE_ID, OnChanges, OnInit, SimpleChanges } from '@angular/core';
 | 
				
			||||||
 | 
					import { echarts, EChartsOption } from '../../graphs/echarts';
 | 
				
			||||||
 | 
					import { of } from 'rxjs';
 | 
				
			||||||
 | 
					import { catchError } from 'rxjs/operators';
 | 
				
			||||||
 | 
					import { ChainStats } from '../../interfaces/electrs.interface';
 | 
				
			||||||
 | 
					import { ElectrsApiService } from '../../services/electrs-api.service';
 | 
				
			||||||
 | 
					import { AmountShortenerPipe } from '../../shared/pipes/amount-shortener.pipe';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					@Component({
 | 
				
			||||||
 | 
					  selector: 'app-address-graph',
 | 
				
			||||||
 | 
					  templateUrl: './address-graph.component.html',
 | 
				
			||||||
 | 
					  styleUrls: ['./address-graph.component.scss'],
 | 
				
			||||||
 | 
					  styles: [`
 | 
				
			||||||
 | 
					    .loadingGraphs {
 | 
				
			||||||
 | 
					      position: absolute;
 | 
				
			||||||
 | 
					      top: 50%;
 | 
				
			||||||
 | 
					      left: calc(50% - 15px);
 | 
				
			||||||
 | 
					      z-index: 100;
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					  `],
 | 
				
			||||||
 | 
					  changeDetection: ChangeDetectionStrategy.OnPush,
 | 
				
			||||||
 | 
					})
 | 
				
			||||||
 | 
					export class AddressGraphComponent implements OnInit, OnChanges {
 | 
				
			||||||
 | 
					  @Input() address: string;
 | 
				
			||||||
 | 
					  @Input() stats: ChainStats;
 | 
				
			||||||
 | 
					  @Input() right: number | string = 10;
 | 
				
			||||||
 | 
					  @Input() left: number | string = 70;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  chartOptions: EChartsOption = {};
 | 
				
			||||||
 | 
					  chartInitOptions = {
 | 
				
			||||||
 | 
					    renderer: 'svg',
 | 
				
			||||||
 | 
					  };
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  error: any;
 | 
				
			||||||
 | 
					  isLoading = true;
 | 
				
			||||||
 | 
					  chartInstance: any = undefined;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  constructor(
 | 
				
			||||||
 | 
					    @Inject(LOCALE_ID) public locale: string,
 | 
				
			||||||
 | 
					    private electrsApiService: ElectrsApiService,
 | 
				
			||||||
 | 
					    private amountShortenerPipe: AmountShortenerPipe,
 | 
				
			||||||
 | 
					    private cd: ChangeDetectorRef,
 | 
				
			||||||
 | 
					  ) {
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  ngOnInit(): void {
 | 
				
			||||||
 | 
					    
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  ngOnChanges(changes: SimpleChanges): void {
 | 
				
			||||||
 | 
					    this.isLoading = true;
 | 
				
			||||||
 | 
					    this.electrsApiService.getAddressSummary$(this.address).pipe(
 | 
				
			||||||
 | 
					      catchError(e => {
 | 
				
			||||||
 | 
					        this.error = `Failed to fetch address balance history: ${e?.status || ''} ${e?.statusText || 'unknown error'}`;
 | 
				
			||||||
 | 
					        return of(null);
 | 
				
			||||||
 | 
					      }),
 | 
				
			||||||
 | 
					    ).subscribe(addressSummary => {
 | 
				
			||||||
 | 
					      if (addressSummary) {
 | 
				
			||||||
 | 
					        this.error = null;
 | 
				
			||||||
 | 
					        this.prepareChartOptions(addressSummary);
 | 
				
			||||||
 | 
					      }
 | 
				
			||||||
 | 
					      this.isLoading = false;
 | 
				
			||||||
 | 
					      this.cd.markForCheck();
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  prepareChartOptions(summary): void {
 | 
				
			||||||
 | 
					    let total = (this.stats.funded_txo_sum - this.stats.spent_txo_sum); // + (summary[0]?.value || 0);
 | 
				
			||||||
 | 
					    const data = summary.map(d => {
 | 
				
			||||||
 | 
					      const balance = total;
 | 
				
			||||||
 | 
					      total -= d.value;
 | 
				
			||||||
 | 
					      return [d.time * 1000, balance, d];
 | 
				
			||||||
 | 
					    }).reverse();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    const maxValue = data.reduce((acc, d) => Math.max(acc, Math.abs(d[1])), 0);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    this.chartOptions = {
 | 
				
			||||||
 | 
					      color: [
 | 
				
			||||||
 | 
					        new echarts.graphic.LinearGradient(0, 0, 0, 1, [
 | 
				
			||||||
 | 
					          { offset: 0, color: '#FDD835' },
 | 
				
			||||||
 | 
					          { offset: 1, color: '#FB8C00' },
 | 
				
			||||||
 | 
					        ]),
 | 
				
			||||||
 | 
					      ],
 | 
				
			||||||
 | 
					      animation: false,
 | 
				
			||||||
 | 
					      grid: {
 | 
				
			||||||
 | 
					        top: 20,
 | 
				
			||||||
 | 
					        bottom: 20,
 | 
				
			||||||
 | 
					        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: function (data): string {
 | 
				
			||||||
 | 
					          const header = data.length === 1
 | 
				
			||||||
 | 
					            ? `${data[0].data[2].txid.slice(0, 6)}...${data[0].data[2].txid.slice(-6)}`
 | 
				
			||||||
 | 
					            : `${data.length} transactions`;
 | 
				
			||||||
 | 
					          const date = new Date(data[0].data[0]).toLocaleTimeString(this.locale, { year: 'numeric', month: 'short', day: 'numeric' });
 | 
				
			||||||
 | 
					          const val = data.reduce((total, d) => total + d.data[2].value, 0);
 | 
				
			||||||
 | 
					          const color = val === 0 ? '' : (val > 0 ? '#1a9436' : '#dc3545');
 | 
				
			||||||
 | 
					          const symbol = val > 0 ? '+' : '';
 | 
				
			||||||
 | 
					          return `
 | 
				
			||||||
 | 
					            <div>
 | 
				
			||||||
 | 
					              <span><b>${header}</b></span>
 | 
				
			||||||
 | 
					              <div style="text-align: right;">
 | 
				
			||||||
 | 
					                <span style="color: ${color}">${symbol} ${(val / 100_000_000).toFixed(8)} BTC</span><br>
 | 
				
			||||||
 | 
					                <span>${(data[0].data[1] / 100_000_000).toFixed(8)} BTC</span>
 | 
				
			||||||
 | 
					              </div>
 | 
				
			||||||
 | 
					              <span>${date}</span>
 | 
				
			||||||
 | 
					            </div>
 | 
				
			||||||
 | 
					          `; 
 | 
				
			||||||
 | 
					        }.bind(this)
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      xAxis: {
 | 
				
			||||||
 | 
					        type: 'time',
 | 
				
			||||||
 | 
					        splitNumber: this.isMobile() ? 5 : 10,
 | 
				
			||||||
 | 
					        axisLabel: {
 | 
				
			||||||
 | 
					          hideOverlap: true,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
 | 
					      yAxis: [
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					          type: 'value',
 | 
				
			||||||
 | 
					          position: 'left',
 | 
				
			||||||
 | 
					          axisLabel: {
 | 
				
			||||||
 | 
					            color: 'rgb(110, 112, 121)',
 | 
				
			||||||
 | 
					            formatter: (val): string => {
 | 
				
			||||||
 | 
					              if (maxValue > 100_000_000) {
 | 
				
			||||||
 | 
					                return `${this.amountShortenerPipe.transform(Math.round(val / 100_000_000), 0)} BTC`;
 | 
				
			||||||
 | 
					              } else if (maxValue > 10_000_000) {
 | 
				
			||||||
 | 
					                return `${Math.round(val / 100_000_000)} BTC`;
 | 
				
			||||||
 | 
					              } else if (maxValue > 100_000) {
 | 
				
			||||||
 | 
					                return `${(val / 100_000_000).toFixed(2)} BTC`;
 | 
				
			||||||
 | 
					              } else {
 | 
				
			||||||
 | 
					                return `${this.amountShortenerPipe.transform(100_000_000, 0)} sats`;
 | 
				
			||||||
 | 
					              }
 | 
				
			||||||
 | 
					            }
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          splitLine: {
 | 
				
			||||||
 | 
					            show: false,
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					      ],
 | 
				
			||||||
 | 
					      series: [
 | 
				
			||||||
 | 
					        {
 | 
				
			||||||
 | 
					          name: $localize`Balance:Balance`,
 | 
				
			||||||
 | 
					          showSymbol: false,
 | 
				
			||||||
 | 
					          symbol: 'circle',
 | 
				
			||||||
 | 
					          symbolSize: 8,
 | 
				
			||||||
 | 
					          data: data,
 | 
				
			||||||
 | 
					          areaStyle: {
 | 
				
			||||||
 | 
					            opacity: 0.5,
 | 
				
			||||||
 | 
					          },
 | 
				
			||||||
 | 
					          type: 'line',
 | 
				
			||||||
 | 
					          smooth: false,
 | 
				
			||||||
 | 
					          step: 'end'
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      ],
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  onChartInit(ec) {
 | 
				
			||||||
 | 
					    this.chartInstance = ec;
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  isMobile() {
 | 
				
			||||||
 | 
					    return (window.innerWidth <= 767.98);
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
@ -49,9 +49,19 @@
 | 
				
			|||||||
          </div>
 | 
					          </div>
 | 
				
			||||||
        </div>
 | 
					        </div>
 | 
				
			||||||
      </div>
 | 
					      </div>
 | 
				
			||||||
 | 
					 | 
				
			||||||
    </div>
 | 
					    </div>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    <ng-container *ngIf="address && transactions && transactions.length > 2">
 | 
				
			||||||
 | 
					      <br>
 | 
				
			||||||
 | 
					      <div class="box">
 | 
				
			||||||
 | 
					        <div class="row">
 | 
				
			||||||
 | 
					          <div class="col-md">
 | 
				
			||||||
 | 
					            <app-address-graph [address]="addressString" [stats]="address.chain_stats" />
 | 
				
			||||||
 | 
					          </div>
 | 
				
			||||||
 | 
					        </div>
 | 
				
			||||||
 | 
					      </div>
 | 
				
			||||||
 | 
					    </ng-container>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    <br>
 | 
					    <br>
 | 
				
			||||||
    <div class="title-tx">
 | 
					    <div class="title-tx">
 | 
				
			||||||
      <h2 class="text-left">
 | 
					      <h2 class="text-left">
 | 
				
			||||||
 | 
				
			|||||||
@ -32,12 +32,15 @@ import { AcceleratorDashboardComponent } from '../components/acceleration/accele
 | 
				
			|||||||
import { HashrateChartComponent } from '../components/hashrate-chart/hashrate-chart.component';
 | 
					import { HashrateChartComponent } from '../components/hashrate-chart/hashrate-chart.component';
 | 
				
			||||||
import { HashrateChartPoolsComponent } from '../components/hashrates-chart-pools/hashrate-chart-pools.component';
 | 
					import { HashrateChartPoolsComponent } from '../components/hashrates-chart-pools/hashrate-chart-pools.component';
 | 
				
			||||||
import { BlockHealthGraphComponent } from '../components/block-health-graph/block-health-graph.component';
 | 
					import { BlockHealthGraphComponent } from '../components/block-health-graph/block-health-graph.component';
 | 
				
			||||||
 | 
					import { AddressComponent } from '../components/address/address.component';
 | 
				
			||||||
 | 
					import { AddressGraphComponent } from '../components/address-graph/address-graph.component';
 | 
				
			||||||
import { CommonModule } from '@angular/common';
 | 
					import { CommonModule } from '@angular/common';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@NgModule({
 | 
					@NgModule({
 | 
				
			||||||
  declarations: [
 | 
					  declarations: [
 | 
				
			||||||
    DashboardComponent,
 | 
					    DashboardComponent,
 | 
				
			||||||
    MempoolBlockComponent,
 | 
					    MempoolBlockComponent,
 | 
				
			||||||
 | 
					    AddressComponent,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    MiningDashboardComponent,
 | 
					    MiningDashboardComponent,
 | 
				
			||||||
    AcceleratorDashboardComponent,
 | 
					    AcceleratorDashboardComponent,
 | 
				
			||||||
@ -67,6 +70,7 @@ import { CommonModule } from '@angular/common';
 | 
				
			|||||||
    HashrateChartComponent,
 | 
					    HashrateChartComponent,
 | 
				
			||||||
    HashrateChartPoolsComponent,
 | 
					    HashrateChartPoolsComponent,
 | 
				
			||||||
    BlockHealthGraphComponent,
 | 
					    BlockHealthGraphComponent,
 | 
				
			||||||
 | 
					    AddressGraphComponent,
 | 
				
			||||||
  ],
 | 
					  ],
 | 
				
			||||||
  imports: [
 | 
					  imports: [
 | 
				
			||||||
    CommonModule,
 | 
					    CommonModule,
 | 
				
			||||||
 | 
				
			|||||||
@ -19,6 +19,7 @@ import { TelevisionComponent } from '../components/television/television.compone
 | 
				
			|||||||
import { DashboardComponent } from '../dashboard/dashboard.component';
 | 
					import { DashboardComponent } from '../dashboard/dashboard.component';
 | 
				
			||||||
import { AccelerationFeesGraphComponent } from '../components/acceleration/acceleration-fees-graph/acceleration-fees-graph.component';
 | 
					import { AccelerationFeesGraphComponent } from '../components/acceleration/acceleration-fees-graph/acceleration-fees-graph.component';
 | 
				
			||||||
import { AccelerationsListComponent } from '../components/acceleration/accelerations-list/accelerations-list.component';
 | 
					import { AccelerationsListComponent } from '../components/acceleration/accelerations-list/accelerations-list.component';
 | 
				
			||||||
 | 
					import { AddressComponent } from '../components/address/address.component';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const routes: Routes = [
 | 
					const routes: Routes = [
 | 
				
			||||||
  {
 | 
					  {
 | 
				
			||||||
@ -67,6 +68,15 @@ const routes: Routes = [
 | 
				
			|||||||
          },
 | 
					          },
 | 
				
			||||||
        ]
 | 
					        ]
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
 | 
					      {
 | 
				
			||||||
 | 
					        path: 'address/:id',
 | 
				
			||||||
 | 
					        children: [],
 | 
				
			||||||
 | 
					        component: AddressComponent,
 | 
				
			||||||
 | 
					        data: {
 | 
				
			||||||
 | 
					          ogImage: true,
 | 
				
			||||||
 | 
					          networkSpecific: true,
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        path: 'graphs',
 | 
					        path: 'graphs',
 | 
				
			||||||
        data: { networks: ['bitcoin', 'liquid'] },
 | 
					        data: { networks: ['bitcoin', 'liquid'] },
 | 
				
			||||||
 | 
				
			|||||||
@ -149,6 +149,13 @@ export interface AddressOrScriptHash {
 | 
				
			|||||||
  mempool_stats: MempoolStats;
 | 
					  mempool_stats: MempoolStats;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export interface AddressTxSummary {
 | 
				
			||||||
 | 
					  txid: string;
 | 
				
			||||||
 | 
					  value: number;
 | 
				
			||||||
 | 
					  height: number;
 | 
				
			||||||
 | 
					  time: number;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export interface ChainStats {
 | 
					export interface ChainStats {
 | 
				
			||||||
  funded_txo_count: number;
 | 
					  funded_txo_count: number;
 | 
				
			||||||
  funded_txo_sum: number;
 | 
					  funded_txo_sum: number;
 | 
				
			||||||
 | 
				
			|||||||
@ -7,7 +7,6 @@ import { LiquidMasterPageComponent } from '../components/liquid-master-page/liqu
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { StartComponent } from '../components/start/start.component';
 | 
					import { StartComponent } from '../components/start/start.component';
 | 
				
			||||||
import { AddressComponent } from '../components/address/address.component';
 | 
					 | 
				
			||||||
import { PushTransactionComponent } from '../components/push-transaction/push-transaction.component';
 | 
					import { PushTransactionComponent } from '../components/push-transaction/push-transaction.component';
 | 
				
			||||||
import { BlocksList } from '../components/blocks-list/blocks-list.component';
 | 
					import { BlocksList } from '../components/blocks-list/blocks-list.component';
 | 
				
			||||||
import { AssetGroupComponent } from '../components/assets/asset-group/asset-group.component';
 | 
					import { AssetGroupComponent } from '../components/assets/asset-group/asset-group.component';
 | 
				
			||||||
@ -51,15 +50,6 @@ const routes: Routes = [
 | 
				
			|||||||
        path: 'trademark-policy',
 | 
					        path: 'trademark-policy',
 | 
				
			||||||
        loadChildren: () => import('../components/trademark-policy/trademark-policy.module').then(m => m.TrademarkModule),
 | 
					        loadChildren: () => import('../components/trademark-policy/trademark-policy.module').then(m => m.TrademarkModule),
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
      {
 | 
					 | 
				
			||||||
        path: 'address/:id',
 | 
					 | 
				
			||||||
        children: [],
 | 
					 | 
				
			||||||
        component: AddressComponent,
 | 
					 | 
				
			||||||
        data: {
 | 
					 | 
				
			||||||
          ogImage: true,
 | 
					 | 
				
			||||||
          networkSpecific: true,
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
      },
 | 
					 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        path: 'tx',
 | 
					        path: 'tx',
 | 
				
			||||||
        component: StartComponent,
 | 
					        component: StartComponent,
 | 
				
			||||||
 | 
				
			|||||||
@ -5,8 +5,6 @@ import { MasterPageComponent } from './components/master-page/master-page.compon
 | 
				
			|||||||
import { SharedModule } from './shared/shared.module';
 | 
					import { SharedModule } from './shared/shared.module';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import { StartComponent } from './components/start/start.component';
 | 
					import { StartComponent } from './components/start/start.component';
 | 
				
			||||||
import { AddressComponent } from './components/address/address.component';
 | 
					 | 
				
			||||||
import { AddressGroupComponent } from './components/address-group/address-group.component';
 | 
					 | 
				
			||||||
import { PushTransactionComponent } from './components/push-transaction/push-transaction.component';
 | 
					import { PushTransactionComponent } from './components/push-transaction/push-transaction.component';
 | 
				
			||||||
import { CalculatorComponent } from './components/calculator/calculator.component';
 | 
					import { CalculatorComponent } from './components/calculator/calculator.component';
 | 
				
			||||||
import { BlocksList } from './components/blocks-list/blocks-list.component';
 | 
					import { BlocksList } from './components/blocks-list/blocks-list.component';
 | 
				
			||||||
@ -56,15 +54,6 @@ const routes: Routes = [
 | 
				
			|||||||
        path: 'trademark-policy',
 | 
					        path: 'trademark-policy',
 | 
				
			||||||
        loadChildren: () => import('./components/trademark-policy/trademark-policy.module').then(m => m.TrademarkModule),
 | 
					        loadChildren: () => import('./components/trademark-policy/trademark-policy.module').then(m => m.TrademarkModule),
 | 
				
			||||||
      },
 | 
					      },
 | 
				
			||||||
      {
 | 
					 | 
				
			||||||
        path: 'address/:id',
 | 
					 | 
				
			||||||
        children: [],
 | 
					 | 
				
			||||||
        component: AddressComponent,
 | 
					 | 
				
			||||||
        data: {
 | 
					 | 
				
			||||||
          ogImage: true,
 | 
					 | 
				
			||||||
          networkSpecific: true,
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
      },
 | 
					 | 
				
			||||||
      {
 | 
					      {
 | 
				
			||||||
        path: 'tx',
 | 
					        path: 'tx',
 | 
				
			||||||
        component: StartComponent,
 | 
					        component: StartComponent,
 | 
				
			||||||
 | 
				
			|||||||
@ -1,7 +1,7 @@
 | 
				
			|||||||
import { Injectable } from '@angular/core';
 | 
					import { Injectable } from '@angular/core';
 | 
				
			||||||
import { HttpClient, HttpParams } from '@angular/common/http';
 | 
					import { HttpClient, HttpParams } from '@angular/common/http';
 | 
				
			||||||
import { BehaviorSubject, Observable, catchError, filter, from, of, shareReplay, switchMap, take, tap } from 'rxjs';
 | 
					import { BehaviorSubject, Observable, catchError, filter, from, of, shareReplay, switchMap, take, tap } from 'rxjs';
 | 
				
			||||||
import { Transaction, Address, Outspend, Recent, Asset, ScriptHash } from '../interfaces/electrs.interface';
 | 
					import { Transaction, Address, Outspend, Recent, Asset, ScriptHash, AddressTxSummary } from '../interfaces/electrs.interface';
 | 
				
			||||||
import { StateService } from './state.service';
 | 
					import { StateService } from './state.service';
 | 
				
			||||||
import { BlockExtended } from '../interfaces/node-api.interface';
 | 
					import { BlockExtended } from '../interfaces/node-api.interface';
 | 
				
			||||||
import { calcScriptHash$ } from '../bitcoin.utils';
 | 
					import { calcScriptHash$ } from '../bitcoin.utils';
 | 
				
			||||||
@ -141,6 +141,14 @@ export class ElectrsApiService {
 | 
				
			|||||||
    return this.httpClient.get<Transaction[]>(this.apiBaseUrl + this.apiBasePath + '/api/address/' + address + '/txs', { params });
 | 
					    return this.httpClient.get<Transaction[]>(this.apiBaseUrl + this.apiBasePath + '/api/address/' + address + '/txs', { params });
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  getAddressSummary$(address: string,  txid?: string): Observable<AddressTxSummary[]> {
 | 
				
			||||||
 | 
					    let params = new HttpParams();
 | 
				
			||||||
 | 
					    if (txid) {
 | 
				
			||||||
 | 
					      params = params.append('after_txid', txid);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    return this.httpClient.get<AddressTxSummary[]>(this.apiBaseUrl + this.apiBasePath + '/api/address/' + address + '/txs/summary', { params });
 | 
				
			||||||
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  getScriptHashTransactions$(script: string,  txid?: string): Observable<Transaction[]> {
 | 
					  getScriptHashTransactions$(script: string,  txid?: string): Observable<Transaction[]> {
 | 
				
			||||||
    let params = new HttpParams();
 | 
					    let params = new HttpParams();
 | 
				
			||||||
    if (txid) {
 | 
					    if (txid) {
 | 
				
			||||||
 | 
				
			|||||||
@ -45,7 +45,6 @@ import { TransactionsListComponent } from '../components/transactions-list/trans
 | 
				
			|||||||
import { BlockOverviewGraphComponent } from '../components/block-overview-graph/block-overview-graph.component';
 | 
					import { BlockOverviewGraphComponent } from '../components/block-overview-graph/block-overview-graph.component';
 | 
				
			||||||
import { BlockOverviewTooltipComponent } from '../components/block-overview-tooltip/block-overview-tooltip.component';
 | 
					import { BlockOverviewTooltipComponent } from '../components/block-overview-tooltip/block-overview-tooltip.component';
 | 
				
			||||||
import { BlockFiltersComponent } from '../components/block-filters/block-filters.component';
 | 
					import { BlockFiltersComponent } from '../components/block-filters/block-filters.component';
 | 
				
			||||||
import { AddressComponent } from '../components/address/address.component';
 | 
					 | 
				
			||||||
import { AddressGroupComponent } from '../components/address-group/address-group.component';
 | 
					import { AddressGroupComponent } from '../components/address-group/address-group.component';
 | 
				
			||||||
import { SearchFormComponent } from '../components/search-form/search-form.component';
 | 
					import { SearchFormComponent } from '../components/search-form/search-form.component';
 | 
				
			||||||
import { AddressLabelsComponent } from '../components/address-labels/address-labels.component';
 | 
					import { AddressLabelsComponent } from '../components/address-labels/address-labels.component';
 | 
				
			||||||
@ -147,7 +146,6 @@ import { OnlyVsizeDirective, OnlyWeightDirective } from './components/weight-dir
 | 
				
			|||||||
    BlockOverviewTooltipComponent,
 | 
					    BlockOverviewTooltipComponent,
 | 
				
			||||||
    BlockFiltersComponent,
 | 
					    BlockFiltersComponent,
 | 
				
			||||||
    TransactionsListComponent,
 | 
					    TransactionsListComponent,
 | 
				
			||||||
    AddressComponent,
 | 
					 | 
				
			||||||
    AddressGroupComponent,
 | 
					    AddressGroupComponent,
 | 
				
			||||||
    SearchFormComponent,
 | 
					    SearchFormComponent,
 | 
				
			||||||
    AddressLabelsComponent,
 | 
					    AddressLabelsComponent,
 | 
				
			||||||
@ -276,7 +274,6 @@ import { OnlyVsizeDirective, OnlyWeightDirective } from './components/weight-dir
 | 
				
			|||||||
    BlockOverviewTooltipComponent,
 | 
					    BlockOverviewTooltipComponent,
 | 
				
			||||||
    BlockFiltersComponent,
 | 
					    BlockFiltersComponent,
 | 
				
			||||||
    TransactionsListComponent,
 | 
					    TransactionsListComponent,
 | 
				
			||||||
    AddressComponent,
 | 
					 | 
				
			||||||
    AddressGroupComponent,
 | 
					    AddressGroupComponent,
 | 
				
			||||||
    SearchFormComponent,
 | 
					    SearchFormComponent,
 | 
				
			||||||
    AddressLabelsComponent,
 | 
					    AddressLabelsComponent,
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user