-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(configuration): add plan and billing settings
- Loading branch information
1 parent
c3a9430
commit bf38b5a
Showing
4 changed files
with
2,369 additions
and
2,078 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,230 @@ | ||
<script> | ||
import { mapGetters } from 'vuex' | ||
import gql from 'graphql-tag' | ||
const STATES = Object.freeze({ | ||
WAITING: 'WAITING', | ||
UPDATING_PLAIN: 'UPDATING_PLAIN', | ||
CREATING_SESSION: 'CREATING_SESSION' | ||
}) | ||
const PLANS_QUERY = ` | ||
getStripePrices { | ||
id | ||
type | ||
productName | ||
recurringInterval | ||
planType | ||
coreMembersCount | ||
amount | ||
communityMembersCount | ||
} | ||
` | ||
export default { | ||
name: 'settings-plans-billing', | ||
components: { | ||
LoadingSpinner: () => import('~/components/common/loading-spinner.vue'), | ||
Widget: () => import('~/components/common/widget.vue') | ||
}, | ||
props: { | ||
form: { | ||
type: Object, | ||
default: () => {} | ||
}, | ||
isAdmin: { | ||
type: Boolean, | ||
default: false | ||
} | ||
}, | ||
apollo: { | ||
_plans: { | ||
query: gql`query PLANS { ${PLANS_QUERY} }`, | ||
update: data => data.getStripePrices | ||
} | ||
}, | ||
data () { | ||
return { | ||
STATES, | ||
state: STATES.WAITING, | ||
paymentInterval: 'year', | ||
planType: 'SAAS' | ||
} | ||
}, | ||
methods: { | ||
async checkout (id) { | ||
this.state = STATES.CREATING_SESSION | ||
const res = await this.$apollo.mutate({ | ||
mutation: gql`mutation createSession($priceId: String!, $redirectDomain: String) { | ||
createCheckoutSession(priceId: $priceId, redirectDomain: $redirectDomain) { | ||
id | ||
url | ||
} | ||
}`, | ||
variables: { priceId: id } | ||
}) | ||
if (res?.data?.createCheckoutSession?.url) { | ||
window.open(res.data.createCheckoutSession.url, '_self') | ||
} | ||
}, | ||
formatMoney (amount) { return amount ? new Intl.NumberFormat().format(parseInt(amount), { style: 'currency' }) : 0 } | ||
}, | ||
computed: { | ||
...mapGetters('dao', ['selectedDaoPlan']), | ||
isPlanModalOpen () { return [STATES.UPDATING_PLAIN, STATES.CREATING_SESSION].includes(this.state) }, | ||
plans () { | ||
return this._plans | ||
.map(_ => ({ | ||
..._, | ||
id: _.id, | ||
name: _.productName.toLowerCase(), | ||
amountUSD: _.amount / 100 / (_.recurringInterval === 'year' ? 12 : 1), | ||
interval: _.recurringInterval | ||
})) | ||
.filter(_ => _.planType === this.planType) | ||
.filter(_ => _.interval === this.paymentInterval) | ||
.sort((a, b) => a.amountUSD - b.amountUSD) | ||
} | ||
} | ||
} | ||
</script> | ||
<template lang="pug"> | ||
.tab | ||
q-dialog(:value="isPlanModalOpen" full-width="full-width") | ||
widget.relative.wrapper( | ||
v-if="state === STATES.UPDATING_PLAIN" | ||
:title="$t('configuration.settings-plans-billing.plan.modal.title')" | ||
titleImage='/svg/paperplane.svg' | ||
bar | ||
).q-pa-none | ||
template(v-slot:header) | ||
q-btn.q-px-xl.rounded-border.text-bold( | ||
:color="planType === 'EAAS' ? 'primary' : 'secondary'" | ||
:label="planType === 'EAAS' ? 'Single' : 'Ecosystem'" | ||
@click="planType = 'EAAS', paymentInterval = null" | ||
no-caps | ||
rounded | ||
unelevated | ||
) | ||
p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose.q-mt-sm {{ $t('configuration.settings-plans-billing.plan.modal.description') }} | ||
.hr.q-my-md | ||
section | ||
nav.full-width.row.justify-center.items-center(v-if="planType ==='SAAS'") | ||
q-btn.q-px-xl.rounded-border.text-bold( | ||
@click="paymentInterval = 'month'" | ||
:color="paymentInterval === 'month' ? 'primary' : 'secondary'" | ||
no-caps | ||
rounded | ||
unelevated | ||
label="Monthly" | ||
) | ||
q-btn.q-px-xl.rounded-border.text-bold( | ||
@click="paymentInterval = 'year'" | ||
:color="paymentInterval === 'year' ? 'primary' : 'secondary'" | ||
no-caps | ||
rounded | ||
unelevated | ||
label="Yearly" | ||
) | ||
//- q-radio(dense v-model="paymentInterval" val="month" label="Monthly") | ||
//- q-radio(dense v-model="paymentInterval" val="year" label="Yearly") | ||
.row.q-mt-xl | ||
template(v-for='(plan,index) in plans') | ||
.col-3(:class="{'q-pr-sm': index !== plans.length - 1}") | ||
widget(bar shadow) | ||
header.row.justify-between | ||
div | ||
.text-xl.text-weight-600.text-primary {{ $t(`plans.${plan.name}`) }} | ||
p.q-pa-none.q-ma-none.text-3xl.text-primary.text-bold ${{ formatMoney(plan.amountUSD) }} | ||
div(v-if="selectedDaoPlan.id === plan.id") | ||
q-chip(dense color="positive" text-color="white") | ||
span.text-uppercase.text-xxs.text-bold.q-px-xxs {{ $t(`statuses.active`) }} | ||
.hr.q-mt-md.q-mb-xs | ||
footer | ||
div.row.justify-between | ||
p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose Core Members | ||
p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose {{ plan.coreMembersCount }} | ||
//- TODO: Return after beta | ||
//- div.row.justify-between.q-mt-xs | ||
//- p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose Community Members | ||
//- p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose {{ plan.communityMembersCount }} | ||
.hr.q-mt-md.q-mb-xs | ||
nav.q-mt-xl.full-width.row.justify-end | ||
q-btn.q-px-xl.rounded-border.text-bold.q-ml-xs( | ||
:disable="selectedDaoPlan.id === plan.id" | ||
:label="$t('configuration.settings-plans-billing.plan.modal.cta')" | ||
@click="checkout(plan.id)" | ||
color="secondary" | ||
no-caps | ||
rounded | ||
unelevated | ||
) | ||
loading-spinner(v-else color="primary" size="40px") | ||
.row | ||
.col-12.col-md-6(:class="{ 'q-pr-sm': $q.screen.gt.md, 'q-mt-md': $q.screen.lt.md }") | ||
widget(:title="$t('configuration.settings-plans-billing.plan.title')" titleImage='/svg/paperplane.svg' bar).q-pa-none | ||
p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose.q-mt-sm {{ $t('configuration.settings-plans-billing.plan.description.free') }} | ||
widget.q-mt-xl(bar shadow) | ||
header.row.justify-between | ||
div | ||
.text-xl.text-weight-600.text-primary {{ $t(`plans.${selectedDaoPlan.id}`) }} | ||
p.q-pa-none.q-ma-none.text-3xl.text-primary.text-bold ${{ formatMoney(selectedDaoPlan.amountUSD) }} | ||
//- span.q-ml-xxs.text-sm.text-weight-500 / {{ $('periods.month') }} | ||
div | ||
q-chip(dense color="positive" text-color="white") | ||
span.text-uppercase.text-xxs.text-bold.q-px-xxs {{ $t(`statuses.${selectedDaoPlan.status}`) }} | ||
.hr.q-mt-md.q-mb-xs | ||
footer | ||
div.row.justify-between | ||
p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose Core Members | ||
p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose {{ selectedDaoPlan.coreMembersCount }} | ||
//- TODO: Return after beta | ||
//- div.row.justify-between.q-mt-xs | ||
//- p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose Community Members | ||
//- p.q-pa-none.q-ma-none.text-sm.text-h-gray.leading-loose {{ selectedDaoPlan.communityMembersCount }} | ||
nav.q-mt-xl.full-width.row.justify-end | ||
q-btn.q-px-xl.rounded-border.text-bold.q-ml-xs( | ||
:label="$t('configuration.settings-plans-billing.plan.cta')" | ||
@click="state = STATES.UPDATING_PLAIN" | ||
color="secondary" | ||
no-caps | ||
rounded | ||
unelevated | ||
) | ||
.col-12.col-md-6(:class="{ 'q-pr-sm': $q.screen.gt.md, 'q-mt-md': $q.screen.lt.md }") | ||
widget(:title="$t('configuration.settings-plans-billing.history.title')" titleImage='/svg/briefcase.svg' bar).q-pa-none | ||
p.text-sm.text-h-gray.leading-loose.q-mt-md {{ $t('configuration.settings-plans-billing.history.description') }} | ||
</template> | ||
<style lang="stylus" scoped> | ||
</style> |
Oops, something went wrong.