init
This commit is contained in:
commit
44f31f8b9f
402 changed files with 47865 additions and 0 deletions
526
components/withdraw/Withdraw.vue
Normal file
526
components/withdraw/Withdraw.vue
Normal file
|
@ -0,0 +1,526 @@
|
|||
<template>
|
||||
<b-tab-item :label="$t('withdraw')">
|
||||
<div class="field">
|
||||
<div class="label-with-buttons">
|
||||
<div class="label">
|
||||
{{ $t('note') }}
|
||||
<b-tooltip :label="$t('noteTooltip')" size="is-small" position="is-right" multilined>
|
||||
<button class="button is-primary has-icon">
|
||||
<span class="icon icon-info"></span>
|
||||
</button>
|
||||
</b-tooltip>
|
||||
</div>
|
||||
<a
|
||||
v-show="!hasErrorNote && depositTxHash"
|
||||
:href="txExplorerUrl(depositTxHash)"
|
||||
target="_blank"
|
||||
class="button is-icon"
|
||||
>
|
||||
<b-tooltip
|
||||
:label="$t('depositTransactionOnEtherscan')"
|
||||
size="is-small"
|
||||
position="is-left"
|
||||
multilined
|
||||
>
|
||||
<LinkIcon />
|
||||
</b-tooltip>
|
||||
</a>
|
||||
<button v-show="shouldSettingsShow" class="button is-icon" @click="onSettings">
|
||||
<b-tooltip :label="$t('withdrawalSettings')" size="is-small" position="is-right" multilined>
|
||||
<SettingsIcon />
|
||||
</b-tooltip>
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<b-input
|
||||
v-model="withdrawNote"
|
||||
:placeholder="$t('pleaseEnterYourNote')"
|
||||
:custom-class="hasErrorNote ? hasErrorNote.type : 'is-primary'"
|
||||
></b-input>
|
||||
<div v-if="hasErrorNote" class="help" :class="hasErrorNote.type">
|
||||
<!-- eslint-disable vue/no-v-html -->
|
||||
<p v-html="hasErrorNote.msg"></p>
|
||||
</div>
|
||||
</div>
|
||||
<div v-if="!hasErrorNote && depositTxHash" class="field field-withdraw">
|
||||
<div class="withdraw-data">
|
||||
<div class="withdraw-data-item">
|
||||
{{ $t('amount') }} <span>{{ selectedAmount }} {{ selectedStatisticCurrency }}</span>
|
||||
</div>
|
||||
<div class="withdraw-data-item">
|
||||
{{ $t('timePassed') }}
|
||||
<b-tooltip
|
||||
:active="notEnoughPassedTime"
|
||||
:label="$t('timePassedTooltip')"
|
||||
position="is-left"
|
||||
multilined
|
||||
size="is-large"
|
||||
:class="{ 'has-low-anonymity': notEnoughPassedTime }"
|
||||
>
|
||||
<span>{{ timePassed }}</span>
|
||||
</b-tooltip>
|
||||
</div>
|
||||
<div class="withdraw-data-item">
|
||||
{{ $t('subsequentDeposits') }}
|
||||
<b-tooltip
|
||||
:active="notEnoughDeposits"
|
||||
:label="$t('subsequentDepositsTooltip')"
|
||||
position="is-left"
|
||||
multilined
|
||||
size="is-large"
|
||||
:class="{ 'has-low-anonymity': notEnoughDeposits }"
|
||||
>
|
||||
<span>{{ this.$tc('userDeposit', depositsPast) }}</span>
|
||||
</b-tooltip>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<fieldset>
|
||||
<div class="field withdraw-address">
|
||||
<div class="label-with-buttons">
|
||||
<div class="label">
|
||||
<span class="name">
|
||||
{{ $t('recipientAddress') }}
|
||||
</span>
|
||||
</div>
|
||||
<button class="button is-primary-text" @click="insertDonate">
|
||||
{{ $t('donate') }}
|
||||
</button>
|
||||
</div>
|
||||
<b-input
|
||||
v-model="withdrawAddress"
|
||||
:placeholder="$t('pleasePasteAddressHere')"
|
||||
:size="!withdrawAddress ? '' : isValidAddress ? 'is-primary' : 'is-warning'"
|
||||
></b-input>
|
||||
<p class="help">
|
||||
<span class="has-text-warning">{{ error.type }}</span> {{ error.message }}
|
||||
</p>
|
||||
</div>
|
||||
<RelayerTotal v-show="shouldShowRelayerTotal" />
|
||||
<WithdrawTotal
|
||||
v-show="shouldShowTotal"
|
||||
:currency="selectedStatisticCurrency"
|
||||
:withdraw-type="withdrawType"
|
||||
:eth-to-receive="ethToReceive"
|
||||
/>
|
||||
<b-tooltip
|
||||
class="is-block"
|
||||
:label="`${$t(tooltipText)}`"
|
||||
position="is-top"
|
||||
:active="shouldTooltipShow"
|
||||
multilined
|
||||
>
|
||||
<b-button
|
||||
type="is-primary is-fullwidth"
|
||||
class="slide-animation"
|
||||
:outlined="isLoading"
|
||||
:expanded="isLoading"
|
||||
:class="{ 'slide-animation-active': isLoading }"
|
||||
:disabled="isWithdrawalButtonDisable"
|
||||
:loading="isLoadingRelayers || isLoading"
|
||||
@click="onWithdraw"
|
||||
>
|
||||
{{ $t('withdrawButton') }}
|
||||
</b-button>
|
||||
</b-tooltip>
|
||||
</fieldset>
|
||||
</b-tab-item>
|
||||
</template>
|
||||
<script>
|
||||
/* eslint-disable no-console */
|
||||
import { mapState, mapGetters } from 'vuex'
|
||||
import { getTornadoKeys } from '@/store/snark'
|
||||
|
||||
import { parseNote } from '@/utils'
|
||||
import { DONATIONS_ADDRESS } from '@/constants'
|
||||
|
||||
import { LinkIcon, SettingsIcon } from '@/components/icons'
|
||||
import RelayerTotal from '@/components/withdraw/RelayerTotal'
|
||||
import WithdrawTotal from '@/components/withdraw/WithdrawTotal'
|
||||
import WithdrawModalBox from '@/components/withdraw/WithdrawModalBox'
|
||||
import SettingsModalBox from '@/components/settings/SettingsModalBox'
|
||||
|
||||
const { toChecksumAddress, isHexStrict, isAddress } = require('web3-utils')
|
||||
|
||||
export default {
|
||||
components: {
|
||||
LinkIcon,
|
||||
RelayerTotal,
|
||||
SettingsIcon,
|
||||
WithdrawTotal
|
||||
},
|
||||
props: {
|
||||
activeTab: {
|
||||
required: true,
|
||||
type: Number
|
||||
}
|
||||
},
|
||||
data() {
|
||||
return {
|
||||
withdrawAddress: '',
|
||||
withdrawNote: '',
|
||||
depositsPast: null,
|
||||
depositTxHash: null,
|
||||
depositTimestamp: null,
|
||||
isSpent: false,
|
||||
isLoading: false,
|
||||
isFileError: false,
|
||||
error: {
|
||||
type: null,
|
||||
message: ''
|
||||
},
|
||||
timePassed: ''
|
||||
}
|
||||
},
|
||||
computed: {
|
||||
...mapState('application', ['note', 'errors', 'withdrawType', 'ethToReceive']),
|
||||
...mapState('relayer', ['isLoadingRelayers']),
|
||||
...mapGetters('txHashKeeper', ['txExplorerUrl']),
|
||||
...mapGetters('application', ['isNotEnoughTokens', 'selectedStatisticCurrency']),
|
||||
...mapGetters('metamask', ['networkConfig', 'netId', 'isLoggedIn', 'nativeCurrency']),
|
||||
notEnoughDeposits() {
|
||||
if (this.depositsPast < 5) {
|
||||
return true
|
||||
}
|
||||
return false
|
||||
},
|
||||
shouldSettingsShow() {
|
||||
return !this.hasErrorNote && !this.error.message
|
||||
},
|
||||
hasErrorNote() {
|
||||
const note = this.withdrawNote.split('-')[4]
|
||||
if (typeof WebAssembly === 'undefined') {
|
||||
return {
|
||||
type: 'is-warning',
|
||||
msg: this.$t('turnOnWasm')
|
||||
}
|
||||
}
|
||||
if (!this.withdrawNote) {
|
||||
return { type: '', msg: '' }
|
||||
}
|
||||
if (!note || note.length < 126 || !isHexStrict(note)) {
|
||||
return { type: 'is-warning', msg: this.$t('noteIsInvalid') }
|
||||
}
|
||||
if (this.isSpent) {
|
||||
return { type: 'is-warning', msg: this.$t('noteHasBeenSpent') }
|
||||
}
|
||||
return false
|
||||
},
|
||||
withDisconnectedWallet() {
|
||||
return this.withdrawType === 'wallet' && !this.isLoggedIn
|
||||
},
|
||||
shouldTooltipShow() {
|
||||
return (!this.isWithdrawDisabled && this.isNotEnoughTokens) || this.withDisconnectedWallet
|
||||
},
|
||||
tooltipText() {
|
||||
if (this.withDisconnectedWallet) {
|
||||
return 'withDisconnectedWallet'
|
||||
}
|
||||
return 'notEnoughTokens'
|
||||
},
|
||||
isValidAddress() {
|
||||
return isAddress(this.withdrawAddress)
|
||||
},
|
||||
notEnoughPassedTime() {
|
||||
return this.$moment().unix() - Number(this.depositTimestamp) < 86400 // less than 24 hours
|
||||
},
|
||||
hasWarning() {
|
||||
return this.depositsPast < 5 || this.notEnoughPassedTime
|
||||
},
|
||||
isWithdrawDisabled() {
|
||||
return (
|
||||
this.isLoading ||
|
||||
!!this.error.type ||
|
||||
this.hasErrorNote ||
|
||||
(this.withdrawType === 'relayer' && !this.selectedRelayer) ||
|
||||
!this.isValidAddress
|
||||
)
|
||||
},
|
||||
isWithdrawalButtonDisable() {
|
||||
return (
|
||||
this.isWithdrawDisabled || this.isNotEnoughTokens || this.isFileError || this.withDisconnectedWallet
|
||||
)
|
||||
},
|
||||
selectedRelayer() {
|
||||
return this.$store.state.relayer.selectedRelayer.name
|
||||
},
|
||||
selectedAmount() {
|
||||
return this.$store.state.application.selectedStatistic.amount
|
||||
},
|
||||
tokens() {
|
||||
return this.networkConfig.tokens
|
||||
},
|
||||
shouldShowTotal() {
|
||||
return this.isValidAddress && !this.isWithdrawDisabled
|
||||
},
|
||||
shouldShowRelayerTotal() {
|
||||
return this.withdrawType === 'relayer' && this.shouldShowTotal
|
||||
}
|
||||
},
|
||||
watch: {
|
||||
netId(netId, oldNetId) {
|
||||
if (netId !== oldNetId) {
|
||||
const [, , , noteNetId] = this.withdrawNote.split('-')
|
||||
|
||||
if (Number(noteNetId) !== netId && noteNetId) {
|
||||
this.error = {
|
||||
type: this.$t('error'),
|
||||
message: this.$t('changeNetworkNote')
|
||||
}
|
||||
} else {
|
||||
this.error = {
|
||||
type: '',
|
||||
message: ''
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
errors: {
|
||||
handler(errors) {
|
||||
console.log('error', errors)
|
||||
this.error = {
|
||||
type: errors.length ? this.$t('error') : null,
|
||||
message: errors[errors.length - 1]
|
||||
}
|
||||
if (this.error.message) {
|
||||
this.$store.dispatch('notice/addNoticeWithInterval', {
|
||||
notice: {
|
||||
untranslatedTitle: this.error.message,
|
||||
type: 'warning'
|
||||
}
|
||||
})
|
||||
}
|
||||
},
|
||||
deep: true
|
||||
},
|
||||
withdrawNote: {
|
||||
async handler(withdrawNote) {
|
||||
this.error = {
|
||||
type: '',
|
||||
message: ''
|
||||
}
|
||||
|
||||
try {
|
||||
this.$store.dispatch('loading/enable', { message: this.$t('gettingTheNoteData') })
|
||||
this.isSpent = false
|
||||
this.depositsPast = null
|
||||
this.depositTxHash = null
|
||||
this.depositTimestamp = null
|
||||
|
||||
if (!this.hasErrorNote) {
|
||||
const [tornadoPrefix, currency, amount, noteNetId, note] = this.withdrawNote.split('-')
|
||||
if (tornadoPrefix !== 'tornado') {
|
||||
this.$store.dispatch('loading/disable')
|
||||
this.withdrawNote = `tornado-${currency}-${amount}-${noteNetId}-${note}`
|
||||
return
|
||||
}
|
||||
this.getLogs(withdrawNote)
|
||||
this.$store.commit('application/SET_WITHDRAW_NOTE', withdrawNote)
|
||||
const netIdMissmatch = Number(noteNetId) !== Number(this.netId)
|
||||
|
||||
if (netIdMissmatch) {
|
||||
throw new Error(this.$t('changeNetworkNote'))
|
||||
}
|
||||
|
||||
const event = await this.$store.dispatch('application/loadDepositEvent', { withdrawNote })
|
||||
|
||||
if (!event) {
|
||||
throw new Error(this.$t('thereIsNoRelatedDeposit'))
|
||||
}
|
||||
|
||||
const { timestamp, txHash, isSpent, depositsPast = 0 } = event
|
||||
|
||||
if (isSpent) {
|
||||
this.$store.dispatch('notice/addNoticeWithInterval', {
|
||||
notice: {
|
||||
title: 'noteWasAlreadySpent',
|
||||
type: 'warning'
|
||||
},
|
||||
interval: 5000
|
||||
})
|
||||
}
|
||||
this.$store.dispatch('application/setAndUpdateStatistic', { currency, amount: Number(amount) })
|
||||
if (currency !== this.nativeCurrency) {
|
||||
this.$store.dispatch('application/setDefaultEthToReceive', { currency })
|
||||
}
|
||||
this.depositsPast = Number(depositsPast) <= 0 ? 0 : depositsPast
|
||||
this.depositTxHash = txHash
|
||||
this.depositTimestamp = timestamp
|
||||
this.isSpent = isSpent
|
||||
}
|
||||
} catch (err) {
|
||||
this.error = {
|
||||
type: this.$t('error'),
|
||||
message: err.message
|
||||
}
|
||||
} finally {
|
||||
this.$store.dispatch('loading/disable')
|
||||
}
|
||||
}
|
||||
},
|
||||
'$i18n.locale'() {
|
||||
this.timePastToRender()
|
||||
},
|
||||
depositTimestamp() {
|
||||
this.timePastToRender()
|
||||
},
|
||||
activeTab(newTab, oldTab) {
|
||||
if (newTab !== oldTab && newTab === 1) {
|
||||
this.error = {
|
||||
type: '',
|
||||
message: ''
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
created() {
|
||||
this.$emit('get-key', this.getKeys)
|
||||
},
|
||||
mounted() {
|
||||
if (this.$route.query.note) {
|
||||
this.withdrawNote = this.$route.query.note
|
||||
}
|
||||
this.$root.$on('resetWithdraw', () => {
|
||||
this.withdrawAddress = ''
|
||||
this.withdrawNote = ''
|
||||
})
|
||||
},
|
||||
methods: {
|
||||
async getKeys() {
|
||||
try {
|
||||
this.isFileError = false
|
||||
this.isLoading = true
|
||||
this.getProgress(0)
|
||||
await getTornadoKeys(this.getProgress)
|
||||
return true
|
||||
} catch (err) {
|
||||
console.error('getKeys has error:', err.message)
|
||||
|
||||
this.$store.dispatch(
|
||||
'notice/addNoticeWithInterval',
|
||||
{
|
||||
notice: {
|
||||
title: 'fetchFile',
|
||||
type: 'warning'
|
||||
}
|
||||
},
|
||||
{ root: true }
|
||||
)
|
||||
this.error = {
|
||||
type: this.$t('downloadError'),
|
||||
message: this.$t('fetchFile')
|
||||
}
|
||||
this.isFileError = true
|
||||
} finally {
|
||||
this.isLoading = false
|
||||
}
|
||||
},
|
||||
getProgress(value) {
|
||||
document.documentElement.style.setProperty('--width-animation', `${value}%`)
|
||||
},
|
||||
getLogs(note) {
|
||||
try {
|
||||
if (!note) {
|
||||
return
|
||||
}
|
||||
|
||||
const { commitmentHex, nullifierHex } = parseNote(note)
|
||||
|
||||
console.log('\n\nYOUR NOTE DATA:')
|
||||
console.log('note:', note)
|
||||
|
||||
console.log('commitment:', commitmentHex)
|
||||
console.log('nullifierHash:', nullifierHex)
|
||||
} catch (err) {
|
||||
console.log(`Get logs: ${err.message}`)
|
||||
}
|
||||
},
|
||||
onWithdraw() {
|
||||
const note = this.withdrawNote.split('-')[4]
|
||||
if (note.length !== 126) {
|
||||
this.error = {
|
||||
type: this.$t('validationError'),
|
||||
message: this.$t('noteIsInvalid')
|
||||
}
|
||||
return
|
||||
}
|
||||
try {
|
||||
this.withdrawAddress = toChecksumAddress(this.withdrawAddress)
|
||||
this.$store.dispatch('application/prepareWithdraw', {
|
||||
note: this.withdrawNote,
|
||||
recipient: this.withdrawAddress
|
||||
})
|
||||
this.error.type = null
|
||||
this.currentModal = this.$buefy.modal.open({
|
||||
parent: this,
|
||||
component: WithdrawModalBox,
|
||||
hasModalCard: true,
|
||||
width: 440,
|
||||
props: {
|
||||
note: this.withdrawNote,
|
||||
withdrawType: this.withdrawType
|
||||
}
|
||||
})
|
||||
} catch (e) {
|
||||
this.error = {
|
||||
type: this.$t('validationError'),
|
||||
message: this.$t('recipientAddressIsInvalid')
|
||||
}
|
||||
console.error('error', e)
|
||||
}
|
||||
},
|
||||
onSettings() {
|
||||
this.$buefy.modal.open({
|
||||
parent: this,
|
||||
component: SettingsModalBox,
|
||||
hasModalCard: true,
|
||||
width: 440,
|
||||
props: {
|
||||
currency: this.selectedStatisticCurrency
|
||||
},
|
||||
customClass: 'is-pinned'
|
||||
})
|
||||
},
|
||||
timePastToRender() {
|
||||
this.timePassed = this.$moment.unix(this.depositTimestamp).fromNow(true)
|
||||
},
|
||||
insertDonate() {
|
||||
this.withdrawAddress = DONATIONS_ADDRESS
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
<style lang="scss" scoped>
|
||||
:root {
|
||||
--width-animation: 0;
|
||||
}
|
||||
|
||||
.slide-animation {
|
||||
position: relative;
|
||||
|
||||
&:before {
|
||||
content: '';
|
||||
position: absolute;
|
||||
left: 0;
|
||||
right: 0;
|
||||
top: 0;
|
||||
bottom: 0;
|
||||
width: var(--width-animation);
|
||||
height: 100%;
|
||||
background-color: #94febf;
|
||||
animation-fill-mode: backwards;
|
||||
}
|
||||
|
||||
::v-deep span {
|
||||
position: relative;
|
||||
z-index: 2;
|
||||
}
|
||||
|
||||
&-active {
|
||||
::v-deep span,
|
||||
&:after {
|
||||
filter: invert(0.5);
|
||||
}
|
||||
}
|
||||
}
|
||||
</style>
|
Loading…
Add table
Add a link
Reference in a new issue