2
0

fix(editor): 🐛 Custom currency payment input

This commit is contained in:
Baptiste Arnaud
2022-05-27 09:55:51 -07:00
parent 30d36b9a09
commit 2cb8330790
7 changed files with 39 additions and 33 deletions

View File

@ -18,6 +18,7 @@ FROM base AS builder
COPY --from=installer /app/ . COPY --from=installer /app/ .
COPY --from=pruner /app/out/full/ . COPY --from=pruner /app/out/full/ .
COPY ./apps/${SCOPE}/.env.docker ./apps/${SCOPE}/.env.production COPY ./apps/${SCOPE}/.env.docker ./apps/${SCOPE}/.env.production
COPY ./apps/${SCOPE}/.env.docker ./apps/${SCOPE}/.env.local
RUN apt-get -qy update && apt-get -qy install openssl RUN apt-get -qy update && apt-get -qy install openssl
RUN yarn turbo run build --scope=${SCOPE} --include-dependencies --no-deps RUN yarn turbo run build --scope=${SCOPE} --include-dependencies --no-deps
RUN find . -name node_modules | xargs rm -rf RUN find . -name node_modules | xargs rm -rf

View File

@ -14,7 +14,7 @@ import { DividerWithText } from './DividerWithText'
import { SocialLoginButtons } from './SocialLoginButtons' import { SocialLoginButtons } from './SocialLoginButtons'
import { useRouter } from 'next/router' import { useRouter } from 'next/router'
import { NextChakraLink } from 'components/nextChakra/NextChakraLink' import { NextChakraLink } from 'components/nextChakra/NextChakraLink'
import { isEmpty } from 'utils' import { isEmpty, isNotEmpty } from 'utils'
const hasNoAuthProvider = const hasNoAuthProvider =
(isEmpty(process.env.NEXT_PUBLIC_SMTP_FROM) || (isEmpty(process.env.NEXT_PUBLIC_SMTP_FROM) ||
@ -78,7 +78,7 @@ export const SignInForm = ({
return ( return (
<Stack spacing="4" w="330px"> <Stack spacing="4" w="330px">
<SocialLoginButtons /> <SocialLoginButtons />
{!isEmpty(process.env.NEXT_PUBLIC_SMTP_FROM) && {isNotEmpty(process.env.NEXT_PUBLIC_SMTP_FROM) &&
process.env.NEXT_PUBLIC_SMTP_AUTH_DISABLED !== 'true' && ( process.env.NEXT_PUBLIC_SMTP_AUTH_DISABLED !== 'true' && (
<> <>
<DividerWithText mt="6">Or with your email</DividerWithText> <DividerWithText mt="6">Or with your email</DividerWithText>

View File

@ -111,11 +111,11 @@ export const PaymentSettings = ({ options, onOptionsChange }: Props) => {
<Text>Currency:</Text> <Text>Currency:</Text>
<Select <Select
placeholder="Select option" placeholder="Select option"
value={options.currency.toLowerCase()} value={options.currency}
onChange={handleCurrencyChange} onChange={handleCurrencyChange}
> >
{currencies.map((currency) => ( {currencies.map((currency) => (
<option value={currency.code.toLowerCase()} key={currency.code}> <option value={currency.code} key={currency.code}>
{currency.code} {currency.code}
</option> </option>
))} ))}

View File

@ -46,6 +46,7 @@ test.describe('Payment input step', () => {
await expect(page.locator('text="Secret test key:"')).toBeHidden() await expect(page.locator('text="Secret test key:"')).toBeHidden()
await expect(page.locator('text="My Stripe Account"')).toBeVisible() await expect(page.locator('text="My Stripe Account"')).toBeVisible()
await page.fill('[placeholder="30.00"] >> nth=-1', '30.00') await page.fill('[placeholder="30.00"] >> nth=-1', '30.00')
await page.selectOption('select', 'EUR')
await page.click('text=Additional information') await page.click('text=Additional information')
await page.fill('[placeholder="John Smith"]', 'Baptiste') await page.fill('[placeholder="John Smith"]', 'Baptiste')
await page.fill('[placeholder="john@gmail.com"]', 'baptiste@typebot.io') await page.fill('[placeholder="john@gmail.com"]', 'baptiste@typebot.io')
@ -59,15 +60,14 @@ test.describe('Payment input step', () => {
.locator(`[placeholder="MM / YY"]`) .locator(`[placeholder="MM / YY"]`)
.fill('12 / 25') .fill('12 / 25')
await stripePaymentForm(page).locator(`[placeholder="CVC"]`).fill('240') await stripePaymentForm(page).locator(`[placeholder="CVC"]`).fill('240')
await stripePaymentForm(page).locator(`[placeholder="90210"]`).fill('90210') await typebotViewer(page).locator(`text="Pay 30€"`).click()
await typebotViewer(page).locator(`text="Pay 30$"`).click()
await expect( await expect(
typebotViewer(page).locator(`text="Your card was declined."`) typebotViewer(page).locator(`text="Your card was declined."`)
).toBeVisible() ).toBeVisible()
await stripePaymentForm(page) await stripePaymentForm(page)
.locator(`[placeholder="1234 1234 1234 1234"]`) .locator(`[placeholder="1234 1234 1234 1234"]`)
.fill('4242424242424242') .fill('4242424242424242')
await typebotViewer(page).locator(`text="Pay 30$"`).click() await typebotViewer(page).locator(`text="Pay 30"`).click()
await expect(typebotViewer(page).locator(`text="Success"`)).toBeVisible() await expect(typebotViewer(page).locator(`text="Success"`)).toBeVisible()
}) })
}) })

View File

@ -74,7 +74,9 @@ const handler = async (req: NextApiRequest, res: NextApiResponse) => {
isPreview && stripeKeys.test?.publicKey isPreview && stripeKeys.test?.publicKey
? stripeKeys.test.publicKey ? stripeKeys.test.publicKey
: stripeKeys.live.publicKey, : stripeKeys.live.publicKey,
amountLabel: `${amount / 100}${currencySymbols[inputOptions.currency]}`, amountLabel: `${amount / 100}${
currencySymbols[inputOptions.currency] ?? inputOptions.currency
}`,
}) })
} }
return methodNotAllowed(res) return methodNotAllowed(res)

View File

@ -4,7 +4,7 @@ import { Elements } from '@stripe/react-stripe-js'
import { createPaymentIntent } from 'services/stripe' import { createPaymentIntent } from 'services/stripe'
import { useTypebot } from 'contexts/TypebotContext' import { useTypebot } from 'contexts/TypebotContext'
import { PaymentInputOptions, Variable } from 'models' import { PaymentInputOptions, Variable } from 'models'
import { SendButton } from '../SendButton' import { SendButton, Spinner } from '../SendButton'
import { useFrame } from 'react-frame-component' import { useFrame } from 'react-frame-component'
import { initStripe } from '../../../../../../lib/stripe' import { initStripe } from '../../../../../../lib/stripe'
import { parseVariables } from 'services/variable' import { parseVariables } from 'services/variable'
@ -43,7 +43,7 @@ export const StripePaymentForm = ({ options, onSuccess }: Props) => {
// eslint-disable-next-line react-hooks/exhaustive-deps // eslint-disable-next-line react-hooks/exhaustive-deps
}, []) }, [])
if (!stripe || !clientSecret) return <></> if (!stripe || !clientSecret) return <Spinner className="text-blue-500" />
return ( return (
<Elements stripe={stripe} options={{ clientSecret }}> <Elements stripe={stripe} options={{ clientSecret }}>
<CheckoutForm <CheckoutForm

View File

@ -1,4 +1,4 @@
import React from 'react' import React, { SVGProps } from 'react'
import { SendIcon } from '../../../../assets/icons' import { SendIcon } from '../../../../assets/icons'
type SendButtonProps = { type SendButtonProps = {
@ -25,28 +25,7 @@ export const SendButton = ({
props.className props.className
} }
> >
{isLoading && ( {isLoading && <Spinner className="text-white" />}
<svg
className="animate-spin -ml-1 mr-3 h-5 w-5 text-white"
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
>
<circle
className="opacity-25"
cx="12"
cy="12"
r="10"
stroke="currentColor"
strokeWidth="4"
></circle>
<path
className="opacity-75"
fill="currentColor"
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
></path>
</svg>
)}
<span className={'xs:flex ' + (disableIcon ? '' : 'hidden')}> <span className={'xs:flex ' + (disableIcon ? '' : 'hidden')}>
{label} {label}
</span> </span>
@ -56,3 +35,27 @@ export const SendButton = ({
</button> </button>
) )
} }
export const Spinner = (props: SVGProps<SVGSVGElement>) => (
<svg
{...props}
className={'animate-spin -ml-1 mr-3 h-5 w-5 ' + props.className}
xmlns="http://www.w3.org/2000/svg"
fill="none"
viewBox="0 0 24 24"
>
<circle
className="opacity-25"
cx="12"
cy="12"
r="10"
stroke="currentColor"
strokeWidth="4"
></circle>
<path
className="opacity-75"
fill="currentColor"
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
></path>
</svg>
)