Merge pull request #4785 from mempool/natsoni/bowtie-tooltip-price
Display more accurate price data on tx bowtie tooltip
This commit is contained in:
		
						commit
						b7d9efebd1
					
				| @ -8,9 +8,12 @@ | |||||||
|     }} |     }} | ||||||
|   </span> |   </span> | ||||||
|   <ng-template #noblockconversion> |   <ng-template #noblockconversion> | ||||||
|     <span class="fiat">{{ addPlus && satoshis >= 0 ? '+' : '' }}{{ (conversions[currency] > -1 ? conversions[currency] : 0) * satoshis / 100000000 | fiatCurrency : digitsInfo : currency }} |     <span class="fiat" *ngIf="!forceBlockConversion; else zeroValue">{{ addPlus && satoshis >= 0 ? '+' : '' }}{{ (conversions[currency] > -1 ? conversions[currency] : 0) * satoshis / 100000000 | fiatCurrency : digitsInfo : currency }} | ||||||
|     </span> |     </span> | ||||||
|   </ng-template> |   </ng-template> | ||||||
|  |   <ng-template #zeroValue> | ||||||
|  |     <span class="fiat">{{ 0 | fiatCurrency : digitsInfo : currency }}</span> | ||||||
|  |   </ng-template> | ||||||
| </ng-container> | </ng-container> | ||||||
| 
 | 
 | ||||||
|   <ng-template #viewFiatVin> |   <ng-template #viewFiatVin> | ||||||
|  | |||||||
| @ -24,6 +24,7 @@ export class AmountComponent implements OnInit, OnDestroy { | |||||||
|   @Input() addPlus = false; |   @Input() addPlus = false; | ||||||
|   @Input() blockConversion: Price; |   @Input() blockConversion: Price; | ||||||
|   @Input() forceBtc: boolean = false; |   @Input() forceBtc: boolean = false; | ||||||
|  |   @Input() forceBlockConversion: boolean = false; // true = displays fiat price as 0 if blockConversion is undefined instead of falling back to conversions
 | ||||||
| 
 | 
 | ||||||
|   constructor( |   constructor( | ||||||
|     private stateService: StateService, |     private stateService: StateService, | ||||||
|  | |||||||
| @ -44,6 +44,28 @@ | |||||||
|           <span *ngSwitchCase="'fee'" i18n="transaction.fee|Transaction fee">Fee</span> |           <span *ngSwitchCase="'fee'" i18n="transaction.fee|Transaction fee">Fee</span> | ||||||
|         </ng-container> |         </ng-container> | ||||||
|         <span *ngIf="line.type !== 'fee'"> #{{ line.index + 1 }}</span> |         <span *ngIf="line.type !== 'fee'"> #{{ line.index + 1 }}</span> | ||||||
|  |         <ng-container [ngSwitch]="line.type"> | ||||||
|  |           <span *ngSwitchCase="'input'">  | ||||||
|  |             <ng-container *ngIf="line.status?.block_height"> | ||||||
|  |                 <ng-container *ngIf="line.blockHeight; else noBlockHeight"> | ||||||
|  |                   <ng-container *ngTemplateOutlet="nBlocksEarlier; context:{n: line.blockHeight - line?.status?.block_height, connector: false}"></ng-container> | ||||||
|  |               </ng-container> | ||||||
|  |               <ng-template #noBlockHeight> | ||||||
|  |                 <ng-container *ngTemplateOutlet="nBlocksEarlier; context:{n: chainTip + 1 - line?.status?.block_height, connector: false}"></ng-container> | ||||||
|  |               </ng-template> | ||||||
|  |             </ng-container> | ||||||
|  |           </span> | ||||||
|  |           <span *ngSwitchCase="'output'"> | ||||||
|  |             <ng-container *ngIf="line.blockHeight && line?.spent"> | ||||||
|  |               <ng-container *ngIf="line?.status?.block_height; else noBlockHeight"> | ||||||
|  |                 <ng-container *ngTemplateOutlet="nBlocksLater; context:{n: line?.status?.block_height - line.blockHeight, connector: false}"></ng-container> | ||||||
|  |               </ng-container> | ||||||
|  |               <ng-template #noBlockHeight> | ||||||
|  |                 <ng-container *ngTemplateOutlet="nBlocksLater; context:{n: chainTip + 1 - line.blockHeight, connector: false}"></ng-container> | ||||||
|  |               </ng-template> | ||||||
|  |             </ng-container> | ||||||
|  |           </span> | ||||||
|  |         </ng-container> | ||||||
|       </p> |       </p> | ||||||
|       <ng-container *ngIf="isConnector && line.txid"> |       <ng-container *ngIf="isConnector && line.txid"> | ||||||
|         <p> |         <p> | ||||||
| @ -51,8 +73,26 @@ | |||||||
|           <app-truncate [text]="line.txid"></app-truncate> |           <app-truncate [text]="line.txid"></app-truncate> | ||||||
|         </p> |         </p> | ||||||
|           <ng-container [ngSwitch]="line.type"> |           <ng-container [ngSwitch]="line.type"> | ||||||
|             <p *ngSwitchCase="'input'"><span i18n="transaction.output">Output</span>  #{{ line.vout + 1 }}</p> |             <p *ngSwitchCase="'input'"><span i18n="transaction.output">Output</span>  #{{ line.vout + 1 }} | ||||||
|             <p *ngSwitchCase="'output'"><span i18n="transaction.input">Input</span>  #{{ line.vin + 1 }}</p> |               <ng-container *ngIf="line.status?.block_height"> | ||||||
|  |                 <ng-container *ngIf="line.blockHeight; else noBlockHeight"> | ||||||
|  |                   <ng-container *ngTemplateOutlet="nBlocksEarlier; context:{n: line.blockHeight - line?.status?.block_height, connector: true}"></ng-container> | ||||||
|  |                 </ng-container> | ||||||
|  |                 <ng-template #noBlockHeight> | ||||||
|  |                   <ng-container *ngTemplateOutlet="nBlocksEarlier; context:{n: chainTip + 1 - line?.status?.block_height, connector: true}"></ng-container> | ||||||
|  |                 </ng-template> | ||||||
|  |               </ng-container> | ||||||
|  |             </p> | ||||||
|  |             <p *ngSwitchCase="'output'"><span i18n="transaction.input">Input</span>  #{{ line.vin + 1 }} | ||||||
|  |               <ng-container *ngIf="line.blockHeight"> | ||||||
|  |                 <ng-container *ngIf="line?.status?.block_height; else noBlockHeight"> | ||||||
|  |                   <ng-container *ngTemplateOutlet="nBlocksLater; context:{n: line?.status?.block_height - line.blockHeight, connector: true}"></ng-container> | ||||||
|  |                 </ng-container> | ||||||
|  |                 <ng-template #noBlockHeight> | ||||||
|  |                   <ng-container *ngTemplateOutlet="nBlocksLater; context:{n: chainTip + 1 - line.blockHeight, connector: true}"></ng-container> | ||||||
|  |                 </ng-template> | ||||||
|  |               </ng-container> | ||||||
|  |             </p> | ||||||
|           </ng-container> |           </ng-container> | ||||||
|       </ng-container> |       </ng-container> | ||||||
|       <p *ngIf="line.displayValue == null && line.confidential" i18n="shared.confidential">Confidential</p> |       <p *ngIf="line.displayValue == null && line.confidential" i18n="shared.confidential">Confidential</p> | ||||||
| @ -66,7 +106,7 @@ | |||||||
|           </ng-template> |           </ng-template> | ||||||
|         </ng-template> |         </ng-template> | ||||||
|         <ng-template #defaultOutput> |         <ng-template #defaultOutput> | ||||||
|           <app-amount [blockConversion]="blockConversion" [satoshis]="line.displayValue"></app-amount> |           <app-amount [blockConversion]="isConnector ? blockConversions[line?.status?.block_time] : blockConversions[line?.timestamp]" [satoshis]="line.displayValue" [forceBlockConversion]="isConnector && line?.status?.block_time"></app-amount> | ||||||
|         </ng-template> |         </ng-template> | ||||||
|       </p> |       </p> | ||||||
|       <p *ngIf="line.type !== 'fee' && line.address" class="address"> |       <p *ngIf="line.type !== 'fee' && line.address" class="address"> | ||||||
| @ -78,3 +118,41 @@ | |||||||
| <ng-template #assetBox let-item> | <ng-template #assetBox let-item> | ||||||
|   {{ item.displayValue / pow(10, assetsMinimal[item.asset][3]) | number: '1.' + assetsMinimal[item.asset][3] + '-' + assetsMinimal[item.asset][3] }} <span class="symbol">{{ assetsMinimal[item.asset][1] }}</span> |   {{ item.displayValue / pow(10, assetsMinimal[item.asset][3]) | number: '1.' + assetsMinimal[item.asset][3] + '-' + assetsMinimal[item.asset][3] }} <span class="symbol">{{ assetsMinimal[item.asset][1] }}</span> | ||||||
| </ng-template> | </ng-template> | ||||||
|  | 
 | ||||||
|  | <ng-template #oneBlockEarlier> | ||||||
|  |   <span i18n="shared.one-block-earlier">1 block earlier</span> | ||||||
|  | </ng-template> | ||||||
|  | 
 | ||||||
|  | <ng-template #oneBlockLater> | ||||||
|  |   <span i18n="shared.one-block-later">1 block later</span> | ||||||
|  | </ng-template> | ||||||
|  | 
 | ||||||
|  | <ng-template #inTheSameBlock> | ||||||
|  |   <span i18n="shared.in-the-same-block">in the same block</span> | ||||||
|  | </ng-template> | ||||||
|  | 
 | ||||||
|  | <ng-template #nBlocksEarlier let-n="n" let-connector="connector"> | ||||||
|  |   (<span *ngIf="!connector">prevout </span> | ||||||
|  |   <ng-container *ngIf="n > 1"> | ||||||
|  |     <span>{{ n }} <ng-container i18n="shared.n-blocks-earlier">blocks earlier</ng-container>)</span> | ||||||
|  |   </ng-container> | ||||||
|  |   <ng-container *ngIf="n === 1"> | ||||||
|  |     <span><ng-container *ngTemplateOutlet="oneBlockEarlier"></ng-container>)</span> | ||||||
|  |   </ng-container> | ||||||
|  |   <ng-container *ngIf="n === 0"> | ||||||
|  |     <span><ng-container *ngTemplateOutlet="inTheSameBlock"></ng-container>)</span> | ||||||
|  |   </ng-container> | ||||||
|  | </ng-template> | ||||||
|  | 
 | ||||||
|  | <ng-template #nBlocksLater let-n="n" let-connector="connector"> | ||||||
|  |   (<span *ngIf="!connector" i18n="shared.spent">spent </span> | ||||||
|  |   <ng-container *ngIf="n > 1"> | ||||||
|  |     <span>{{ n }} <ng-container i18n="shared.n-blocks-later">blocks later</ng-container>)</span> | ||||||
|  |   </ng-container> | ||||||
|  |   <ng-container *ngIf="n === 1"> | ||||||
|  |     <span><ng-container *ngTemplateOutlet="oneBlockLater"></ng-container>)</span> | ||||||
|  |   </ng-container> | ||||||
|  |   <ng-container *ngIf="n === 0"> | ||||||
|  |     <span><ng-container *ngTemplateOutlet="inTheSameBlock"></ng-container>)</span> | ||||||
|  |   </ng-container> | ||||||
|  | </ng-template> | ||||||
| @ -7,7 +7,7 @@ | |||||||
|   padding: 10px 15px; |   padding: 10px 15px; | ||||||
|   text-align: left; |   text-align: left; | ||||||
|   pointer-events: none; |   pointer-events: none; | ||||||
|   max-width: 300px; |   max-width: 350px; | ||||||
| 
 | 
 | ||||||
|   p { |   p { | ||||||
|     margin: 0; |     margin: 0; | ||||||
|  | |||||||
| @ -2,6 +2,7 @@ import { Component, ElementRef, ViewChild, Input, OnChanges, OnInit } from '@ang | |||||||
| import { Subscription, of, switchMap, tap } from 'rxjs'; | import { Subscription, of, switchMap, tap } from 'rxjs'; | ||||||
| import { Price, PriceService } from '../../services/price.service'; | import { Price, PriceService } from '../../services/price.service'; | ||||||
| import { StateService } from '../../services/state.service'; | import { StateService } from '../../services/state.service'; | ||||||
|  | import { ApiService } from '../../services/api.service'; | ||||||
| import { environment } from '../../../environments/environment'; | import { environment } from '../../../environments/environment'; | ||||||
| 
 | 
 | ||||||
| interface Xput { | interface Xput { | ||||||
| @ -19,6 +20,9 @@ interface Xput { | |||||||
|   pegout?: string; |   pegout?: string; | ||||||
|   confidential?: boolean; |   confidential?: boolean; | ||||||
|   timestamp?: number; |   timestamp?: number; | ||||||
|  |   blockHeight?: number; | ||||||
|  |   status?: any; | ||||||
|  |   spent?: boolean; | ||||||
|   asset?: string; |   asset?: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -34,8 +38,14 @@ export class TxBowtieGraphTooltipComponent implements OnChanges { | |||||||
|   @Input() assetsMinimal: any; |   @Input() assetsMinimal: any; | ||||||
| 
 | 
 | ||||||
|   tooltipPosition = { x: 0, y: 0 }; |   tooltipPosition = { x: 0, y: 0 }; | ||||||
|   blockConversion: Price; |   blockConversions: { [timestamp: number]: Price } = {}; | ||||||
|  |   inputStatus: { [index: number]: any } = {}; | ||||||
|  |   currency: string; | ||||||
|  |   viewFiat: boolean; | ||||||
|  |   chainTip: number; | ||||||
|   currencyChangeSubscription: Subscription; |   currencyChangeSubscription: Subscription; | ||||||
|  |   viewFiatSubscription: Subscription; | ||||||
|  |   chainTipSubscription: Subscription; | ||||||
| 
 | 
 | ||||||
|   nativeAssetId = this.stateService.network === 'liquidtestnet' ? environment.nativeTestAssetId : environment.nativeAssetId; |   nativeAssetId = this.stateService.network === 'liquidtestnet' ? environment.nativeTestAssetId : environment.nativeAssetId; | ||||||
| 
 | 
 | ||||||
| @ -44,18 +54,37 @@ export class TxBowtieGraphTooltipComponent implements OnChanges { | |||||||
|   constructor( |   constructor( | ||||||
|     private priceService: PriceService, |     private priceService: PriceService, | ||||||
|     private stateService: StateService, |     private stateService: StateService, | ||||||
|  |     private apiService: ApiService, | ||||||
|   ) {} |   ) {} | ||||||
| 
 | 
 | ||||||
|  |   ngOnInit(): void { | ||||||
|  |     this.currencyChangeSubscription = this.stateService.fiatCurrency$.subscribe(currency => { | ||||||
|  |       this.currency = currency; | ||||||
|  |       this.blockConversions = {}; | ||||||
|  |       this.inputStatus = {}; | ||||||
|  |     }); | ||||||
|  |     this.viewFiatSubscription = this.stateService.viewFiat$.subscribe(viewFiat => this.viewFiat = viewFiat); | ||||||
|  |     this.chainTipSubscription = this.stateService.chainTip$.subscribe(tip => this.chainTip = tip); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   ngOnChanges(changes): void { |   ngOnChanges(changes): void { | ||||||
|     if (changes.line?.currentValue) { |     if (changes.line?.currentValue) { | ||||||
|       this.currencyChangeSubscription?.unsubscribe(); |       if (changes.line.currentValue.type === 'input') { | ||||||
|           this.currencyChangeSubscription = this.stateService.fiatCurrency$.pipe( |         if (!this.inputStatus[changes.line.currentValue.index]) { | ||||||
|             switchMap((currency) => { |           this.apiService.getTransactionStatus$(changes.line.currentValue.txid).pipe( | ||||||
|               return changes.line?.currentValue.timestamp ? this.priceService.getBlockPrice$(changes.line?.currentValue.timestamp, true, currency).pipe( |             tap((status) => { | ||||||
|                 tap((price) => this.blockConversion = price), |               changes.line.currentValue.status = status; | ||||||
|               ) : of(undefined); |               this.inputStatus[changes.line.currentValue.index] = status; | ||||||
|  |               this.fetchPrices(changes); | ||||||
|             }) |             }) | ||||||
|           ).subscribe(); |           ).subscribe(); | ||||||
|  |         } else { | ||||||
|  |           changes.line.currentValue.status = this.inputStatus[changes.line.currentValue.index]; | ||||||
|  |           this.fetchPrices(changes); | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         this.fetchPrices(changes); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (changes.cursorPosition && changes.cursorPosition.currentValue) { |     if (changes.cursorPosition && changes.cursorPosition.currentValue) { | ||||||
| @ -75,7 +104,32 @@ export class TxBowtieGraphTooltipComponent implements OnChanges { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   fetchPrices(changes: any) { | ||||||
|  |     if (!this.currency || !this.viewFiat) return; | ||||||
|  |     if (this.isConnector) { // If the tooltip is on a connector, we fetch prices at the time of the input / output
 | ||||||
|  |       if (['input', 'output'].includes(changes.line.currentValue.type) && changes.line.currentValue?.status?.block_time && !this.blockConversions?.[changes.line.currentValue?.status.block_time]) { | ||||||
|  |         this.priceService.getBlockPrice$(changes.line.currentValue?.status.block_time, true, this.currency).pipe( | ||||||
|  |           tap((price) => this.blockConversions[changes.line.currentValue.status.block_time] = price), | ||||||
|  |         ).subscribe(); | ||||||
|  |       } | ||||||
|  |     } else { // If the tooltip is on the transaction itself, we fetch prices at the time of the transaction
 | ||||||
|  |       if (changes.line.currentValue.timestamp && !this.blockConversions[changes.line.currentValue.timestamp]) { | ||||||
|  |         if (changes.line.currentValue.timestamp) { | ||||||
|  |           this.priceService.getBlockPrice$(changes.line.currentValue.timestamp, true, this.currency).pipe( | ||||||
|  |             tap((price) => this.blockConversions[changes.line.currentValue.timestamp] = price), | ||||||
|  |           ).subscribe(); | ||||||
|  |         } | ||||||
|  |       }  | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   pow(base: number, exponent: number): number { |   pow(base: number, exponent: number): number { | ||||||
|     return Math.pow(base, exponent); |     return Math.pow(base, exponent); | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|  |   ngOnDestroy(): void { | ||||||
|  |     this.currencyChangeSubscription?.unsubscribe(); | ||||||
|  |     this.viewFiatSubscription?.unsubscribe(); | ||||||
|  |     this.chainTipSubscription?.unsubscribe(); | ||||||
|  |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -34,6 +34,7 @@ interface Xput { | |||||||
|   pegout?: string; |   pegout?: string; | ||||||
|   confidential?: boolean; |   confidential?: boolean; | ||||||
|   timestamp?: number; |   timestamp?: number; | ||||||
|  |   blockHeight?: number; | ||||||
|   asset?: string; |   asset?: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -178,6 +179,7 @@ export class TxBowtieGraphComponent implements OnInit, OnChanges { | |||||||
|         pegout: v?.pegout?.scriptpubkey_address, |         pegout: v?.pegout?.scriptpubkey_address, | ||||||
|         confidential: (this.isLiquid && v?.value === undefined), |         confidential: (this.isLiquid && v?.value === undefined), | ||||||
|         timestamp: this.tx.status.block_time, |         timestamp: this.tx.status.block_time, | ||||||
|  |         blockHeight: this.tx.status.block_height, | ||||||
|         asset: v?.asset, |         asset: v?.asset, | ||||||
|       } as Xput; |       } as Xput; | ||||||
|     }); |     }); | ||||||
| @ -200,6 +202,7 @@ export class TxBowtieGraphComponent implements OnInit, OnChanges { | |||||||
|         pegin: v?.is_pegin, |         pegin: v?.is_pegin, | ||||||
|         confidential: (this.isLiquid && v?.prevout?.value === undefined), |         confidential: (this.isLiquid && v?.prevout?.value === undefined), | ||||||
|         timestamp: this.tx.status.block_time, |         timestamp: this.tx.status.block_time, | ||||||
|  |         blockHeight: this.tx.status.block_height, | ||||||
|         asset: v?.prevout?.asset, |         asset: v?.prevout?.asset, | ||||||
|       } as Xput; |       } as Xput; | ||||||
|     }); |     }); | ||||||
|  | |||||||
| @ -240,6 +240,10 @@ export class ApiService { | |||||||
|     return this.httpClient.post<any>(this.apiBaseUrl + this.apiBasePath + '/api/tx', hexPayload, { responseType: 'text' as 'json'}); |     return this.httpClient.post<any>(this.apiBaseUrl + this.apiBasePath + '/api/tx', hexPayload, { responseType: 'text' as 'json'}); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   getTransactionStatus$(txid: string): Observable<any> { | ||||||
|  |     return this.httpClient.get<any>(this.apiBaseUrl + this.apiBasePath + '/api/tx/' + txid + '/status'); | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   listPools$(interval: string | undefined) : Observable<any> { |   listPools$(interval: string | undefined) : Observable<any> { | ||||||
|     return this.httpClient.get<any>( |     return this.httpClient.get<any>( | ||||||
|       this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/pools` + |       this.apiBaseUrl + this.apiBasePath + `/api/v1/mining/pools` + | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user