Merge pull request #4689 from mempool/mononaut/wallet-balance
Add embeddable wallet balance page
This commit is contained in:
		
						commit
						7f18d73443
					
				| @ -6,6 +6,7 @@ import { EightBlocksComponent } from './components/eight-blocks/eight-blocks.com | ||||
| import { MempoolBlockViewComponent } from './components/mempool-block-view/mempool-block-view.component'; | ||||
| import { ClockComponent } from './components/clock/clock.component'; | ||||
| import { StatusViewComponent } from './components/status-view/status-view.component'; | ||||
| import { AddressGroupComponent } from './components/address-group/address-group.component'; | ||||
| 
 | ||||
| const browserWindow = window || {}; | ||||
| // @ts-ignore
 | ||||
| @ -26,6 +27,14 @@ let routes: Routes = [ | ||||
|         loadChildren: () => import('./master-page.module').then(m => m.MasterPageModule), | ||||
|         data: { preload: true }, | ||||
|       }, | ||||
|       { | ||||
|         path: 'wallet', | ||||
|         children: [], | ||||
|         component: AddressGroupComponent, | ||||
|         data: { | ||||
|           networkSpecific: true, | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         path: 'status', | ||||
|         data: { networks: ['bitcoin', 'liquid'] }, | ||||
| @ -61,6 +70,14 @@ let routes: Routes = [ | ||||
|         loadChildren: () => import('./master-page.module').then(m => m.MasterPageModule), | ||||
|         data: { preload: true }, | ||||
|       }, | ||||
|       { | ||||
|         path: 'wallet', | ||||
|         children: [], | ||||
|         component: AddressGroupComponent, | ||||
|         data: { | ||||
|           networkSpecific: true, | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         path: 'status', | ||||
|         data: { networks: ['bitcoin', 'liquid'] }, | ||||
| @ -88,6 +105,14 @@ let routes: Routes = [ | ||||
|     loadChildren: () => import('./master-page.module').then(m => m.MasterPageModule), | ||||
|     data: { preload: true }, | ||||
|   }, | ||||
|   { | ||||
|     path: 'wallet', | ||||
|     children: [], | ||||
|     component: AddressGroupComponent, | ||||
|     data: { | ||||
|       networkSpecific: true, | ||||
|     } | ||||
|   }, | ||||
|   { | ||||
|     path: 'preview', | ||||
|     children: [ | ||||
| @ -168,6 +193,14 @@ if (browserWindowEnv && browserWindowEnv.BASE_MODULE === 'liquid') { | ||||
|           loadChildren: () => import ('./liquid/liquid-master-page.module').then(m => m.LiquidMasterPageModule), | ||||
|           data: { preload: true }, | ||||
|         }, | ||||
|         { | ||||
|           path: 'wallet', | ||||
|           children: [], | ||||
|           component: AddressGroupComponent, | ||||
|           data: { | ||||
|             networkSpecific: true, | ||||
|           } | ||||
|         }, | ||||
|         { | ||||
|           path: 'status', | ||||
|           data: { networks: ['bitcoin', 'liquid'] }, | ||||
| @ -195,6 +228,14 @@ if (browserWindowEnv && browserWindowEnv.BASE_MODULE === 'liquid') { | ||||
|       loadChildren: () => import ('./liquid/liquid-master-page.module').then(m => m.LiquidMasterPageModule), | ||||
|       data: { preload: true }, | ||||
|     }, | ||||
|     { | ||||
|       path: 'wallet', | ||||
|       children: [], | ||||
|       component: AddressGroupComponent, | ||||
|       data: { | ||||
|         networkSpecific: true, | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       path: 'preview', | ||||
|       children: [ | ||||
|  | ||||
| @ -0,0 +1,24 @@ | ||||
| <div class="frame {{ screenSize }}" [class.liquid-address]="network === 'liquid' || network === 'liquidtestnet'"> | ||||
|   <div class="heading"> | ||||
|     <app-svg-images name="officialMempoolSpace" style="width: 144px; height: 36px" width="500" height="126" viewBox="0 0 500 126"></app-svg-images> | ||||
|     <h3 i18n="addresses.balance">Balances</h3> | ||||
|     <div class="spacer"></div> | ||||
|   </div> | ||||
|   <table class="table table-borderless table-striped table-fixed"> | ||||
|     <tr> | ||||
|       <th class="address" i18n="addresses.total">Total</th> | ||||
|       <th class="btc"><app-amount [satoshis]="balance" [digitsInfo]="digitsInfo" [noFiat]="true"></app-amount></th> | ||||
|       <th class="fiat"><app-fiat [value]="balance"></app-fiat></th> | ||||
|     </tr> | ||||
|     <tr *ngFor="let address of page"> | ||||
|       <td class="address"> | ||||
|         <app-truncate [text]="address" [lastChars]="8" [link]="['/address/' | relativeUrl, address]" [external]="true"></app-truncate> | ||||
|       </td> | ||||
|       <td class="btc"><app-amount [satoshis]="addresses[address]" [digitsInfo]="digitsInfo" [noFiat]="true"></app-amount></td> | ||||
|       <td class="fiat"><app-fiat [value]="addresses[address]"></app-fiat></td> | ||||
|     </tr> | ||||
|   </table> | ||||
|   <div *ngIf="addressStrings.length > itemsPerPage" class="pagination"> | ||||
|     <ngb-pagination class="pagination-container float-right" [collectionSize]="addressStrings.length" [rotate]="true" [pageSize]="itemsPerPage" [(page)]="pageIndex" (pageChange)="pageChange(pageIndex)" [boundaryLinks]="false" [ellipses]="false"></ngb-pagination> | ||||
|   </div> | ||||
| </div> | ||||
| @ -0,0 +1,101 @@ | ||||
| .frame { | ||||
|   position: relative; | ||||
|   background: #24273e; | ||||
|   padding: 0.5rem; | ||||
|   height: calc(100% + 60px); | ||||
| } | ||||
| 
 | ||||
| .heading { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   justify-content: space-between; | ||||
|   align-items: start; | ||||
| 
 | ||||
|   & > * { | ||||
|     flex-basis: 0; | ||||
|     flex-grow: 1; | ||||
|   } | ||||
| 
 | ||||
|   h3 { | ||||
|     text-align: center; | ||||
|     margin: 0 1em; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| .pagination { | ||||
|   position: absolute; | ||||
|   bottom: 0.5rem; | ||||
|   right: 0.5rem; | ||||
| } | ||||
| 
 | ||||
| .table { | ||||
|   margin-top: 0.5em; | ||||
| 
 | ||||
|   td, th { | ||||
|     padding: 0.15rem 0.5rem; | ||||
| 
 | ||||
|     &.address { | ||||
|       width: auto; | ||||
|     } | ||||
|     &.btc { | ||||
|       width: 140px; | ||||
|       text-align: right; | ||||
|     } | ||||
|     &.fiat { | ||||
|       width: 142px; | ||||
|       text-align: right; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   tr { | ||||
|     border-collapse: collapse; | ||||
| 
 | ||||
|     &:first-child { | ||||
|       border-bottom: solid 1px white; | ||||
|       td, th { | ||||
|         padding-bottom: 0.3rem; | ||||
|       } | ||||
|     } | ||||
|     &:nth-child(2) { | ||||
|       td, th { | ||||
|         padding-top: 0.3rem; | ||||
|       } | ||||
|     } | ||||
|     &:nth-child(even) { | ||||
|       background: #181b2d; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @media (min-width: 528px) { | ||||
|     td, th { | ||||
|       &.btc { | ||||
|         width: 160px; | ||||
|       } | ||||
|       &.fiat { | ||||
|         width: 140px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @media (min-width: 576px) { | ||||
|     td, th { | ||||
|       &.btc { | ||||
|         width: 170px; | ||||
|       } | ||||
|       &.fiat { | ||||
|         width: 140px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   @media (min-width: 992px) { | ||||
|     td, th { | ||||
|       &.btc { | ||||
|         width: 210px; | ||||
|       } | ||||
|       &.fiat { | ||||
|         width: 140px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| @ -0,0 +1,212 @@ | ||||
| import { Component, OnInit, OnDestroy, ChangeDetectorRef, HostListener } from '@angular/core'; | ||||
| import { ActivatedRoute, ParamMap } from '@angular/router'; | ||||
| import { ElectrsApiService } from '../../services/electrs-api.service'; | ||||
| import { switchMap, catchError } from 'rxjs/operators'; | ||||
| import { Address, Transaction } from '../../interfaces/electrs.interface'; | ||||
| import { WebsocketService } from '../../services/websocket.service'; | ||||
| import { StateService } from '../../services/state.service'; | ||||
| import { AudioService } from '../../services/audio.service'; | ||||
| import { ApiService } from '../../services/api.service'; | ||||
| import { of, Subscription, forkJoin } from 'rxjs'; | ||||
| import { SeoService } from '../../services/seo.service'; | ||||
| import { AddressInformation } from '../../interfaces/node-api.interface'; | ||||
| 
 | ||||
| @Component({ | ||||
|   selector: 'app-address-group', | ||||
|   templateUrl: './address-group.component.html', | ||||
|   styleUrls: ['./address-group.component.scss'] | ||||
| }) | ||||
| export class AddressGroupComponent implements OnInit, OnDestroy { | ||||
|   network = ''; | ||||
| 
 | ||||
|   balance = 0; | ||||
|   confirmed = 0; | ||||
|   mempool = 0; | ||||
|   addresses: { [address: string]: number | null }; | ||||
|   addressStrings: string[] = []; | ||||
|   addressInfo: { [address: string]: AddressInformation | null }; | ||||
|   seenTxs: { [txid: string ]: boolean } = {}; | ||||
|   isLoadingAddress = true; | ||||
|   error: any; | ||||
|   mainSubscription: Subscription; | ||||
|   wsSubscription: Subscription; | ||||
| 
 | ||||
|   page: string[] = []; | ||||
|   pageIndex: number = 1; | ||||
|   itemsPerPage: number = 10; | ||||
| 
 | ||||
|   screenSize: 'lg' | 'md' | 'sm' = 'lg'; | ||||
|   digitsInfo: string = '1.8-8'; | ||||
| 
 | ||||
|   constructor( | ||||
|     private route: ActivatedRoute, | ||||
|     private electrsApiService: ElectrsApiService, | ||||
|     private websocketService: WebsocketService, | ||||
|     private stateService: StateService, | ||||
|     private audioService: AudioService, | ||||
|     private apiService: ApiService, | ||||
|     private seoService: SeoService, | ||||
|     private cd: ChangeDetectorRef, | ||||
|   ) { } | ||||
| 
 | ||||
|   ngOnInit(): void { | ||||
|     this.onResize(); | ||||
|     this.stateService.networkChanged$.subscribe((network) => this.network = network); | ||||
|     this.websocketService.want(['blocks']); | ||||
| 
 | ||||
|     this.mainSubscription = this.route.queryParamMap | ||||
|       .pipe( | ||||
|         switchMap((params: ParamMap) => { | ||||
|           this.error = undefined; | ||||
|           this.isLoadingAddress = true; | ||||
|           this.addresses = {}; | ||||
|           this.addressInfo = {}; | ||||
|           this.balance = 0; | ||||
|            | ||||
|           this.addressStrings = params.get('addresses').split(',').map(address => { | ||||
|             if (/^[A-Z]{2,5}1[AC-HJ-NP-Z02-9]{8,100}|04[a-fA-F0-9]{128}|(02|03)[a-fA-F0-9]{64}$/.test(address)) { | ||||
|               return address.toLowerCase(); | ||||
|             } else { | ||||
|               return address; | ||||
|             } | ||||
|           }); | ||||
| 
 | ||||
|           return forkJoin(this.addressStrings.map(address => { | ||||
|             const getLiquidInfo = ((this.stateService.network === 'liquid' || this.stateService.network === 'liquidtestnet') && /^([a-zA-HJ-NP-Z1-9]{26,35}|[a-z]{2,5}1[ac-hj-np-z02-9]{8,100}|[a-km-zA-HJ-NP-Z1-9]{80})$/.test(address)); | ||||
|             return forkJoin([ | ||||
|               of(address), | ||||
|               this.electrsApiService.getAddress$(address), | ||||
|               (getLiquidInfo ? this.apiService.validateAddress$(address) : of(null)), | ||||
|             ]); | ||||
|           })); | ||||
|         }), | ||||
|         catchError(e => { | ||||
|           this.error = e; | ||||
|           return of([]); | ||||
|         }) | ||||
|       ).subscribe((addresses) => { | ||||
|         for (const addressData of addresses) { | ||||
|           const address = addressData[0]; | ||||
|           const addressBalance = addressData[1] as Address; | ||||
|           if (addressBalance) { | ||||
|             this.addresses[address] = addressBalance.chain_stats.funded_txo_sum | ||||
|               + addressBalance.mempool_stats.funded_txo_sum | ||||
|               - addressBalance.chain_stats.spent_txo_sum | ||||
|               - addressBalance.mempool_stats.spent_txo_sum; | ||||
|             this.balance += this.addresses[address]; | ||||
|             this.confirmed += (addressBalance.chain_stats.funded_txo_sum - addressBalance.chain_stats.spent_txo_sum); | ||||
|           } | ||||
|           this.addressInfo[address] = addressData[2] ? addressData[2] as AddressInformation : null; | ||||
|         } | ||||
|         this.websocketService.startTrackAddresses(this.addressStrings); | ||||
|         this.isLoadingAddress = false; | ||||
|         this.pageChange(this.pageIndex); | ||||
|       }); | ||||
| 
 | ||||
|     this.wsSubscription = this.stateService.multiAddressTransactions$.subscribe(update => { | ||||
|       for (const address of Object.keys(update)) { | ||||
|         for (const tx of update[address].mempool) { | ||||
|           this.addTransaction(tx, false, false); | ||||
|         } | ||||
|         for (const tx of update[address].confirmed) { | ||||
|           this.addTransaction(tx, true, false); | ||||
|         } | ||||
|         for (const tx of update[address].removed) { | ||||
|           this.removeTransaction(tx, tx.status.confirmed); | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   pageChange(index): void { | ||||
|     this.page = this.addressStrings.slice((index - 1) * this.itemsPerPage, index * this.itemsPerPage); | ||||
|   } | ||||
| 
 | ||||
|   addTransaction(transaction: Transaction, confirmed = false, playSound: boolean = true): boolean { | ||||
|     if (this.seenTxs[transaction.txid]) { | ||||
|       this.removeTransaction(transaction, false); | ||||
|     } | ||||
|     this.seenTxs[transaction.txid] = true; | ||||
| 
 | ||||
|     let balance = 0; | ||||
|     transaction.vin.forEach((vin) => { | ||||
|       if (this.addressStrings.includes(vin?.prevout?.scriptpubkey_address)) { | ||||
|         this.addresses[vin?.prevout?.scriptpubkey_address] -= vin.prevout.value; | ||||
|         balance -= vin.prevout.value; | ||||
|         this.balance -= vin.prevout.value; | ||||
|         if (confirmed) { | ||||
|           this.confirmed -= vin.prevout.value; | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|     transaction.vout.forEach((vout) => { | ||||
|       if (this.addressStrings.includes(vout?.scriptpubkey_address)) { | ||||
|         this.addresses[vout?.scriptpubkey_address] += vout.value; | ||||
|         balance += vout.value; | ||||
|         this.balance += vout.value; | ||||
|         if (confirmed) { | ||||
|           this.confirmed += vout.value; | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
| 
 | ||||
|     if (playSound) { | ||||
|       if (balance > 0) { | ||||
|         this.audioService.playSound('cha-ching'); | ||||
|       } else { | ||||
|         this.audioService.playSound('chime'); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return true; | ||||
|   } | ||||
| 
 | ||||
|   removeTransaction(transaction: Transaction, confirmed = false): boolean { | ||||
|     transaction.vin.forEach((vin) => { | ||||
|       if (this.addressStrings.includes(vin?.prevout?.scriptpubkey_address)) { | ||||
|         this.addresses[vin?.prevout?.scriptpubkey_address] += vin.prevout.value; | ||||
|         this.balance += vin.prevout.value; | ||||
|         if (confirmed) { | ||||
|           this.confirmed += vin.prevout.value; | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
|     transaction.vout.forEach((vout) => { | ||||
|       if (this.addressStrings.includes(vout?.scriptpubkey_address)) { | ||||
|         this.addresses[vout?.scriptpubkey_address] -= vout.value; | ||||
|         this.balance -= vout.value; | ||||
|         if (confirmed) { | ||||
|           this.confirmed -= vout.value; | ||||
|         } | ||||
|       } | ||||
|     }); | ||||
| 
 | ||||
|     return true; | ||||
|   } | ||||
| 
 | ||||
|   @HostListener('window:resize', ['$event']) | ||||
|   onResize(): void { | ||||
|     if (window.innerWidth >= 992) { | ||||
|       this.screenSize = 'lg'; | ||||
|       this.digitsInfo = '1.8-8'; | ||||
|     } else if (window.innerWidth >= 528) { | ||||
|       this.screenSize = 'md'; | ||||
|       this.digitsInfo = '1.4-4'; | ||||
|     } else { | ||||
|       this.screenSize = 'sm'; | ||||
|       this.digitsInfo = '1.2-2'; | ||||
|     } | ||||
|     const newItemsPerPage = Math.floor((window.innerHeight - 150) / 30); | ||||
|     if (newItemsPerPage !== this.itemsPerPage) { | ||||
|       this.itemsPerPage = newItemsPerPage; | ||||
|       this.pageIndex = 1; | ||||
|       this.pageChange(this.pageIndex); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   ngOnDestroy(): void { | ||||
|     this.mainSubscription.unsubscribe(); | ||||
|     this.wsSubscription.unsubscribe(); | ||||
|     this.websocketService.stopTrackingAddresses(); | ||||
|   } | ||||
| } | ||||
| @ -6,6 +6,7 @@ import { SharedModule } from './shared/shared.module'; | ||||
| 
 | ||||
| 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 { CalculatorComponent } from './components/calculator/calculator.component'; | ||||
| import { BlocksList } from './components/blocks-list/blocks-list.component'; | ||||
|  | ||||
| @ -118,6 +118,7 @@ export class StateService { | ||||
|   mempoolTransactions$ = new Subject<Transaction>(); | ||||
|   mempoolTxPosition$ = new Subject<{ txid: string, position: MempoolPosition, cpfp: CpfpInfo | null}>(); | ||||
|   mempoolRemovedTransactions$ = new Subject<Transaction>(); | ||||
|   multiAddressTransactions$ = new Subject<{ [address: string]: { mempool: Transaction[], confirmed: Transaction[], removed: Transaction[] }}>(); | ||||
|   blockTransactions$ = new Subject<Transaction>(); | ||||
|   isLoadingWebSocket$ = new ReplaySubject<boolean>(1); | ||||
|   isLoadingMempool$ = new BehaviorSubject<boolean>(true); | ||||
|  | ||||
| @ -32,6 +32,7 @@ export class WebsocketService { | ||||
|   private isTrackingRbf: 'all' | 'fullRbf' | false = false; | ||||
|   private isTrackingRbfSummary = false; | ||||
|   private isTrackingAddress: string | false = false; | ||||
|   private isTrackingAddresses: string[] | false = false; | ||||
|   private trackingMempoolBlock: number; | ||||
|   private latestGitCommit = ''; | ||||
|   private onlineCheckTimeout: number; | ||||
| @ -126,6 +127,9 @@ export class WebsocketService { | ||||
|           if (this.isTrackingAddress) { | ||||
|             this.startTrackAddress(this.isTrackingAddress); | ||||
|           } | ||||
|           if (this.isTrackingAddresses) { | ||||
|             this.startTrackAddresses(this.isTrackingAddresses); | ||||
|           } | ||||
|           this.stateService.connectionState$.next(2); | ||||
|         } | ||||
| 
 | ||||
| @ -175,6 +179,16 @@ export class WebsocketService { | ||||
|     this.isTrackingAddress = false; | ||||
|   } | ||||
| 
 | ||||
|   startTrackAddresses(addresses: string[]) { | ||||
|     this.websocketSubject.next({ 'track-addresses': addresses }); | ||||
|     this.isTrackingAddresses = addresses; | ||||
|   } | ||||
| 
 | ||||
|   stopTrackingAddresses() { | ||||
|     this.websocketSubject.next({ 'track-addresses': [] }); | ||||
|     this.isTrackingAddresses = false; | ||||
|   } | ||||
| 
 | ||||
|   startTrackAsset(asset: string) { | ||||
|     this.websocketSubject.next({ 'track-asset': asset }); | ||||
|   } | ||||
| @ -374,6 +388,10 @@ export class WebsocketService { | ||||
|       }); | ||||
|     } | ||||
| 
 | ||||
|     if (response['multi-address-transactions']) { | ||||
|       this.stateService.multiAddressTransactions$.next(response['multi-address-transactions']); | ||||
|     } | ||||
| 
 | ||||
|     if (response['block-transactions']) { | ||||
|       response['block-transactions'].forEach((addressTransaction: Transaction) => { | ||||
|         this.stateService.blockTransactions$.next(addressTransaction); | ||||
|  | ||||
| @ -1,6 +1,6 @@ | ||||
| <span class="truncate" [style.max-width]="maxWidth ? maxWidth + 'px' : null" [style.justify-content]="textAlign" [class.inline]="inline"> | ||||
|     <ng-container *ngIf="link"> | ||||
|       <a [routerLink]="link" class="truncate-link"> | ||||
|       <a [routerLink]="link" class="truncate-link" [target]="external ? '_blank' : ''"> | ||||
|         <ng-container *ngIf="rtl; then rtlTruncated; else ltrTruncated;"></ng-container> | ||||
|       </a> | ||||
|     </ng-container> | ||||
|  | ||||
| @ -9,6 +9,7 @@ import { Component, Input, Inject, LOCALE_ID, ChangeDetectionStrategy } from '@a | ||||
| export class TruncateComponent { | ||||
|   @Input() text: string; | ||||
|   @Input() link: any = null; | ||||
|   @Input() external: boolean = false; | ||||
|   @Input() lastChars: number = 4; | ||||
|   @Input() maxWidth: number = null; | ||||
|   @Input() inline: boolean = false; | ||||
|  | ||||
| @ -46,6 +46,7 @@ import { BlockOverviewGraphComponent } from '../components/block-overview-graph/ | ||||
| import { BlockOverviewTooltipComponent } from '../components/block-overview-tooltip/block-overview-tooltip.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 { SearchFormComponent } from '../components/search-form/search-form.component'; | ||||
| import { AddressLabelsComponent } from '../components/address-labels/address-labels.component'; | ||||
| import { FooterComponent } from '../components/footer/footer.component'; | ||||
| @ -147,6 +148,7 @@ import { OnlyVsizeDirective, OnlyWeightDirective } from './components/weight-dir | ||||
|     BlockFiltersComponent, | ||||
|     TransactionsListComponent, | ||||
|     AddressComponent, | ||||
|     AddressGroupComponent, | ||||
|     SearchFormComponent, | ||||
|     AddressLabelsComponent, | ||||
|     FooterComponent, | ||||
| @ -275,6 +277,7 @@ import { OnlyVsizeDirective, OnlyWeightDirective } from './components/weight-dir | ||||
|     BlockFiltersComponent, | ||||
|     TransactionsListComponent, | ||||
|     AddressComponent, | ||||
|     AddressGroupComponent, | ||||
|     SearchFormComponent, | ||||
|     AddressLabelsComponent, | ||||
|     FooterComponent, | ||||
|  | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user