You've already forked pages
forked from didirus/AstralRinth
* polish: increase gap between svg and text in empty state * fix: use ts & change cancel btn style * fix: btn style * polish: new transaction page design * fix: navstack match nested + csv download * fix: lint & i18n * Add tooltip to CSV download button + standard btn style Signed-off-by: Calum H. <contact@cal.engineer> * fix: lint & i18n --------- Signed-off-by: Calum H. <contact@cal.engineer>
153 lines
4.5 KiB
Vue
153 lines
4.5 KiB
Vue
<template>
|
|
<div class="flex flex-row gap-2 md:gap-3">
|
|
<div
|
|
class="flex h-10 min-h-10 w-10 min-w-10 justify-center rounded-full border-[1px] border-solid border-button-bg bg-bg-raised !p-0 shadow-md md:h-12 md:min-h-12 md:w-12 md:min-w-12"
|
|
>
|
|
<ArrowDownIcon v-if="isIncome" class="my-auto size-6 text-secondary md:size-8" />
|
|
<ArrowUpIcon v-else class="my-auto size-6 text-secondary md:size-8" />
|
|
</div>
|
|
<div class="flex w-full flex-row justify-between">
|
|
<div class="flex flex-col">
|
|
<span class="text-base font-semibold text-contrast md:text-lg">{{
|
|
transaction.type === 'payout_available'
|
|
? formatPayoutSource(transaction.payout_source)
|
|
: formatMethodName(transaction.method_type || transaction.method)
|
|
}}</span>
|
|
<span class="text-xs text-secondary md:text-sm">
|
|
<template v-if="transaction.type === 'withdrawal'">
|
|
<span
|
|
:class="[
|
|
transaction.status === 'cancelling' || transaction.status === 'cancelled'
|
|
? 'text-red'
|
|
: '',
|
|
]"
|
|
>{{ formatTransactionStatus(transaction.status) }} <BulletDivider
|
|
/></span>
|
|
</template>
|
|
{{ dayjs(transaction.created).format('MMM DD YYYY') }}
|
|
<template v-if="transaction.type === 'withdrawal' && transaction.fee">
|
|
<BulletDivider /> Fee {{ formatMoney(transaction.fee) }}
|
|
</template>
|
|
</span>
|
|
</div>
|
|
<div class="my-auto flex flex-row items-center gap-2">
|
|
<span
|
|
class="text-base font-semibold md:text-lg"
|
|
:class="transaction.type === 'payout_available' ? 'text-green' : 'text-contrast'"
|
|
>{{ formatMoney(transaction.amount) }}</span
|
|
>
|
|
<template v-if="transaction.type === 'withdrawal' && transaction.status === 'in-transit'">
|
|
<Tooltip theme="dismissable-prompt" :triggers="['hover', 'focus']" no-auto-focus>
|
|
<span class="my-auto align-middle"
|
|
><ButtonStyled circular type="outlined" size="small">
|
|
<button class="align-middle" @click="cancelPayout">
|
|
<XIcon />
|
|
</button> </ButtonStyled
|
|
></span>
|
|
<template #popper>
|
|
<div class="font-semibold text-contrast">Cancel transaction</div>
|
|
</template>
|
|
</Tooltip>
|
|
</template>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ArrowDownIcon, ArrowUpIcon, XIcon } from '@modrinth/assets'
|
|
import { BulletDivider, ButtonStyled, injectNotificationManager } from '@modrinth/ui'
|
|
import { capitalizeString, formatMoney } from '@modrinth/utils'
|
|
import dayjs from 'dayjs'
|
|
import { Tooltip } from 'floating-vue'
|
|
|
|
type PayoutStatus = 'in-transit' | 'cancelling' | 'cancelled' | 'success' | 'failed'
|
|
type PayoutMethodType = 'paypal' | 'venmo' | 'tremendous' | 'muralpay'
|
|
type PayoutSource = 'creator_rewards' | 'affilites'
|
|
|
|
type WithdrawalTransaction = {
|
|
type: 'withdrawal'
|
|
id: string
|
|
status: PayoutStatus
|
|
created: string
|
|
amount: number
|
|
fee?: number | null
|
|
method_type?: PayoutMethodType | null
|
|
method?: string
|
|
method_id?: string
|
|
method_address?: string | null
|
|
}
|
|
|
|
type PayoutAvailableTransaction = {
|
|
type: 'payout_available'
|
|
created: string
|
|
payout_source: PayoutSource
|
|
amount: number
|
|
}
|
|
|
|
type Transaction = WithdrawalTransaction | PayoutAvailableTransaction
|
|
|
|
const props = defineProps<{
|
|
transaction: Transaction
|
|
}>()
|
|
|
|
const emit = defineEmits<{
|
|
(e: 'cancelled'): void
|
|
}>()
|
|
|
|
const { addNotification } = injectNotificationManager()
|
|
|
|
const isIncome = computed(() => props.transaction.type === 'payout_available')
|
|
|
|
function formatTransactionStatus(status: string): string {
|
|
if (status === 'in-transit') return 'In Transit'
|
|
return capitalizeString(status)
|
|
}
|
|
|
|
function formatMethodName(method: string | undefined): string {
|
|
if (!method) return 'Unknown'
|
|
switch (method) {
|
|
case 'paypal':
|
|
return 'PayPal'
|
|
case 'venmo':
|
|
return 'Venmo'
|
|
case 'tremendous':
|
|
return 'Tremendous'
|
|
case 'muralpay':
|
|
return 'Muralpay'
|
|
default:
|
|
return capitalizeString(method)
|
|
}
|
|
}
|
|
|
|
function formatPayoutSource(source: string | undefined): string {
|
|
if (!source) return 'Income'
|
|
return source
|
|
.split('_')
|
|
.map((word: string) => capitalizeString(word))
|
|
.join(' ')
|
|
}
|
|
|
|
async function cancelPayout(): Promise<void> {
|
|
startLoading()
|
|
try {
|
|
const transaction = props.transaction
|
|
if (transaction.type !== 'withdrawal') return
|
|
|
|
await useBaseFetch(`payout/${transaction.id}`, {
|
|
method: 'DELETE',
|
|
apiVersion: 3,
|
|
})
|
|
await useAuth()
|
|
emit('cancelled')
|
|
} catch (err: any) {
|
|
addNotification({
|
|
title: 'Failed to cancel transaction',
|
|
text: err.data ? err.data.description : err,
|
|
type: 'error',
|
|
})
|
|
}
|
|
stopLoading()
|
|
}
|
|
</script>
|