diff --git a/frontend/proxy.conf.local-esplora.js b/frontend/proxy.conf.local-esplora.js
index 8bb57e623..a7137f3bc 100644
--- a/frontend/proxy.conf.local-esplora.js
+++ b/frontend/proxy.conf.local-esplora.js
@@ -112,6 +112,14 @@ PROXY_CONFIG.push(...[
"^/testnet": ""
},
},
+ {
+ context: ['/api/v1/services/**'],
+ target: `http://localhost:9000`,
+ secure: false,
+ ws: true,
+ changeOrigin: true,
+ proxyTimeout: 30000,
+ },
{
context: ['/api/v1/**'],
target: `http://127.0.0.1:8999`,
diff --git a/frontend/proxy.conf.local.js b/frontend/proxy.conf.local.js
index b2fb1bb27..3a502e0ed 100644
--- a/frontend/proxy.conf.local.js
+++ b/frontend/proxy.conf.local.js
@@ -112,6 +112,14 @@ PROXY_CONFIG.push(...[
"^/testnet": ""
},
},
+ {
+ context: ['/api/v1/services/**'],
+ target: `http://localhost:9000`,
+ secure: false,
+ ws: true,
+ changeOrigin: true,
+ proxyTimeout: 30000,
+ },
{
context: ['/api/v1/**'],
target: `http://localhost:8999`,
diff --git a/frontend/proxy.conf.mixed.js b/frontend/proxy.conf.mixed.js
index c0c7157ff..76bb06607 100644
--- a/frontend/proxy.conf.mixed.js
+++ b/frontend/proxy.conf.mixed.js
@@ -95,6 +95,14 @@ if (configContent && configContent.BASE_MODULE === 'bisq') {
}
PROXY_CONFIG.push(...[
+ {
+ context: ['/api/v1/services/**'],
+ target: `http://localhost:9000`,
+ secure: false,
+ ws: true,
+ changeOrigin: true,
+ proxyTimeout: 30000,
+ },
{
context: ['/api/v1/**'],
target: `http://localhost:8999`,
diff --git a/frontend/src/app/components/about/about.component.html b/frontend/src/app/components/about/about.component.html
index 261f72f55..6bff838aa 100644
--- a/frontend/src/app/components/about/about.component.html
+++ b/frontend/src/app/components/about/about.component.html
@@ -4,7 +4,8 @@
®
diff --git a/frontend/src/app/components/about/about.component.scss b/frontend/src/app/components/about/about.component.scss
index 2a5710ca1..f7aa0b965 100644
--- a/frontend/src/app/components/about/about.component.scss
+++ b/frontend/src/app/components/about/about.component.scss
@@ -22,6 +22,7 @@
.intro {
margin: 25px auto 30px;
+ margin-top: 25px;
width: 250px;
display: flex;
flex-direction: column;
diff --git a/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.html b/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.html
new file mode 100644
index 000000000..fe0718ecc
--- /dev/null
+++ b/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.html
@@ -0,0 +1,21 @@
+
+
+
+
+
+
+
+ {{ bar.label }}
+
+
+
+
+
+
+
{{ bar.class === 'tx' ? '' : '+' }} {{ bar.fee | number }} sat
+
+
+
+
+
+
diff --git a/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.scss b/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.scss
new file mode 100644
index 000000000..6137b53ee
--- /dev/null
+++ b/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.scss
@@ -0,0 +1,157 @@
+.fee-graph {
+ height: 100%;
+ min-width: 120px;
+ width: 120px;
+ max-height: 90vh;
+ margin-left: 4em;
+ margin-right: 1.5em;
+ padding-bottom: 63px;
+
+ .column {
+ width: 100%;
+ height: 100%;
+ position: relative;
+ background: #181b2d;
+
+ .bar {
+ position: absolute;
+ bottom: 0;
+ left: 0;
+ right: 0;
+ display: flex;
+ flex-direction: column;
+ justify-content: center;
+ align-items: center;
+
+ .fill {
+ position: absolute;
+ left: 0;
+ right: 0;
+ top: 0;
+ bottom: 0;
+ opacity: 0.75;
+ pointer-events: none;
+ }
+
+ .fee {
+ font-size: 0.9em;
+ opacity: 0;
+ pointer-events: none;
+ }
+
+ .spacer {
+ width: 100%;
+ height: 1px;
+ flex-grow: 1;
+ pointer-events: none;
+ }
+
+ .line {
+ position: absolute;
+ right: 0;
+ top: 0;
+ left: -4.5em;
+ border-top: dashed white 1.5px;
+
+ .fee-rate {
+ width: 100%;
+ position: absolute;
+ left: 0;
+ right: 0.2em;
+ font-size: 0.8em;
+ display: flex;
+ flex-direction: row-reverse;
+ justify-content: space-between;
+ margin: 0;
+
+ .label {
+ margin-right: .2em;
+ }
+
+ .rate .symbol {
+ color: white;
+ }
+ }
+ }
+
+ &.tx {
+ .fill {
+ background: #3bcc49;
+ }
+ .line {
+ .fee-rate {
+ top: 0;
+ }
+ }
+ .fee {
+ position: absolute;
+ opacity: 1;
+ z-index: 11;
+ }
+ }
+
+ &.target {
+ .fill {
+ background: #653b9c;
+ }
+ .fee {
+ position: absolute;
+ opacity: 1;
+ z-index: 11;
+ }
+ .line .fee-rate {
+ bottom: 2px;
+ }
+ }
+
+ &.max {
+ cursor: pointer;
+ .line .fee-rate {
+ .label {
+ opacity: 0;
+ }
+ bottom: 2px;
+ }
+ &.active, &:hover {
+ .fill {
+ background: #105fb0;
+ }
+ .line {
+ .fee-rate .label {
+ opacity: 1;
+ }
+ }
+ }
+ }
+
+ &:hover {
+ .fill {
+ z-index: 10;
+ }
+ .line {
+ z-index: 11;
+ }
+ .fee {
+ opacity: 1;
+ z-index: 12;
+ }
+ }
+ }
+
+ &:hover > .bar:not(:hover) {
+ &.target, &.max {
+ .fee {
+ opacity: 0;
+ }
+ .line .fee-rate .label {
+ opacity: 0;
+ }
+ }
+ &.max {
+ .fill {
+ background: none;
+ }
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.ts b/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.ts
new file mode 100644
index 000000000..4d746a0d9
--- /dev/null
+++ b/frontend/src/app/components/accelerate-preview/accelerate-fee-graph.component.ts
@@ -0,0 +1,96 @@
+import { Component, OnInit, Input, Output, OnChanges, EventEmitter, HostListener, Inject, LOCALE_ID } from '@angular/core';
+import { StateService } from '../../services/state.service';
+import { Outspend, Transaction, Vin, Vout } from '../../interfaces/electrs.interface';
+import { Router } from '@angular/router';
+import { ReplaySubject, merge, Subscription, of } from 'rxjs';
+import { tap, switchMap } from 'rxjs/operators';
+import { ApiService } from '../../services/api.service';
+import { AccelerationEstimate, RateOption } from './accelerate-preview.component';
+
+interface GraphBar {
+ rate: number;
+ style: any;
+ class: 'tx' | 'target' | 'max';
+ label: string;
+ active?: boolean;
+ rateIndex?: number;
+ fee?: number;
+}
+
+@Component({
+ selector: 'app-accelerate-fee-graph',
+ templateUrl: './accelerate-fee-graph.component.html',
+ styleUrls: ['./accelerate-fee-graph.component.scss'],
+})
+export class AccelerateFeeGraphComponent implements OnInit, OnChanges {
+ @Input() tx: Transaction;
+ @Input() estimate: AccelerationEstimate;
+ @Input() maxRateOptions: RateOption[] = [];
+ @Input() maxRateIndex: number = 0;
+ @Output() setUserBid = new EventEmitter<{ fee: number, index: number }>();
+
+ bars: GraphBar[] = [];
+ tooltipPosition = { x: 0, y: 0 };
+
+ ngOnInit(): void {
+ this.initGraph();
+ }
+
+ ngOnChanges(): void {
+ this.initGraph();
+ }
+
+ initGraph(): void {
+ if (!this.tx || !this.estimate) {
+ return;
+ }
+ const maxRate = Math.max(...this.maxRateOptions.map(option => option.rate));
+ const baseRate = this.estimate.txSummary.effectiveFee / this.estimate.txSummary.effectiveVsize;
+ const baseHeight = baseRate / maxRate;
+ const bars: GraphBar[] = this.maxRateOptions.slice().reverse().map(option => {
+ return {
+ rate: option.rate,
+ style: this.getStyle(option.rate, maxRate, baseHeight),
+ class: 'max',
+ label: 'maximum',
+ active: option.index === this.maxRateIndex,
+ rateIndex: option.index,
+ fee: option.fee,
+ }
+ });
+ bars.push({
+ rate: this.estimate.targetFeeRate,
+ style: this.getStyle(this.estimate.targetFeeRate, maxRate, baseHeight),
+ class: 'target',
+ label: 'next block',
+ fee: this.estimate.nextBlockFee - this.estimate.txSummary.effectiveFee
+ });
+ bars.push({
+ rate: baseRate,
+ style: this.getStyle(baseRate, maxRate, 0),
+ class: 'tx',
+ label: '',
+ fee: this.estimate.txSummary.effectiveFee,
+ });
+ this.bars = bars;
+ }
+
+ getStyle(rate, maxRate, base) {
+ const top = (rate / maxRate);
+ return {
+ height: `${(top - base) * 100}%`,
+ bottom: base ? `${base * 100}%` : '0',
+ }
+ }
+
+ onClick(event, bar): void {
+ if (bar.rateIndex != null) {
+ this.setUserBid.emit({ fee: bar.fee, index: bar.rateIndex });
+ }
+ }
+
+ @HostListener('pointermove', ['$event'])
+ onPointerMove(event) {
+ this.tooltipPosition = { x: event.offsetX, y: event.offsetY };
+ }
+}
diff --git a/frontend/src/app/components/accelerate-preview/accelerate-preview.component.html b/frontend/src/app/components/accelerate-preview/accelerate-preview.component.html
new file mode 100644
index 000000000..9bb66eda1
--- /dev/null
+++ b/frontend/src/app/components/accelerate-preview/accelerate-preview.component.html
@@ -0,0 +1,262 @@
+
+
+
+ Transaction has now been submitted to mining pools for acceleration. You can track the progress
here .
+
+
+
+
+
+
+
+
+
+
+
+
+
+
Your transaction
+
+
+
+ Plus {{ estimate.txSummary.ancestorCount - 1 }} unconfirmed ancestor{{ estimate.txSummary.ancestorCount > 2 ? 's' : ''}}.
+
+
+
+
+
+ Virtual size
+
+
+
+
+
+ Size in vbytes of this transaction and its unconfirmed ancestors
+
+
+
+
+ In-band fees
+
+
+ {{ estimate.txSummary.effectiveFee | number : '1.0-0' }} sats
+
+
+
+
+ Fees already paid by this transaction and its unconfirmed ancestors
+
+
+
+
+
+
+
+
How much more are you willing to pay?
+
+
+
+ Choose the maximum extra transaction fee you're willing to pay to get into the next block.
+ If the estimated next block rate rises beyond this limit, we will automatically cancel your acceleration request.
+
+
+
+
+
+
Acceleration summary
+
+
+
+
+ Estimated cost
+
+
+ Maximum cost
+
+
+
+
+
+
+
+
+ Next block market rate
+
+
+ {{ estimate.targetFeeRate | number : '1.0-0' }}
+
+ sat/vB
+
+
+
+ Estimated extra fee required
+
+
+ {{ math.max(0, estimate.nextBlockFee - estimate.txSummary.effectiveFee) | number }}
+
+
+ sats
+
+
+
+
+
+
+
+
+ Your maximum
+
+
+ ~{{ ((estimate.txSummary.effectiveFee + userBid) / estimate.txSummary.effectiveVsize) | number : '1.0-0' }}
+
+ sat/vB
+
+
+
+ The maximum extra transaction fee you could pay
+
+
+
+ {{ userBid | number }}
+
+
+
+ sats
+
+
+
+
+
+
+
+
+ Mempool Acceleratorâ„¢ fees
+
+
+
+
+ mempool.space fee
+
+
+ +{{ estimate.mempoolBaseFee | number }}
+
+
+ sats
+
+
+
+
+
+ Transaction vsize fee
+
+
+ +{{ estimate.vsizeFee | number }}
+
+
+ sats
+
+
+
+
+
+
+
+
+ Estimated acceleration cost
+
+
+
+ {{ estimate.cost + estimate.mempoolBaseFee + estimate.vsizeFee | number }}
+
+
+
+ sats
+
+
+
+
+
+ If your tx is accelerated to {{ estimate.targetFeeRate | number : '1.0-0' }} sat/vB
+
+
+
+
+
+
+
+
+ Maximum acceleration cost
+
+
+
+ {{ maxCost | number }}
+
+
+
+ sats
+
+
+
+
+
+
+
+ If your tx is accelerated to ~{{ ((estimate.txSummary.effectiveFee + userBid) / estimate.txSummary.effectiveVsize) | number : '1.0-0' }} sat/vB
+
+
+
+
+
+
+
+
+ Available balance
+
+
+ {{ estimate.userBalance | number }}
+
+
+ sats
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/frontend/src/app/components/accelerate-preview/accelerate-preview.component.scss b/frontend/src/app/components/accelerate-preview/accelerate-preview.component.scss
new file mode 100644
index 000000000..433c05520
--- /dev/null
+++ b/frontend/src/app/components/accelerate-preview/accelerate-preview.component.scss
@@ -0,0 +1,88 @@
+.fee-card {
+ padding: 15px;
+ background-color: #1d1f31;
+
+ .feerate {
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ justify-content: center;
+
+ .fee {
+ font-size: 1.2em;
+ }
+ .rate {
+ font-size: 0.9em;
+ .symbol {
+ color: white;
+ }
+ }
+ }
+}
+
+.btn-border {
+ border: solid 1px black;
+ background-color: #0c4a87;
+}
+
+.feerate.active {
+ background-color: #105fb0 !important;
+ opacity: 1;
+ border: 1px solid white !important;
+}
+
+.estimateDisabled {
+ opacity: 0.5;
+ pointer-events: none;
+}
+
+.table-toggle {
+ width: 100%;
+ margin-top: 0.5em;
+}
+
+.table-accelerator {
+ tr {
+ text-wrap: wrap;
+
+ td {
+ padding-top: 0;
+ padding-bottom: 0;
+ vertical-align: baseline;
+ }
+
+ &.group-first {
+ td {
+ padding-top: 0.75rem;
+ }
+ }
+ &.group-last {
+ td {
+ padding-bottom: 0.75rem;
+ }
+ }
+ }
+ td {
+ &:first-child {
+ width: 100vw;
+ }
+ &.info {
+ color: #6c757d;
+ }
+ &.amt {
+ text-align: right;
+ padding-right: 0.2em;
+ }
+ &.units {
+ padding-left: 0.2em;
+ white-space: nowrap;
+ }
+ }
+}
+
+.accelerate-cols {
+ display: flex;
+ flex-direction: row;
+ align-items: stretch;
+ margin-top: 1em;
+}
\ No newline at end of file
diff --git a/frontend/src/app/components/accelerate-preview/accelerate-preview.component.ts b/frontend/src/app/components/accelerate-preview/accelerate-preview.component.ts
new file mode 100644
index 000000000..1c356a80b
--- /dev/null
+++ b/frontend/src/app/components/accelerate-preview/accelerate-preview.component.ts
@@ -0,0 +1,205 @@
+import { Component, OnInit, Input, OnDestroy, OnChanges, SimpleChanges, HostListener } from '@angular/core';
+import { ApiService } from '../../services/api.service';
+import { Subscription, catchError, of, tap } from 'rxjs';
+import { StorageService } from '../../services/storage.service';
+import { Transaction } from '../../interfaces/electrs.interface';
+import { nextRoundNumber } from '../../shared/common.utils';
+
+export type AccelerationEstimate = {
+ txSummary: TxSummary;
+ nextBlockFee: number;
+ targetFeeRate: number;
+ userBalance: number;
+ enoughBalance: boolean;
+ cost: number;
+ mempoolBaseFee: number;
+ vsizeFee: number;
+}
+export type TxSummary = {
+ txid: string; // txid of the current transaction
+ effectiveVsize: number; // Total vsize of the dependency tree
+ effectiveFee: number; // Total fee of the dependency tree in sats
+ ancestorCount: number; // Number of ancestors
+}
+
+export interface RateOption {
+ fee: number;
+ rate: number;
+ index: number;
+}
+
+export const MIN_BID_RATIO = 1;
+export const DEFAULT_BID_RATIO = 2;
+export const MAX_BID_RATIO = 4;
+
+@Component({
+ selector: 'app-accelerate-preview',
+ templateUrl: 'accelerate-preview.component.html',
+ styleUrls: ['accelerate-preview.component.scss']
+})
+export class AcceleratePreviewComponent implements OnInit, OnDestroy, OnChanges {
+ @Input() tx: Transaction | undefined;
+ @Input() scrollEvent: boolean;
+
+ math = Math;
+ error = '';
+ showSuccess = false;
+ estimateSubscription: Subscription;
+ accelerationSubscription: Subscription;
+ estimate: any;
+ hasAncestors: boolean = false;
+ minExtraCost = 0;
+ minBidAllowed = 0;
+ maxBidAllowed = 0;
+ defaultBid = 0;
+ maxCost = 0;
+ userBid = 0;
+ selectFeeRateIndex = 1;
+ showTable: 'estimated' | 'maximum' = 'maximum';
+ isMobile: boolean = window.innerWidth <= 767.98;
+
+ maxRateOptions: RateOption[] = [];
+
+ constructor(
+ private apiService: ApiService,
+ private storageService: StorageService
+ ) { }
+
+ ngOnDestroy(): void {
+ if (this.estimateSubscription) {
+ this.estimateSubscription.unsubscribe();
+ }
+ }
+
+ ngOnChanges(changes: SimpleChanges): void {
+ if (changes.scrollEvent) {
+ this.scrollToPreview('acceleratePreviewAnchor', 'center');
+ }
+ }
+
+ ngOnInit() {
+ this.estimateSubscription = this.apiService.estimate$(this.tx.txid).pipe(
+ tap((response) => {
+ if (response.status === 204) {
+ this.estimate = undefined;
+ this.error = `cannot_accelerate_tx`;
+ this.scrollToPreviewWithTimeout('mempoolError', 'center');
+ this.estimateSubscription.unsubscribe();
+ } else {
+ this.estimate = response.body;
+ if (!this.estimate) {
+ this.error = `cannot_accelerate_tx`;
+ this.scrollToPreviewWithTimeout('mempoolError', 'center');
+ this.estimateSubscription.unsubscribe();
+ }
+
+ if (this.estimate.userBalance <= 0) {
+ if (this.isLoggedIn()) {
+ this.error = `not_enough_balance`;
+ this.scrollToPreviewWithTimeout('mempoolError', 'center');
+ }
+ }
+
+ this.hasAncestors = this.estimate.txSummary.ancestorCount > 1;
+
+ // Make min extra fee at least 50% of the current tx fee
+ this.minExtraCost = nextRoundNumber(Math.max(this.estimate.cost * 2, this.estimate.txSummary.effectiveFee));
+
+ this.maxRateOptions = [1, 2, 4].map((multiplier, index) => {
+ return {
+ fee: this.minExtraCost * multiplier,
+ rate: (this.estimate.txSummary.effectiveFee + (this.minExtraCost * multiplier)) / this.estimate.txSummary.effectiveVsize,
+ index,
+ };
+ });
+
+ this.minBidAllowed = this.minExtraCost * MIN_BID_RATIO;
+ this.defaultBid = this.minExtraCost * DEFAULT_BID_RATIO;
+ this.maxBidAllowed = this.minExtraCost * MAX_BID_RATIO;
+
+ this.userBid = this.defaultBid;
+ if (this.userBid < this.minBidAllowed) {
+ this.userBid = this.minBidAllowed;
+ } else if (this.userBid > this.maxBidAllowed) {
+ this.userBid = this.maxBidAllowed;
+ }
+ this.maxCost = this.userBid + this.estimate.mempoolBaseFee + this.estimate.vsizeFee;
+
+ if (!this.error) {
+ this.scrollToPreview('acceleratePreviewAnchor', 'center');
+ }
+ }
+ }),
+ catchError((response) => {
+ this.estimate = undefined;
+ this.error = response.error;
+ this.scrollToPreviewWithTimeout('mempoolError', 'center');
+ this.estimateSubscription.unsubscribe();
+ return of(null);
+ })
+ ).subscribe();
+ }
+
+ /**
+ * User changed his bid
+ */
+ setUserBid({ fee, index }: { fee: number, index: number}) {
+ if (this.estimate) {
+ this.selectFeeRateIndex = index;
+ this.userBid = Math.max(0, fee);
+ this.maxCost = this.userBid + this.estimate.mempoolBaseFee + this.estimate.vsizeFee;
+ }
+ }
+
+ /**
+ * Scroll to element id with or without setTimeout
+ */
+ scrollToPreviewWithTimeout(id: string, position: ScrollLogicalPosition) {
+ setTimeout(() => {
+ this.scrollToPreview(id, position);
+ }, 100);
+ }
+ scrollToPreview(id: string, position: ScrollLogicalPosition) {
+ const acceleratePreviewAnchor = document.getElementById(id);
+ if (acceleratePreviewAnchor) {
+ acceleratePreviewAnchor.scrollIntoView({
+ behavior: 'smooth',
+ inline: position,
+ block: position,
+ });
+ }
+}
+
+ /**
+ * Send acceleration request
+ */
+ accelerate() {
+ if (this.accelerationSubscription) {
+ this.accelerationSubscription.unsubscribe();
+ }
+ this.accelerationSubscription = this.apiService.accelerate$(
+ this.tx.txid,
+ this.userBid
+ ).subscribe({
+ next: () => {
+ this.showSuccess = true;
+ this.scrollToPreviewWithTimeout('successAlert', 'center');
+ this.estimateSubscription.unsubscribe();
+ },
+ error: (response) => {
+ this.error = response.error;
+ this.scrollToPreviewWithTimeout('mempoolError', 'center');
+ }
+ });
+ }
+
+ isLoggedIn() {
+ const auth = this.storageService.getAuth();
+ return auth !== null;
+ }
+
+ @HostListener('window:resize', ['$event'])
+ onResize(): void {
+ this.isMobile = window.innerWidth <= 767.98;
+ }
+}
\ No newline at end of file
diff --git a/frontend/src/app/components/blockchain/blockchain.component.ts b/frontend/src/app/components/blockchain/blockchain.component.ts
index 7619587d8..56b7c39e6 100644
--- a/frontend/src/app/components/blockchain/blockchain.component.ts
+++ b/frontend/src/app/components/blockchain/blockchain.component.ts
@@ -1,4 +1,4 @@
-import { Component, OnInit, OnDestroy, ChangeDetectionStrategy, Input, Output, EventEmitter, HostListener, ChangeDetectorRef } from '@angular/core';
+import { Component, OnInit, OnDestroy, ChangeDetectionStrategy, Input, Output, EventEmitter, HostListener, ChangeDetectorRef, OnChanges, SimpleChanges } from '@angular/core';
import { firstValueFrom, Subscription } from 'rxjs';
import { StateService } from '../../services/state.service';
@@ -8,12 +8,13 @@ import { StateService } from '../../services/state.service';
styleUrls: ['./blockchain.component.scss'],
changeDetection: ChangeDetectionStrategy.OnPush,
})
-export class BlockchainComponent implements OnInit, OnDestroy {
+export class BlockchainComponent implements OnInit, OnDestroy, OnChanges {
@Input() pages: any[] = [];
@Input() pageIndex: number;
@Input() blocksPerPage: number = 8;
@Input() minScrollWidth: number = 0;
@Input() scrollableMempool: boolean = false;
+ @Input() containerWidth: number;
@Output() mempoolOffsetChange: EventEmitter = new EventEmitter();
@@ -85,19 +86,25 @@ export class BlockchainComponent implements OnInit, OnDestroy {
this.mempoolOffsetChange.emit(this.mempoolOffset);
}
- @HostListener('window:resize', ['$event'])
+ ngOnChanges(changes: SimpleChanges): void {
+ if (changes.containerWidth) {
+ this.onResize();
+ }
+ }
+
onResize(): void {
- if (window.innerWidth >= 768) {
+ const width = this.containerWidth || window.innerWidth;
+ if (width >= 768) {
if (this.stateService.isLiquid()) {
this.dividerOffset = 420;
} else {
- this.dividerOffset = window.innerWidth * 0.5;
+ this.dividerOffset = width * 0.5;
}
} else {
if (this.stateService.isLiquid()) {
- this.dividerOffset = window.innerWidth * 0.5;
+ this.dividerOffset = width * 0.5;
} else {
- this.dividerOffset = window.innerWidth * 0.95;
+ this.dividerOffset = width * 0.95;
}
}
this.cd.markForCheck();
diff --git a/frontend/src/app/components/fiat-selector/fiat-selector.component.html b/frontend/src/app/components/fiat-selector/fiat-selector.component.html
index eec6f4b0a..4fa55deb9 100644
--- a/frontend/src/app/components/fiat-selector/fiat-selector.component.html
+++ b/frontend/src/app/components/fiat-selector/fiat-selector.component.html
@@ -1,5 +1,5 @@
-
- {{ currency[1].name + " (" + currency[1].code + ")" }}
+
+ {{ currency[1].code }}
diff --git a/frontend/src/app/components/language-selector/language-selector.component.html b/frontend/src/app/components/language-selector/language-selector.component.html
index 41e0efb0e..bfd36af77 100644
--- a/frontend/src/app/components/language-selector/language-selector.component.html
+++ b/frontend/src/app/components/language-selector/language-selector.component.html
@@ -1,5 +1,5 @@
-
+
{{ lang.name }}
diff --git a/frontend/src/app/components/master-page/master-page.component.html b/frontend/src/app/components/master-page/master-page.component.html
index f12fbc960..8863b335f 100644
--- a/frontend/src/app/components/master-page/master-page.component.html
+++ b/frontend/src/app/components/master-page/master-page.component.html
@@ -1,6 +1,20 @@
-