You've already forked AstralRinth
forked from didirus/AstralRinth
471 lines
14 KiB
Vue
471 lines
14 KiB
Vue
<template>
|
|
<NewModal ref="refundModal">
|
|
<template #title>
|
|
<span class="text-lg font-extrabold text-contrast">Refund charge</span>
|
|
</template>
|
|
<div class="flex flex-col gap-3">
|
|
<div class="flex flex-col gap-2">
|
|
<label for="visibility" class="flex flex-col gap-1">
|
|
<span class="text-lg font-semibold text-contrast">
|
|
Refund type
|
|
<span class="text-brand-red">*</span>
|
|
</span>
|
|
<span> The type of refund to issue. </span>
|
|
</label>
|
|
<DropdownSelect
|
|
id="refund-type"
|
|
v-model="refundType"
|
|
:options="refundTypes"
|
|
name="Refund type"
|
|
/>
|
|
</div>
|
|
<div v-if="refundType === 'partial'" class="flex flex-col gap-2">
|
|
<label for="amount" class="flex flex-col gap-1">
|
|
<span class="text-lg font-semibold text-contrast">
|
|
Amount
|
|
<span class="text-brand-red">*</span>
|
|
</span>
|
|
<span>
|
|
Enter the amount in cents of USD. For example for $2, enter 200. (net
|
|
{{ selectedCharge.net }})
|
|
</span>
|
|
</label>
|
|
<input id="amount" v-model="refundAmount" type="number" autocomplete="off" />
|
|
</div>
|
|
<div class="flex flex-col gap-2">
|
|
<label for="unprovision" class="flex flex-col gap-1">
|
|
<span class="text-lg font-semibold text-contrast">
|
|
Unprovision
|
|
<span class="text-brand-red">*</span>
|
|
</span>
|
|
<span> Whether or not the subscription should be unprovisioned on refund. </span>
|
|
</label>
|
|
<Toggle id="unprovision" v-model="unprovision" />
|
|
</div>
|
|
<div class="flex gap-2">
|
|
<ButtonStyled color="brand">
|
|
<button :disabled="refunding" @click="refundCharge">
|
|
<CheckIcon aria-hidden="true" />
|
|
Refund charge
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled>
|
|
<button @click="refundModal.hide()">
|
|
<XIcon aria-hidden="true" />
|
|
Cancel
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
</NewModal>
|
|
<NewModal ref="modifyModal">
|
|
<template #title>
|
|
<span class="text-lg font-extrabold text-contrast">Modify charge</span>
|
|
</template>
|
|
<div class="flex flex-col gap-3">
|
|
<div class="flex flex-col gap-2">
|
|
<label for="cancel" class="flex flex-col gap-1">
|
|
<span class="text-lg font-semibold text-contrast">
|
|
Cancel server
|
|
<span class="text-brand-red">*</span>
|
|
</span>
|
|
<span>
|
|
Whether or not the subscription should be cancelled. Submitting this as "true" will
|
|
cancel the subscription, while submitting it as "false" will
|
|
{{
|
|
selectedCharge.status === 'open'
|
|
? 'keep it as-is'
|
|
: 'force another charge attempt to be made'
|
|
}}.
|
|
</span>
|
|
</label>
|
|
<Toggle id="cancel" v-model="cancel" />
|
|
</div>
|
|
<div class="flex gap-2">
|
|
<ButtonStyled color="brand">
|
|
<button :disabled="modifying" @click="modifyCharge">
|
|
<CheckIcon aria-hidden="true" />
|
|
Modify charge
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled>
|
|
<button @click="modifyModal.hide()">
|
|
<XIcon aria-hidden="true" />
|
|
Cancel
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
</NewModal>
|
|
<div class="page experimental-styles-within">
|
|
<div
|
|
class="mb-4 flex items-center justify-between border-0 border-b border-solid border-divider pb-4"
|
|
>
|
|
<div class="flex items-center gap-2">
|
|
<Avatar :src="user.avatar_url" :alt="user.username" size="32px" circle />
|
|
<h1 class="m-0 text-2xl font-extrabold">{{ user.username }}'s subscriptions</h1>
|
|
</div>
|
|
<div class="flex items-center gap-2">
|
|
<ButtonStyled>
|
|
<nuxt-link :to="`/user/${user.id}`">
|
|
<UserIcon aria-hidden="true" />
|
|
User profile
|
|
<ExternalIcon class="h-4 w-4" />
|
|
</nuxt-link>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<div v-for="subscription in subscriptionCharges" :key="subscription.id" class="card">
|
|
<div class="mb-4 grid grid-cols-[1fr_auto]">
|
|
<div>
|
|
<span class="flex items-center gap-2 font-semibold text-contrast">
|
|
<!-- TODO(backend): provide proper metadata for midas (MR+) subscriptions -->
|
|
<template v-if="subscription.price_id === 'a6eRm92L'">
|
|
<ModrinthPlusIcon class="h-7 w-min" />
|
|
</template>
|
|
<template v-else-if="subscription.metadata?.type === 'pyro'">
|
|
<ModrinthServersIcon class="h-7 w-min" />
|
|
</template>
|
|
<template v-else-if="subscription.metadata?.type === 'medal'">
|
|
<span>Medal Trial Server</span>
|
|
</template>
|
|
<template v-else> Unknown product </template>
|
|
</span>
|
|
<div class="mb-4 mt-2 flex w-full items-center gap-1 text-sm text-secondary">
|
|
{{ formatCategory(subscription.interval) }} ⋅ {{ subscription.status }} ⋅
|
|
{{ dayjs(subscription.created).format('MMMM D, YYYY [at] h:mma') }} ({{
|
|
formatRelativeTime(subscription.created)
|
|
}})
|
|
</div>
|
|
</div>
|
|
<div v-if="subscription.metadata?.id" class="flex flex-col items-end gap-2">
|
|
<ButtonStyled
|
|
v-if="
|
|
subscription.metadata?.type === 'pyro' || subscription.metadata?.type === 'medal'
|
|
"
|
|
>
|
|
<nuxt-link
|
|
:to="`/servers/manage/${subscription.metadata.id}`"
|
|
target="_blank"
|
|
class="w-fit"
|
|
>
|
|
<ServerIcon /> Server panel <ExternalIcon class="h-4 w-4" />
|
|
</nuxt-link>
|
|
</ButtonStyled>
|
|
<CopyCode :text="subscription.metadata.id" />
|
|
</div>
|
|
</div>
|
|
<div class="flex flex-col gap-2">
|
|
<div
|
|
v-for="(charge, index) in subscription.charges"
|
|
:key="charge.id"
|
|
class="relative overflow-clip rounded-xl bg-bg px-4 py-3"
|
|
>
|
|
<div
|
|
class="absolute bottom-0 left-0 top-0 w-1"
|
|
:class="
|
|
charge.type === 'refund'
|
|
? 'bg-purple'
|
|
: (chargeStatuses[charge.status]?.color ?? 'bg-blue')
|
|
"
|
|
/>
|
|
<div class="grid w-full grid-cols-[1fr_auto] items-center gap-4">
|
|
<div class="flex flex-col gap-2">
|
|
<span>
|
|
<span class="font-bold text-contrast">
|
|
<template v-if="charge.status === 'succeeded'"> Succeeded </template>
|
|
<template v-else-if="charge.status === 'failed'"> Failed </template>
|
|
<template v-else-if="charge.status === 'cancelled'"> Cancelled </template>
|
|
<template v-else-if="charge.status === 'processing'"> Processing </template>
|
|
<template v-else-if="charge.status === 'open'"> Upcoming </template>
|
|
<template v-else-if="charge.status === 'expiring'"> Expiring </template>
|
|
<template v-else> {{ charge.status }} </template>
|
|
</span>
|
|
⋅
|
|
<span>
|
|
<template v-if="charge.type === 'refund'"> Refund </template>
|
|
<template v-else-if="charge.type === 'subscription'">
|
|
<template v-if="charge.status === 'cancelled'"> Subscription </template>
|
|
<template v-else-if="index === subscription.charges.length - 1">
|
|
Started subscription
|
|
</template>
|
|
<template v-else> Subscription renewal </template>
|
|
</template>
|
|
<template v-else-if="charge.type === 'one-time'"> One-time charge </template>
|
|
<template v-else-if="charge.type === 'proration'"> Proration charge </template>
|
|
<template v-else> {{ charge.status }} </template>
|
|
</span>
|
|
<template v-if="charge.status !== 'cancelled'">
|
|
⋅
|
|
{{ formatPrice(vintl.locale, charge.amount, charge.currency_code) }}
|
|
</template>
|
|
</span>
|
|
<span class="text-sm text-secondary">
|
|
<span
|
|
v-if="charge.status === 'cancelled' && $dayjs(charge.due).isBefore($dayjs())"
|
|
class="font-bold"
|
|
>
|
|
Ended:
|
|
</span>
|
|
<span v-else-if="charge.status === 'cancelled'" class="font-bold">Ends:</span>
|
|
<span v-else-if="charge.type === 'refund'" class="font-bold">Issued:</span>
|
|
<span v-else class="font-bold">Due:</span>
|
|
{{ dayjs(charge.due).format('MMMM D, YYYY [at] h:mma') }}
|
|
<span class="text-secondary">({{ formatRelativeTime(charge.due) }}) </span>
|
|
</span>
|
|
<span v-if="charge.last_attempt != null" class="text-sm text-secondary">
|
|
<span v-if="charge.status === 'failed'" class="font-bold">Last attempt:</span>
|
|
<span v-else class="font-bold">Charged:</span>
|
|
{{ dayjs(charge.last_attempt).format('MMMM D, YYYY [at] h:mma') }}
|
|
<span class="text-secondary"
|
|
>({{ formatRelativeTime(charge.last_attempt) }})
|
|
</span>
|
|
</span>
|
|
<div class="flex w-full items-center gap-1 text-xs text-secondary">
|
|
{{ charge.status }}
|
|
⋅
|
|
{{ charge.type }}
|
|
⋅
|
|
{{ formatPrice(vintl.locale, charge.amount, charge.currency_code) }}
|
|
⋅
|
|
{{ dayjs(charge.due).format('YYYY-MM-DD h:mma') }}
|
|
<template v-if="charge.subscription_interval">
|
|
⋅ {{ charge.subscription_interval }}
|
|
</template>
|
|
</div>
|
|
</div>
|
|
<div class="flex gap-2">
|
|
<ButtonStyled
|
|
v-if="
|
|
charges.some((x) => x.type === 'refund' && x.parent_charge_id === charge.id)
|
|
"
|
|
>
|
|
<div class="button-like disabled"><CheckIcon /> Charge refunded</div>
|
|
</ButtonStyled>
|
|
<ButtonStyled
|
|
v-else-if="charge.status === 'succeeded' && charge.type !== 'refund'"
|
|
color="red"
|
|
color-fill="text"
|
|
>
|
|
<button @click="showRefundModal(charge)">
|
|
<CurrencyIcon />
|
|
Refund options
|
|
</button>
|
|
</ButtonStyled>
|
|
<ButtonStyled
|
|
v-else-if="charge.status === 'failed' || charge.status === 'open'"
|
|
color="red"
|
|
color-fill="text"
|
|
>
|
|
<button @click="showModifyModal(charge, subscription)">
|
|
<CurrencyIcon />
|
|
Modify charge
|
|
</button>
|
|
</ButtonStyled>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
<script setup>
|
|
import {
|
|
CheckIcon,
|
|
CurrencyIcon,
|
|
ExternalIcon,
|
|
ModrinthPlusIcon,
|
|
ServerIcon,
|
|
UserIcon,
|
|
XIcon,
|
|
} from '@modrinth/assets'
|
|
import {
|
|
Avatar,
|
|
ButtonStyled,
|
|
CopyCode,
|
|
DropdownSelect,
|
|
injectNotificationManager,
|
|
NewModal,
|
|
Toggle,
|
|
useRelativeTime,
|
|
} from '@modrinth/ui'
|
|
import { formatCategory, formatPrice } from '@modrinth/utils'
|
|
import dayjs from 'dayjs'
|
|
|
|
import ModrinthServersIcon from '~/components/ui/servers/ModrinthServersIcon.vue'
|
|
|
|
const { addNotification } = injectNotificationManager()
|
|
|
|
const route = useRoute()
|
|
const vintl = useVIntl()
|
|
|
|
const { formatMessage } = vintl
|
|
const formatRelativeTime = useRelativeTime()
|
|
|
|
const messages = defineMessages({
|
|
userNotFoundError: {
|
|
id: 'admin.billing.error.not-found',
|
|
defaultMessage: 'User not found',
|
|
},
|
|
})
|
|
|
|
const { data: user } = await useAsyncData(`user/${route.params.id}`, () =>
|
|
useBaseFetch(`user/${route.params.id}`),
|
|
)
|
|
|
|
if (!user.value) {
|
|
throw createError({
|
|
fatal: true,
|
|
statusCode: 404,
|
|
message: formatMessage(messages.userNotFoundError),
|
|
})
|
|
}
|
|
|
|
let subscriptions, charges, refreshCharges
|
|
try {
|
|
;[{ data: subscriptions }, { data: charges, refresh: refreshCharges }] = await Promise.all([
|
|
useAsyncData(`billing/subscriptions?user_id=${route.params.id}`, () =>
|
|
useBaseFetch(`billing/subscriptions?user_id=${user.value.id}`, {
|
|
internal: true,
|
|
}),
|
|
),
|
|
useAsyncData(`billing/payments?user_id=${route.params.id}`, () =>
|
|
useBaseFetch(`billing/payments?user_id=${user.value.id}`, {
|
|
internal: true,
|
|
}),
|
|
),
|
|
])
|
|
} catch {
|
|
throw createError({
|
|
fatal: true,
|
|
statusCode: 404,
|
|
message: formatMessage(messages.userNotFoundError),
|
|
})
|
|
}
|
|
|
|
const subscriptionCharges = computed(() => {
|
|
return subscriptions.value.map((subscription) => {
|
|
return {
|
|
...subscription,
|
|
charges: charges.value
|
|
.filter((charge) => charge.subscription_id === subscription.id)
|
|
.slice()
|
|
.sort((a, b) => dayjs(b.due).diff(dayjs(a.due))),
|
|
}
|
|
})
|
|
})
|
|
|
|
const refunding = ref(false)
|
|
const refundModal = ref()
|
|
const selectedCharge = ref(null)
|
|
const selectedSubscription = ref(null)
|
|
const refundType = ref('full')
|
|
const refundTypes = ref(['full', 'partial', 'none'])
|
|
const refundAmount = ref(0)
|
|
const unprovision = ref(true)
|
|
|
|
const modifying = ref(false)
|
|
const modifyModal = ref()
|
|
const cancel = ref(false)
|
|
|
|
function showRefundModal(charge) {
|
|
selectedCharge.value = charge
|
|
refundType.value = 'full'
|
|
refundAmount.value = 0
|
|
unprovision.value = true
|
|
refundModal.value.show()
|
|
}
|
|
|
|
function showModifyModal(charge, subscription) {
|
|
selectedCharge.value = charge
|
|
selectedSubscription.value = subscription
|
|
cancel.value = false
|
|
modifyModal.value.show()
|
|
}
|
|
|
|
async function refundCharge() {
|
|
refunding.value = true
|
|
try {
|
|
const amountParsed = Math.max(0, Math.floor(Number(refundAmount.value) || 0))
|
|
const payload =
|
|
refundType.value === 'partial'
|
|
? { type: 'partial', amount: amountParsed, unprovision: unprovision.value }
|
|
: refundType.value === 'none'
|
|
? { type: 'none', unprovision: unprovision.value }
|
|
: { type: 'full', unprovision: unprovision.value }
|
|
|
|
await useBaseFetch(`billing/charge/${selectedCharge.value.id}/refund`, {
|
|
method: 'POST',
|
|
body: JSON.stringify(payload),
|
|
internal: true,
|
|
})
|
|
await refreshCharges()
|
|
refundModal.value.hide()
|
|
} catch (err) {
|
|
addNotification({
|
|
title: 'Error refunding',
|
|
text: err.data?.description ?? err,
|
|
type: 'error',
|
|
})
|
|
}
|
|
refunding.value = false
|
|
}
|
|
|
|
async function modifyCharge() {
|
|
modifying.value = true
|
|
try {
|
|
await useBaseFetch(`billing/subscription/${selectedSubscription.value.id}`, {
|
|
method: 'PATCH',
|
|
body: JSON.stringify({
|
|
cancelled: cancel.value,
|
|
}),
|
|
internal: true,
|
|
})
|
|
addNotification({
|
|
title: 'Modifications made',
|
|
text: 'If the server is currently suspended, it may take up to 10 minutes for another charge attempt to be made.',
|
|
type: 'success',
|
|
})
|
|
await refreshCharges()
|
|
} catch (err) {
|
|
addNotification({
|
|
title: 'Error while sending request',
|
|
text: err.data?.description ?? err,
|
|
type: 'error',
|
|
})
|
|
}
|
|
modifying.value = false
|
|
}
|
|
|
|
const chargeStatuses = {
|
|
open: {
|
|
color: 'bg-blue',
|
|
},
|
|
processing: {
|
|
color: 'bg-orange',
|
|
},
|
|
succeeded: {
|
|
color: 'bg-green',
|
|
},
|
|
failed: {
|
|
color: 'bg-red',
|
|
},
|
|
cancelled: {
|
|
color: 'bg-red',
|
|
},
|
|
expiring: {
|
|
color: 'bg-orange',
|
|
},
|
|
}
|
|
</script>
|
|
<style scoped>
|
|
.page {
|
|
padding: 1rem;
|
|
margin-left: auto;
|
|
margin-right: auto;
|
|
max-width: 56rem;
|
|
}
|
|
</style>
|