270 lines
11 KiB
Vue
270 lines
11 KiB
Vue
<template>
|
|
<f7-page ptr @ptr:refresh="update">
|
|
<f7-navbar>
|
|
<f7-nav-left :back-link="$t('Back')"></f7-nav-left>
|
|
<f7-nav-title :title="$t('Exchange Rates Data')"></f7-nav-title>
|
|
<f7-nav-right>
|
|
<f7-link icon-f7="ellipsis" @click="showMoreActionSheet = true"></f7-link>
|
|
</f7-nav-right>
|
|
</f7-navbar>
|
|
|
|
<f7-list strong inset dividers class="margin-vertical" v-if="exchangeRatesData && exchangeRatesData.exchangeRates && exchangeRatesData.exchangeRates.length">
|
|
<f7-list-item
|
|
class="list-item-with-header-and-title list-item-no-item-after"
|
|
:header="$t('Base Currency')"
|
|
smart-select :smart-select-params="{ openIn: 'popup', popupPush: true, closeOnSelect: true, scrollToSelectedItem: true, searchbar: true, searchbarPlaceholder: $t('Currency Name'), searchbarDisableText: $t('Cancel'), appendSearchbarNotFound: $t('No results'), pageTitle: $t('Base Currency'), popupCloseLinkText: $t('Done') }"
|
|
>
|
|
<template #title>
|
|
<div class="no-padding no-margin">
|
|
<span>{{ $t(`currency.${baseCurrency}`) }} </span>
|
|
<small class="smaller">{{ baseCurrency }}</small>
|
|
</div>
|
|
</template>
|
|
<select v-model="baseCurrency">
|
|
<option :value="exchangeRate.currencyCode"
|
|
:key="exchangeRate.currencyCode"
|
|
v-for="exchangeRate in availableExchangeRates">{{ exchangeRate.currencyDisplayName }}</option>
|
|
</select>
|
|
</f7-list-item>
|
|
<f7-list-item
|
|
class="currency-base-amount"
|
|
link="#" no-chevron
|
|
:class="baseAmountFontSizeClass"
|
|
:header="$t('Base Amount')"
|
|
:title="displayBaseAmount"
|
|
@click="showBaseAmountSheet = true"
|
|
>
|
|
<number-pad-sheet :min-value="allowedMinAmount"
|
|
:max-value="allowedMaxAmount"
|
|
v-model:show="showBaseAmountSheet"
|
|
v-model="baseAmount"
|
|
></number-pad-sheet>
|
|
</f7-list-item>
|
|
</f7-list>
|
|
|
|
<f7-list strong inset dividers class="margin-vertical" v-if="!exchangeRatesData || !exchangeRatesData.exchangeRates || !exchangeRatesData.exchangeRates.length">
|
|
<f7-list-item :title="$t('No exchange rates data')"></f7-list-item>
|
|
</f7-list>
|
|
|
|
<f7-list strong inset dividers class="margin-vertical" v-if="exchangeRatesData && exchangeRatesData.exchangeRates && exchangeRatesData.exchangeRates.length">
|
|
<f7-list-item swipeout
|
|
:after="getDisplayConvertedAmount(exchangeRate)"
|
|
:key="exchangeRate.currencyCode" v-for="exchangeRate in availableExchangeRates">
|
|
<template #title>
|
|
<div class="no-padding no-margin">
|
|
<span style="margin-right: 5px">{{ exchangeRate.currencyDisplayName }}</span>
|
|
<small class="smaller">{{ exchangeRate.currencyCode }}</small>
|
|
</div>
|
|
</template>
|
|
<f7-swipeout-actions right>
|
|
<f7-swipeout-button color="primary" close :text="$t('Set As Baseline')" @click="setAsBaseline(exchangeRate.currencyCode, getConvertedAmount(exchangeRate))"></f7-swipeout-button>
|
|
</f7-swipeout-actions>
|
|
</f7-list-item>
|
|
</f7-list>
|
|
|
|
<f7-list strong inset dividers class="margin-vertical" v-if="exchangeRatesData && exchangeRatesData.exchangeRates && exchangeRatesData.exchangeRates.length">
|
|
<f7-list-item v-if="exchangeRatesDataUpdateTime">
|
|
<small>{{ $t('Last Updated') }}</small>
|
|
<small>{{ exchangeRatesDataUpdateTime }}</small>
|
|
</f7-list-item>
|
|
<f7-list-item>
|
|
<small>{{ $t('Data source') }}</small>
|
|
<small>
|
|
<f7-link external target="_blank" :href="exchangeRatesData.referenceUrl" v-if="exchangeRatesData.referenceUrl">{{ exchangeRatesData.dataSource }}</f7-link>
|
|
<span v-else-if="!exchangeRatesData.referenceUrl">{{ exchangeRatesData.dataSource }}</span>
|
|
</small>
|
|
</f7-list-item>
|
|
</f7-list>
|
|
|
|
<f7-actions close-by-outside-click close-on-escape :opened="showMoreActionSheet" @actions:closed="showMoreActionSheet = false">
|
|
<f7-actions-group>
|
|
<f7-actions-button :class="{ 'disabled': updating }" @click="update(null)">
|
|
<span>{{ $t('Update') }}</span>
|
|
</f7-actions-button>
|
|
</f7-actions-group>
|
|
<f7-actions-group>
|
|
<f7-actions-button bold close>{{ $t('Cancel') }}</f7-actions-button>
|
|
</f7-actions-group>
|
|
</f7-actions>
|
|
</f7-page>
|
|
</template>
|
|
|
|
<script>
|
|
import { mapStores } from 'pinia';
|
|
import { useUserStore } from '@/stores/user.js';
|
|
import { useExchangeRatesStore } from '@/stores/exchangeRates.js';
|
|
|
|
import transactionConstants from '@/consts/transaction.js';
|
|
import { isNumber, appendThousandsSeparator } from '@/lib/common.js';
|
|
import { stringCurrencyToNumeric, getExchangedAmount } from '@/lib/currency.js';
|
|
|
|
export default {
|
|
data() {
|
|
const userStore = useUserStore();
|
|
|
|
return {
|
|
baseCurrency: userStore.currentUserDefaultCurrency,
|
|
baseAmount: 100,
|
|
updating: false,
|
|
showMoreActionSheet: false,
|
|
showBaseAmountSheet: false
|
|
};
|
|
},
|
|
computed: {
|
|
...mapStores(useUserStore, useExchangeRatesStore),
|
|
exchangeRatesData() {
|
|
return this.exchangeRatesStore.latestExchangeRates.data;
|
|
},
|
|
exchangeRatesDataUpdateTime() {
|
|
const exchangeRatesLastUpdateTime = this.exchangeRatesStore.exchangeRatesLastUpdateTime;
|
|
return exchangeRatesLastUpdateTime ? this.$locale.formatUnixTimeToLongDate(this.userStore, exchangeRatesLastUpdateTime) : '';
|
|
},
|
|
availableExchangeRates() {
|
|
if (!this.exchangeRatesData || !this.exchangeRatesData.exchangeRates) {
|
|
return [];
|
|
}
|
|
|
|
const availableExchangeRates = [];
|
|
|
|
for (let i = 0; i < this.exchangeRatesData.exchangeRates.length; i++) {
|
|
const exchangeRate = this.exchangeRatesData.exchangeRates[i];
|
|
|
|
availableExchangeRates.push({
|
|
currencyCode: exchangeRate.currency,
|
|
currencyDisplayName: this.$t(`currency.${exchangeRate.currency}`),
|
|
rate: exchangeRate.rate
|
|
});
|
|
}
|
|
|
|
availableExchangeRates.sort(function(c1, c2) {
|
|
return c1.currencyDisplayName.localeCompare(c2.currencyDisplayName);
|
|
})
|
|
|
|
return availableExchangeRates;
|
|
},
|
|
displayBaseAmount() {
|
|
return this.$locale.getDisplayCurrency(this.baseAmount);
|
|
},
|
|
baseAmountFontSizeClass() {
|
|
if (this.baseAmount >= 100000000 || this.baseAmount <= -100000000) {
|
|
return 'ebk-small-amount';
|
|
} else if (this.baseAmount >= 1000000 || this.baseAmount <= -1000000) {
|
|
return 'ebk-normal-amount';
|
|
} else {
|
|
return 'ebk-large-amount';
|
|
}
|
|
},
|
|
allowedMinAmount() {
|
|
return transactionConstants.minAmount;
|
|
},
|
|
allowedMaxAmount() {
|
|
return transactionConstants.maxAmount;
|
|
}
|
|
},
|
|
created() {
|
|
if (!this.exchangeRatesData || !this.exchangeRatesData.exchangeRates) {
|
|
return;
|
|
}
|
|
|
|
for (let i = 0; i < this.exchangeRatesData.exchangeRates.length; i++) {
|
|
const exchangeRate = this.exchangeRatesData.exchangeRates[i];
|
|
if (exchangeRate.currency === this.baseCurrency) {
|
|
return;
|
|
}
|
|
}
|
|
|
|
this.$toast('There is no exchange rates data for your default currency');
|
|
},
|
|
methods: {
|
|
update(done) {
|
|
const self = this;
|
|
|
|
if (self.updating) {
|
|
if (done) {
|
|
done();
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
self.updating = true;
|
|
|
|
if (!done) {
|
|
self.$showLoading();
|
|
}
|
|
|
|
self.exchangeRatesStore.getLatestExchangeRates({
|
|
silent: false,
|
|
force: true
|
|
}).then(() => {
|
|
if (done) {
|
|
done();
|
|
}
|
|
|
|
self.updating = false;
|
|
self.$hideLoading();
|
|
|
|
self.$toast('Exchange rates data has been updated');
|
|
}).catch(error => {
|
|
if (done) {
|
|
done();
|
|
}
|
|
|
|
self.updating = false;
|
|
self.$hideLoading();
|
|
|
|
if (!error.processed) {
|
|
self.$toast(error.message || error);
|
|
}
|
|
});
|
|
},
|
|
getConvertedAmount(toExchangeRate) {
|
|
const fromExchangeRate = this.exchangeRatesStore.latestExchangeRateMap[this.baseCurrency];
|
|
|
|
if (!fromExchangeRate) {
|
|
return '';
|
|
}
|
|
|
|
return getExchangedAmount(this.baseAmount / 100, fromExchangeRate.rate, toExchangeRate.rate);
|
|
},
|
|
getDisplayConvertedAmount(toExchangeRate) {
|
|
const rateStr = this.getConvertedAmount(toExchangeRate).toString();
|
|
|
|
if (rateStr.indexOf('.') < 0) {
|
|
return appendThousandsSeparator(rateStr);
|
|
} else {
|
|
let firstNonZeroPos = 0;
|
|
|
|
for (let i = 0; i < rateStr.length; i++) {
|
|
if (rateStr.charAt(i) !== '.' && rateStr.charAt(i) !== '0') {
|
|
firstNonZeroPos = Math.min(i + 4, rateStr.length);
|
|
break;
|
|
}
|
|
}
|
|
|
|
const trimmedRateStr = rateStr.substring(0, Math.max(6, Math.max(firstNonZeroPos, rateStr.indexOf('.') + 2)));
|
|
return appendThousandsSeparator(trimmedRateStr);
|
|
}
|
|
},
|
|
setAsBaseline(currency, amount) {
|
|
if (!isNumber(amount)) {
|
|
amount = '';
|
|
}
|
|
|
|
this.baseCurrency = currency;
|
|
this.baseAmount = stringCurrencyToNumeric(amount.toString());
|
|
}
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<style>
|
|
.currency-base-amount {
|
|
line-height: 53px;
|
|
}
|
|
|
|
.currency-base-amount .item-header {
|
|
padding-top: calc(var(--f7-typography-padding) / 2);
|
|
}
|
|
</style>
|