diff --git a/package.json b/package.json
index 8d8002b..b2096b2 100644
--- a/package.json
+++ b/package.json
@@ -59,6 +59,7 @@
"rimraf": "^2.6.2",
"rollup": "^0.54.1",
"rollup-plugin-babel": "^3.0.3",
+ "rollup-plugin-node-resolve": "^4.2.3",
"styled-components": "4.2.0"
},
"peerDependencies": {
diff --git a/rollup.config.js b/rollup.config.js
index 81e7041..f43c4f5 100644
--- a/rollup.config.js
+++ b/rollup.config.js
@@ -1,33 +1,47 @@
import babel from 'rollup-plugin-babel';
+// https://github.com/rollup/rollup/issues/346
+import resolve from 'rollup-plugin-node-resolve';
-export default {
+const globals = {
+ react: 'React',
+ payment: 'payment',
+ 'credit-card-type': 'creditCardType',
+ 'styled-components': 'styled'
+};
+
+const external = [
+ 'react',
+ 'credit-card-type',
+ 'payment',
+ 'styled-components'
+];
+
+const plugins = [
+ resolve({}),
+ babel({
+ presets: [['env', { modules: false }], 'react', 'flow'],
+ plugins: [
+ 'external-helpers',
+ 'transform-class-properties',
+ 'transform-object-rest-spread'
+ ]
+ })
+]
+
+export default [{
input: `${__dirname}/src/index.js`,
output: {
name: 'CreditCardInput',
file: `${__dirname}/lib/index.js`,
format: 'umd',
- globals: {
- react: 'React',
- payment: 'payment',
- 'credit-card-type': 'creditCardType',
- 'styled-components': 'styled'
- }
- },
- sourcemap: true,
- external: [
- 'react',
- 'credit-card-type',
- 'payment',
- 'styled-components'
- ],
- plugins: [
- babel({
- presets: [['env', { modules: false }], 'react', 'flow'],
- plugins: [
- 'external-helpers',
- 'transform-class-properties',
- 'transform-object-rest-spread'
- ]
- })
- ]
-};
+ globals
+ }
+}, {
+ input: `${__dirname}/src/hooks/use-credit-card-input.js`,
+ output: {
+ name: 'useCreditCardInput',
+ file: `${__dirname}/lib/useCreditCardInput.js`,
+ format: 'umd',
+ globals
+ }
+}].map(output => Object.assign(output, {external, plugins, sourcemap: true}));
diff --git a/src/credit-card-input.js b/src/credit-card-input.js
new file mode 100644
index 0000000..a316032
--- /dev/null
+++ b/src/credit-card-input.js
@@ -0,0 +1,214 @@
+// @flow
+
+import React, { useEffect } from 'react';
+import styled from 'styled-components';
+import useCreditCardInput from './hooks/use-credit-card-input';
+
+const Container = styled.div`
+ display: inline-block;
+ ${({ styled }) => ({ ...styled })};
+`;
+
+const FieldWrapper = styled.div`
+ display: flex;
+ align-items: center;
+ position: relative;
+ background-color: white;
+ padding: 10px;
+ border-radius: 3px;
+ overflow: hidden;
+ ${({ styled }) => ({ ...styled })};
+
+ &.is-invalid {
+ border: 1px solid #ff3860;
+ ${({ invalidStyled }) => ({ ...invalidStyled })};
+ }
+`;
+
+const CardImage = styled.img`
+ height: 1em;
+ ${({ styled }) => ({ ...styled })};
+`;
+
+const InputWrapper = styled.label`
+ align-items: center;
+ display: ${props => (props.isActive ? 'flex' : 'none')};
+ margin-left: 0.5em;
+ position: relative;
+ transition: transform 0.5s;
+ transform: translateX(${props => (props.translateX ? '4rem' : '0')});
+
+ &::after {
+ content: attr(data-max);
+ visibility: hidden;
+ height: 0;
+ }
+
+ & .credit-card-input {
+ border: 0px;
+ position: absolute;
+ width: 100%;
+ font-size: 1em;
+ ${({ inputStyled }) => ({ ...inputStyled })};
+
+ &:focus {
+ outline: 0px;
+ }
+ }
+
+ & .zip-input {
+ display: ${props => (props.isZipActive ? 'flex' : 'none')};
+ }
+`;
+
+const DangerText = styled.p`
+ font-size: 0.8rem;
+ margin: 5px 0 0 0;
+ color: #ff3860;
+ ${({ styled }) => ({ ...styled })};
+`;
+
+type CreditCardInputProps = {
+ cardTypes: Object,
+ onError?: Function,
+ cardExpiryInputProps: Object,
+ cardNumberInputProps: Object,
+ cardCVCInputProps: Object,
+ cardZipInputProps: Object,
+ cardImageClassName: string,
+ cardImageStyle: Object,
+ containerClassName: string,
+ containerStyle: Object,
+ dangerTextClassName: string,
+ dangerTextStyle: Object,
+ fieldClassName: string,
+ fieldStyle: Object,
+ images: Object,
+ inputClassName: string,
+ inputStyle: Object,
+ invalidClassName: string,
+ invalidStyle: Object,
+ customTextLabels: Object
+};
+
+const defaultProps: CreditCardInputProps = {
+ cardExpiryInputProps: {},
+ cardNumberInputProps: {},
+ cardCVCInputProps: {},
+ cardZipInputProps: {},
+ cardImageClassName: '',
+ cardImageStyle: {},
+ containerClassName: '',
+ containerStyle: {},
+ dangerTextClassName: '',
+ dangerTextStyle: {},
+ fieldClassName: '',
+ fieldStyle: {},
+ inputClassName: '',
+ inputStyle: {},
+ invalidClassName: 'is-invalid',
+ invalidStyle: {},
+ customTextLabels: {}
+};
+
+const CreditCardInput = (props: CreditCardInputProps) => {
+ const {
+ state,
+ cardNumberProps,
+ cardExpiryProps,
+ cardCvcProps,
+ cardZipProps
+ } = useCreditCardInput(props);
+
+ const {
+ cardImageClassName,
+ cardImageStyle,
+ containerClassName,
+ containerStyle,
+ dangerTextClassName,
+ dangerTextStyle,
+ fieldClassName,
+ fieldStyle,
+ inputStyle,
+ invalidClassName,
+ invalidStyle
+ } = props;
+
+ const { cardImage, enableZipInput, errorText, showZip } = state;
+
+ const setFieldInvalid = (errorText: string) => {
+ // $FlowFixMe
+ document.getElementById('field-wrapper').classList.add(invalidClassName);
+ };
+
+ const setFieldValid = () => {
+ // $FlowFixMe
+ document.getElementById('field-wrapper').classList.remove(invalidClassName);
+ };
+
+ useEffect(() => {
+ if (errorText) {
+ setFieldInvalid(errorText);
+ } else {
+ setFieldValid();
+ }
+ }, [errorText]);
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {errorText && (
+
+ {errorText}
+
+ )}
+
+ );
+};
+
+CreditCardInput.defaultProps = defaultProps;
+
+export default CreditCardInput;
diff --git a/src/events/events.js b/src/events/events.js
new file mode 100644
index 0000000..4272766
--- /dev/null
+++ b/src/events/events.js
@@ -0,0 +1,30 @@
+export default function() {
+ var topics = {};
+ var hOP = topics.hasOwnProperty;
+
+ return {
+ subscribe: function(topic, listener) {
+ // Create the topic's object if not yet created
+ if (!hOP.call(topics, topic)) topics[topic] = [];
+
+ // Add the listener to queue
+ var index = topics[topic].push(listener) - 1;
+
+ // Provide handle back for removal of topic
+ return {
+ remove: function() {
+ delete topics[topic][index];
+ }
+ };
+ },
+ publish: function(topic, info) {
+ // If the topic doesn't exist, or there's no listeners in queue, just leave
+ if (!hOP.call(topics, topic)) return;
+
+ // Cycle through topics queue, fire!
+ topics[topic].forEach(function(item) {
+ item(info !== undefined ? info : {});
+ });
+ }
+ };
+}
diff --git a/src/events/index.js b/src/events/index.js
new file mode 100644
index 0000000..b890785
--- /dev/null
+++ b/src/events/index.js
@@ -0,0 +1,9 @@
+import events from './events';
+import * as types from './types';
+
+export default function() {
+ return {
+ ...events(),
+ types
+ };
+}
diff --git a/src/events/types.js b/src/events/types.js
new file mode 100644
index 0000000..3ee911e
--- /dev/null
+++ b/src/events/types.js
@@ -0,0 +1,4 @@
+export const FOCUS_ON_CARD_NUMBER = Symbol('FOCUS_ON_CARD_NUMBER');
+export const FOCUS_ON_CARD_EXPIRY = Symbol('FOCUS_ON_CARD_EXPIRY');
+export const FOCUS_ON_CARD_CVC = Symbol('FOCUS_ON_CARD_CVC');
+export const FOCUS_ON_CARD_ZIP = Symbol('FOCUS_ON_CARD_ZIP');
diff --git a/src/hooks/use-card-cvc-input.js b/src/hooks/use-card-cvc-input.js
new file mode 100644
index 0000000..3da6cec
--- /dev/null
+++ b/src/hooks/use-card-cvc-input.js
@@ -0,0 +1,126 @@
+// @flow
+
+import { useEffect, useRef } from 'react';
+import payment from 'payment';
+import {
+ formatCvc,
+ hasCVCReachedMaxLength,
+ isHighlighted
+} from '../utils/formatter';
+import checkIsNumeric from '../utils/check-is-numeric';
+import handleKeyDown from '../utils/handle-key-down';
+import { CHANGE_ERROR_TEXT } from '../reducer/actions';
+
+type CardCVCInputProps = {
+ cardCVCInputProps: Object,
+ dispatch: Function,
+ state: Object
+};
+
+const defaultProps: CardCVCInputProps = {
+ cardCVCInputProps: {}
+};
+
+const useCardCVCInput = (props: CardCVCInputProps = defaultProps) => {
+ const cvcField = useRef(null);
+ const { cardCVCInputProps, dispatch, state } = props;
+ const {
+ cardNumber,
+ customTextLabels,
+ enableZipInput,
+ events,
+ inputClassName,
+ showZip
+ } = state;
+
+ useEffect(() => {
+ const subscription = events.subscribe(
+ events.types.FOCUS_ON_CARD_CVC,
+ () => {
+ cvcField.current.focus();
+ }
+ );
+
+ return () => {
+ subscription.remove();
+ };
+ }, []);
+
+ const setFieldInvalid = (errorText: string) => {
+ const { onError } = cardCVCInputProps;
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText });
+ if (onError) {
+ onError({ error: errorText });
+ }
+ };
+
+ const setFieldValid = () => {
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText: null });
+ };
+
+ const handleCardCVCBlur = (
+ { onBlur }: { onBlur?: ?Function } = { onBlur: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ if (!payment.fns.validateCardCVC(e.target.value)) {
+ setFieldInvalid(customTextLabels.invalidCvc || 'CVC is invalid');
+ }
+ cardCVCInputProps.onBlur && cardCVCInputProps.onBlur(e);
+ onBlur && onBlur(e);
+ };
+
+ const handleCardCVCChange = (
+ { onChange }: { onChange?: ?Function } = { onChange: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ const value = formatCvc(e.target.value);
+ cvcField.current.value = value;
+ const CVC = value;
+ const CVCLength = CVC.length;
+ const isZipFieldAvailable = enableZipInput && showZip;
+ const cardType = payment.fns.cardType(cardNumber);
+
+ setFieldValid();
+ if (CVCLength >= 4) {
+ if (!payment.fns.validateCardCVC(CVC, cardType)) {
+ setFieldInvalid(customTextLabels.invalidCvc || 'CVC is invalid');
+ }
+ }
+
+ if (isZipFieldAvailable && hasCVCReachedMaxLength(cardType, CVCLength)) {
+ events.publish(events.types.FOCUS_ON_CARD_ZIP);
+ }
+
+ cardCVCInputProps.onChange && cardCVCInputProps.onChange(e);
+ onChange && onChange(e);
+ };
+
+ const handleCardCVCKeyPress = (e: any) => {
+ const cardType = payment.fns.cardType(cardNumber);
+ const value = e.target.value;
+ checkIsNumeric(e);
+ if (value && !isHighlighted()) {
+ const valueLength = value.split(' / ').join('').length;
+ if (hasCVCReachedMaxLength(cardType, valueLength)) {
+ e.preventDefault();
+ }
+ }
+ };
+
+ return {
+ id: 'cvc',
+ ref: cvcField,
+ maxLength: '5',
+ autoComplete: 'off',
+ className: `credit-card-input ${inputClassName}`,
+ placeholder: customTextLabels.cvcPlaceholder || 'CVC',
+ type: 'tel',
+ ...cardCVCInputProps,
+ onBlur: handleCardCVCBlur(),
+ onChange: handleCardCVCChange(),
+ onKeyDown: handleKeyDown(() => {
+ events.publish(events.types.FOCUS_ON_CARD_EXPIRY);
+ }),
+ onKeyPress: handleCardCVCKeyPress
+ };
+};
+
+export default useCardCVCInput;
diff --git a/src/hooks/use-card-expiry-input.js b/src/hooks/use-card-expiry-input.js
new file mode 100644
index 0000000..2b56ce7
--- /dev/null
+++ b/src/hooks/use-card-expiry-input.js
@@ -0,0 +1,120 @@
+// @flow
+
+import { useEffect, useRef } from 'react';
+import { CHANGE_ERROR_TEXT } from '../reducer/actions';
+import { formatExpiry, isHighlighted } from '../utils/formatter';
+import isExpiryInvalid from '../utils/is-expiry-invalid';
+import checkIsNumeric from '../utils/check-is-numeric';
+import handleKeyDown from '../utils/handle-key-down';
+import isMonthDashKey from '../utils/is-month-dash-key';
+
+type CardExpiryInputProps = {
+ cardExpiryInputProps: Object,
+ dispatch: Function,
+ state: Object
+};
+
+const defaultProps: CardExpiryInputProps = {
+ cardExpiryInputProps: {}
+};
+
+const useCardExpiryInput = (props: CardExpiryInputProps = defaultProps) => {
+ const cardExpiryField = useRef(null);
+ const { cardExpiryInputProps, dispatch, state } = props;
+ const { customTextLabels, events, inputClassName } = state;
+
+ useEffect(() => {
+ const subscription = events.subscribe(
+ events.types.FOCUS_ON_CARD_EXPIRY,
+ () => {
+ cardExpiryField.current.focus();
+ }
+ );
+
+ return () => {
+ subscription.remove();
+ };
+ }, []);
+
+ const setFieldInvalid = (errorText: string) => {
+ const { onError } = cardExpiryInputProps;
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText });
+ if (onError) {
+ onError({ error: errorText });
+ }
+ };
+
+ const setFieldValid = () => {
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText: null });
+ };
+
+ const handleCardExpiryBlur = (
+ { onBlur }: { onBlur?: ?Function } = { onBlur: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ const cardExpiry = e.target.value.split(' / ').join('/');
+ const expiryError = isExpiryInvalid(
+ cardExpiry,
+ customTextLabels.expiryError
+ );
+ if (expiryError) {
+ setFieldInvalid(expiryError);
+ }
+
+ cardExpiryInputProps.onBlur && cardExpiryInputProps.onBlur(e);
+ onBlur && onBlur(e);
+ };
+
+ const handleCardExpiryChange = (
+ { onChange }: { onChange?: ?Function } = { onChange: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ cardExpiryField.current.value = formatExpiry(e);
+ const value = cardExpiryField.current.value.split(' / ').join('/');
+
+ setFieldValid();
+
+ const expiryError = isExpiryInvalid(value, customTextLabels.expiryError);
+ if (value.length > 4) {
+ if (expiryError) {
+ setFieldInvalid(expiryError);
+ } else {
+ events.publish(events.types.FOCUS_ON_CARD_CVC);
+ }
+ }
+
+ cardExpiryInputProps.onChange && cardExpiryInputProps.onChange(e);
+ onChange && onChange(e);
+ };
+
+ const handleCardExpiryKeyPress = (e: any) => {
+ const value = e.target.value;
+
+ if (!isMonthDashKey(e)) {
+ checkIsNumeric(e);
+ }
+
+ if (value && !isHighlighted()) {
+ const valueLength = value.split(' / ').join('').length;
+ if (valueLength >= 4) {
+ e.preventDefault();
+ }
+ }
+ };
+
+ return {
+ id: 'card-expiry',
+ ref: cardExpiryField,
+ autoComplete: 'cc-exp',
+ className: `credit-card-input ${inputClassName}`,
+ placeholder: customTextLabels.expiryPlaceholder || 'MM/YY',
+ type: 'tel',
+ ...cardExpiryInputProps,
+ onBlur: handleCardExpiryBlur(),
+ onChange: handleCardExpiryChange(),
+ onKeyDown: handleKeyDown(() => {
+ events.publish(events.types.FOCUS_ON_CARD_NUMBER);
+ }),
+ onKeyPress: handleCardExpiryKeyPress
+ };
+};
+
+export default useCardExpiryInput;
diff --git a/src/hooks/use-card-number-input.js b/src/hooks/use-card-number-input.js
new file mode 100644
index 0000000..6d1737e
--- /dev/null
+++ b/src/hooks/use-card-number-input.js
@@ -0,0 +1,157 @@
+// @flow
+
+import { useEffect, useRef } from 'react';
+import payment from 'payment';
+import creditCardType from 'credit-card-type';
+import checkIsNumeric from '../utils/check-is-numeric';
+import {
+ formatCardNumber,
+ hasCardNumberReachedMaxLength,
+ isHighlighted
+} from '../utils/formatter';
+import {
+ CHANGE_CARD_IMAGE,
+ CHANGE_CARD_NUMBER,
+ CHANGE_ERROR_TEXT,
+ SHOW_ZIP
+} from '../reducer/actions';
+
+type CardNumberInputProps = {
+ cardNumberInputProps: Object,
+ dispatch: Function,
+ state: Object
+};
+
+const defaultProps: CardNumberInputProps = {
+ cardNumberInputProps: {}
+};
+
+const useCardNumberInput = (props: CardNumberInputProps = defaultProps) => {
+ const cardNumberField = useRef(null);
+ const { cardNumberInputProps, dispatch, state } = props;
+ const {
+ cardTypes: CARD_TYPES,
+ customTextLabels,
+ enableZipInput,
+ events,
+ images,
+ inputClassName
+ } = state;
+
+ useEffect(() => {
+ const subscription = events.subscribe(
+ events.types.FOCUS_ON_CARD_NUMBER,
+ () => {
+ cardNumberField.current.focus();
+ }
+ );
+
+ return () => {
+ subscription.remove();
+ };
+ }, []);
+
+ const setFieldInvalid = (errorText: string) => {
+ const { onError } = cardNumberInputProps;
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText });
+ if (onError) {
+ onError({ error: errorText });
+ }
+ };
+
+ const setFieldValid = () => {
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText: null });
+ };
+
+ const handleCardNumberBlur = (
+ { onBlur }: { onBlur?: ?Function } = { onBlur: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ if (!payment.fns.validateCardNumber(e.target.value)) {
+ setFieldInvalid(
+ customTextLabels.invalidCardNumber || 'Card number is invalid'
+ );
+ }
+
+ cardNumberInputProps.onBlur && cardNumberInputProps.onBlur(e);
+ onBlur && onBlur(e);
+ };
+
+ const handleCardNumberChange = (
+ { onChange }: { onChange?: ?Function } = { onChange: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ const cardNumber = e.target.value;
+ const cardNumberLength = cardNumber.split(' ').join('').length;
+ const cardType = payment.fns.cardType(cardNumber);
+ const cardTypeInfo =
+ creditCardType.getTypeInfo(creditCardType.types[CARD_TYPES[cardType]]) ||
+ {};
+ const cardTypeLengths = cardTypeInfo.lengths || [16];
+
+ cardNumberField.current.value = formatCardNumber(cardNumber);
+
+ dispatch({
+ type: CHANGE_CARD_IMAGE,
+ cardImage: images[cardType] || images.placeholder
+ });
+ dispatch({
+ type: CHANGE_CARD_NUMBER,
+ cardNumber
+ });
+
+ if (enableZipInput) {
+ dispatch({
+ type: SHOW_ZIP,
+ showZip: cardNumberLength >= 6
+ });
+ }
+
+ setFieldValid();
+ if (cardTypeLengths) {
+ const lastCardTypeLength = cardTypeLengths[cardTypeLengths.length - 1];
+ for (let length of cardTypeLengths) {
+ if (
+ length === cardNumberLength &&
+ payment.fns.validateCardNumber(cardNumber)
+ ) {
+ events.publish(events.types.FOCUS_ON_CARD_EXPIRY);
+ break;
+ }
+ if (cardNumberLength === lastCardTypeLength) {
+ setFieldInvalid(
+ customTextLabels.invalidCardNumber || 'Card number is invalid'
+ );
+ }
+ }
+ }
+
+ cardNumberInputProps.onChange && cardNumberInputProps.onChange(e);
+ onChange && onChange(e);
+ };
+
+ const handleCardNumberKeyPress = (e: any) => {
+ const value = e.target.value;
+ checkIsNumeric(e);
+ if (value && !isHighlighted()) {
+ const valueLength = value.split(' ').join('').length;
+ if (hasCardNumberReachedMaxLength(value, valueLength)) {
+ e.preventDefault();
+ }
+ }
+ };
+
+ return {
+ id: 'card-number',
+ ref: cardNumberField,
+ maxLength: '19',
+ autoComplete: 'cc-number',
+ className: `credit-card-input ${inputClassName}`,
+ placeholder: customTextLabels.cardNumberPlaceholder || 'Card number',
+ type: 'tel',
+ ...cardNumberInputProps,
+ onBlur: handleCardNumberBlur(),
+ onChange: handleCardNumberChange(),
+ onKeyPress: handleCardNumberKeyPress
+ };
+};
+
+export default useCardNumberInput;
diff --git a/src/hooks/use-card-zip-input.js b/src/hooks/use-card-zip-input.js
new file mode 100644
index 0000000..b752517
--- /dev/null
+++ b/src/hooks/use-card-zip-input.js
@@ -0,0 +1,108 @@
+// @flow
+
+import { useEffect, useRef } from 'react';
+import payment from 'payment';
+import { hasZipReachedMaxLength, isHighlighted } from '../utils/formatter';
+import isZipValid from '../utils/is-zip-valid';
+import checkIsNumeric from '../utils/check-is-numeric';
+import handleKeyDown from '../utils/handle-key-down';
+import { CHANGE_ERROR_TEXT } from '../reducer/actions';
+
+type CardZipInputProps = {
+ cardZipInputProps: Object,
+ dispatch: Function,
+ state: Object
+};
+
+const defaultProps: CardZipInputProps = {
+ cardZipInputProps: {}
+};
+
+const useCardZipInput = (props: CardZipInputProps = defaultProps) => {
+ const zipField = useRef(null);
+ const { cardZipInputProps, dispatch, state } = props;
+ const { cardNumber, customTextLabels, events, inputClassName } = state;
+
+ useEffect(() => {
+ const subscription = events.subscribe(
+ events.types.FOCUS_ON_CARD_ZIP,
+ () => {
+ zipField.current.focus();
+ }
+ );
+
+ return () => {
+ subscription.remove();
+ };
+ }, []);
+
+ const setFieldInvalid = (errorText: string) => {
+ const { onError } = cardZipInputProps;
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText });
+ if (onError) {
+ onError({ error: errorText });
+ }
+ };
+
+ const setFieldValid = () => {
+ dispatch({ type: CHANGE_ERROR_TEXT, errorText: null });
+ };
+
+ const handleCardZipBlur = (
+ { onBlur }: { onBlur?: ?Function } = { onBlur: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ if (!isZipValid(e.target.value)) {
+ setFieldInvalid(customTextLabels.invalidZipCode || 'Zip code is invalid');
+ }
+
+ cardZipInputProps.onBlur && cardZipInputProps.onBlur(e);
+ onBlur && onBlur(e);
+ };
+
+ const handleCardZipChange = (
+ { onChange }: { onChange?: ?Function } = { onChange: null }
+ ) => (e: SyntheticInputEvent<*>) => {
+ const zip = e.target.value;
+ const zipLength = zip.length;
+
+ setFieldValid();
+
+ if (zipLength >= 5 && !isZipValid(zip)) {
+ setFieldInvalid(customTextLabels.invalidZipCode || 'Zip code is invalid');
+ }
+
+ cardZipInputProps.onChange && cardZipInputProps.onChange(e);
+ onChange && onChange(e);
+ };
+
+ const handleCardZipKeyPress = (e: any) => {
+ const cardType = payment.fns.cardType(cardNumber);
+ const value = e.target.value;
+ checkIsNumeric(e);
+ if (value && !isHighlighted()) {
+ const valueLength = value.split(' / ').join('').length;
+ if (hasZipReachedMaxLength(cardType, valueLength)) {
+ e.preventDefault();
+ }
+ }
+ };
+
+ return {
+ id: 'zip',
+ ref: zipField,
+ maxLength: '6',
+ className: `credit-card-input zip-input ${inputClassName}`,
+ pattern: '[0-9]*',
+ placeholder: customTextLabels.zipPlaceholder || 'Zip',
+ type: 'text',
+ ...cardZipInputProps,
+ onBlur: handleCardZipBlur(),
+ onChange: handleCardZipChange(),
+ onKeyDown: handleKeyDown(() => {
+ events.publish(events.types.FOCUS_ON_CARD_CVC);
+ }),
+ onKeyPress: handleCardZipKeyPress
+ };
+};
+
+export default useCardZipInput;
diff --git a/src/hooks/use-credit-card-input.js b/src/hooks/use-credit-card-input.js
new file mode 100644
index 0000000..4545705
--- /dev/null
+++ b/src/hooks/use-credit-card-input.js
@@ -0,0 +1,99 @@
+// @flow
+
+import { useEffect, useReducer } from 'react';
+import useCardNumberInput from './use-card-number-input';
+import useCardExpiryInput from './use-card-expiry-input';
+import useCardCvcInput from './use-card-cvc-input';
+import useCardZipInput from './use-card-zip-input';
+import { inititalState, reducer } from '../reducer';
+import {
+ CHANGE_CARD_TYPES,
+ CHANGE_IMAGES,
+ CHANGE_CUSTOM_TEXT_LABELS,
+ CHANGE_INPUT_CLASS_NAME,
+ ENABLE_ZIP_INPUT
+} from '../reducer/actions';
+
+type CreditCardInputProps = {
+ cardTypes: Object,
+ onError?: Function,
+ cardExpiryInputProps: Object,
+ cardNumberInputProps: Object,
+ cardCVCInputProps: Object,
+ cardZipInputProps: Object,
+ images: Object,
+ inputClassName: string,
+ customTextLabels: Object
+};
+
+const defaultProps: CreditCardInputProps = {
+ cardExpiryInputProps: {},
+ cardNumberInputProps: {},
+ cardCVCInputProps: {},
+ cardZipInputProps: {},
+ inputClassName: '',
+ customTextLabels: {}
+};
+
+const useCreditCardInput = (props: CreditCardInputProps = defaultProps) => {
+ const [state, dispatch] = useReducer(reducer, inititalState);
+
+ const {
+ cardCVCInputProps,
+ cardZipInputProps,
+ cardExpiryInputProps,
+ cardNumberInputProps,
+ onError
+ } = props;
+
+ const { errorText } = state;
+
+ useEffect(() => {
+ const {
+ images,
+ inputClassName,
+ cardTypes,
+ customTextLabels,
+ enableZipInput
+ } = props;
+ dispatch({ type: CHANGE_CARD_TYPES, cardTypes });
+ dispatch({ type: CHANGE_IMAGES, images });
+ dispatch({ type: ENABLE_ZIP_INPUT, enableZipInput });
+ dispatch({ type: CHANGE_CUSTOM_TEXT_LABELS, customTextLabels });
+ dispatch({ type: CHANGE_INPUT_CLASS_NAME, inputClassName });
+ }, []);
+
+ const setFieldInvalid = (errorText: string) => {
+ if (onError) {
+ onError({ error: errorText });
+ }
+ };
+
+ const setFieldValid = () => {};
+
+ useEffect(() => {
+ if (errorText) {
+ setFieldInvalid(errorText);
+ } else {
+ setFieldValid();
+ }
+ }, [errorText]);
+
+ return {
+ cardNumberProps: useCardNumberInput({
+ cardNumberInputProps,
+ dispatch,
+ state
+ }),
+ cardExpiryProps: useCardExpiryInput({
+ cardExpiryInputProps,
+ dispatch,
+ state
+ }),
+ cardCvcProps: useCardCvcInput({ cardCVCInputProps, dispatch, state }),
+ cardZipProps: useCardZipInput({ cardZipInputProps, dispatch, state }),
+ state
+ };
+};
+
+export default useCreditCardInput;
diff --git a/src/index.js b/src/index.js
index b191cee..ee6d1b4 100644
--- a/src/index.js
+++ b/src/index.js
@@ -1,633 +1,5 @@
-// @flow
-
-import React, { Component } from 'react';
-import payment from 'payment';
-import creditCardType from 'credit-card-type';
-import styled from 'styled-components';
-
-import {
- formatCardNumber,
- formatExpiry,
- formatCvc,
- hasCardNumberReachedMaxLength,
- hasCVCReachedMaxLength,
- hasZipReachedMaxLength,
- isHighlighted
-} from './utils/formatter';
-import images from './utils/images';
-import isExpiryInvalid from './utils/is-expiry-invalid';
-import isZipValid from './utils/is-zip-valid';
-
-const Container = styled.div`
- display: inline-block;
- ${({ styled }) => ({ ...styled })};
-`;
-
-const FieldWrapper = styled.div`
- display: flex;
- align-items: center;
- position: relative;
- background-color: white;
- padding: 10px;
- border-radius: 3px;
- overflow: hidden;
- ${({ styled }) => ({ ...styled })};
-
- &.is-invalid {
- border: 1px solid #ff3860;
- ${({ invalidStyled }) => ({ ...invalidStyled })};
- }
-`;
-
-const CardImage = styled.img`
- height: 1em;
- ${({ styled }) => ({ ...styled })};
-`;
-
-const InputWrapper = styled.label`
- align-items: center;
- display: ${props => (props.isActive ? 'flex' : 'none')};
- margin-left: 0.5em;
- position: relative;
- transition: transform 0.5s;
- transform: translateX(${props => (props.translateX ? '4rem' : '0')});
-
- &::after {
- content: attr(data-max);
- visibility: hidden;
- height: 0;
- }
-
- & .credit-card-input {
- border: 0px;
- position: absolute;
- width: 100%;
- font-size: 1em;
- ${({ inputStyled }) => ({ ...inputStyled })};
-
- &:focus {
- outline: 0px;
- }
- }
-
- & .zip-input {
- display: ${props => (props.isZipActive ? 'flex' : 'none')};
- }
-`;
-
-const DangerText = styled.p`
- font-size: 0.8rem;
- margin: 5px 0 0 0;
- color: #ff3860;
- ${({ styled }) => ({ ...styled })};
-`;
-
-const BACKSPACE_KEY_CODE = 8;
-const CARD_TYPES = {
- mastercard: 'MASTERCARD',
- visa: 'VISA',
- amex: 'AMERICAN_EXPRESS'
-};
-
-type Props = {
- CARD_TYPES: Object,
- cardCVCInputRenderer: Function,
- cardExpiryInputRenderer: Function,
- cardNumberInputRenderer: Function,
- cardZipInputRenderer: Function,
- onError?: Function,
- cardExpiryInputProps: Object,
- cardNumberInputProps: Object,
- cardCVCInputProps: Object,
- cardZipInputProps: Object,
- cardImageClassName: string,
- cardImageStyle: Object,
- containerClassName: string,
- containerStyle: Object,
- dangerTextClassName: string,
- dangerTextStyle: Object,
- fieldClassName: string,
- fieldStyle: Object,
- enableZipInput: boolean,
- images: Object,
- inputComponent: Function | Object | string,
- inputClassName: string,
- inputStyle: Object,
- invalidClassName: string,
- invalidStyle: Object,
- customTextLabels: Object
-};
-type State = {
- cardImage: string,
- cardNumberLength: number,
- cardNumber: ?string,
- errorText: ?string,
- showZip: boolean
-};
-
-const inputRenderer = ({ props }: Object) => ;
-
-class CreditCardInput extends Component {
- cardExpiryField: any;
- cardNumberField: any;
- cvcField: any;
- zipField: any;
-
- static defaultProps = {
- cardCVCInputRenderer: inputRenderer,
- cardExpiryInputRenderer: inputRenderer,
- cardNumberInputRenderer: inputRenderer,
- cardZipInputRenderer: inputRenderer,
- cardExpiryInputProps: {},
- cardNumberInputProps: {},
- cardCVCInputProps: {},
- cardZipInputProps: {},
- cardImageClassName: '',
- cardImageStyle: {},
- containerClassName: '',
- containerStyle: {},
- dangerTextClassName: '',
- dangerTextStyle: {},
- enableZipInput: false,
- fieldClassName: '',
- fieldStyle: {},
- inputComponent: 'input',
- inputClassName: '',
- inputStyle: {},
- invalidClassName: 'is-invalid',
- invalidStyle: {},
- customTextLabels: {}
- };
-
- constructor(props: Props) {
- super(props);
- this.CARD_TYPES = Object.assign({}, CARD_TYPES, props.CARD_TYPES);
- this.images = Object.assign({}, images, props.images);
- this.state = {
- cardImage: this.images.placeholder,
- cardNumberLength: 0,
- cardNumber: null,
- errorText: null,
- showZip: false
- };
- }
-
- componentDidMount = () => {
- this.setState({ cardNumber: this.cardNumberField.value }, () => {
- const cardType = payment.fns.cardType(this.state.cardNumber);
- const images = this.images;
- this.setState({
- cardImage: images[cardType] || images.placeholder
- });
- });
- };
-
- isMonthDashKey = ({ key, target: { value } } = {}) => {
- return !value.match(/[/-]/) && /^[/-]$/.test(key);
- };
-
- checkIsNumeric = (e: any) => {
- if (!/^\d*$/.test(e.key)) {
- e.preventDefault();
- }
- };
-
- handleCardNumberBlur = (
- { onBlur }: { onBlur?: ?Function } = { onBlur: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
- if (!payment.fns.validateCardNumber(e.target.value)) {
- this.setFieldInvalid(
- customTextLabels.invalidCardNumber || 'Card number is invalid',
- 'cardNumber'
- );
- }
-
- const { cardNumberInputProps } = this.props;
- cardNumberInputProps.onBlur && cardNumberInputProps.onBlur(e);
- onBlur && onBlur(e);
- };
-
- handleCardNumberChange = (
- { onChange }: { onChange?: ?Function } = { onChange: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const {
- customTextLabels,
- enableZipInput,
- cardNumberInputProps
- } = this.props;
- const images = this.images;
- const cardNumber = e.target.value;
- const cardNumberLength = cardNumber.split(' ').join('').length;
- const cardType = payment.fns.cardType(cardNumber);
- const cardTypeInfo =
- creditCardType.getTypeInfo(
- creditCardType.types[this.CARD_TYPES[cardType]]
- ) || {};
- const cardTypeLengths = cardTypeInfo.lengths || [16];
-
- this.cardNumberField.value = formatCardNumber(cardNumber);
-
- this.setState({
- cardImage: images[cardType] || images.placeholder,
- cardNumber
- });
-
- if (enableZipInput) {
- this.setState({ showZip: cardNumberLength >= 6 });
- }
-
- this.setFieldValid();
- if (cardTypeLengths) {
- const lastCardTypeLength = cardTypeLengths[cardTypeLengths.length - 1];
- for (let length of cardTypeLengths) {
- if (
- length === cardNumberLength &&
- payment.fns.validateCardNumber(cardNumber)
- ) {
- this.cardExpiryField.focus();
- break;
- }
- if (cardNumberLength === lastCardTypeLength) {
- this.setFieldInvalid(
- customTextLabels.invalidCardNumber || 'Card number is invalid',
- 'cardNumber'
- );
- }
- }
- }
-
- cardNumberInputProps.onChange && cardNumberInputProps.onChange(e);
- onChange && onChange(e);
- };
-
- handleCardNumberKeyPress = (e: any) => {
- const value = e.target.value;
- this.checkIsNumeric(e);
- if (value && !isHighlighted()) {
- const valueLength = value.split(' ').join('').length;
- if (hasCardNumberReachedMaxLength(value, valueLength)) {
- e.preventDefault();
- }
- }
- };
-
- handleCardExpiryBlur = (
- { onBlur }: { onBlur?: ?Function } = { onBlur: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
- const cardExpiry = e.target.value.split(' / ').join('/');
- const expiryError = isExpiryInvalid(
- cardExpiry,
- customTextLabels.expiryError
- );
- if (expiryError) {
- this.setFieldInvalid(expiryError, 'cardExpiry');
- }
-
- const { cardExpiryInputProps } = this.props;
- cardExpiryInputProps.onBlur && cardExpiryInputProps.onBlur(e);
- onBlur && onBlur(e);
- };
-
- handleCardExpiryChange = (
- { onChange }: { onChange?: ?Function } = { onChange: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
-
- this.cardExpiryField.value = formatExpiry(e);
- const value = this.cardExpiryField.value.split(' / ').join('/');
-
- this.setFieldValid();
-
- const expiryError = isExpiryInvalid(value, customTextLabels.expiryError);
- if (value.length > 4) {
- if (expiryError) {
- this.setFieldInvalid(expiryError, 'cardExpiry');
- } else {
- this.cvcField.focus();
- }
- }
-
- const { cardExpiryInputProps } = this.props;
- cardExpiryInputProps.onChange && cardExpiryInputProps.onChange(e);
- onChange && onChange(e);
- };
-
- handleCardExpiryKeyPress = (e: any) => {
- const value = e.target.value;
-
- if (!this.isMonthDashKey(e)) {
- this.checkIsNumeric(e);
- }
-
- if (value && !isHighlighted()) {
- const valueLength = value.split(' / ').join('').length;
- if (valueLength >= 4) {
- e.preventDefault();
- }
- }
- };
-
- handleCardCVCBlur = (
- { onBlur }: { onBlur?: ?Function } = { onBlur: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
- if (!payment.fns.validateCardCVC(e.target.value)) {
- this.setFieldInvalid(
- customTextLabels.invalidCvc || 'CVC is invalid',
- 'cardCVC'
- );
- }
-
- const { cardCVCInputProps } = this.props;
- cardCVCInputProps.onBlur && cardCVCInputProps.onBlur(e);
- onBlur && onBlur(e);
- };
-
- handleCardCVCChange = (
- { onChange }: { onChange?: ?Function } = { onChange: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
- const value = formatCvc(e.target.value);
- this.cvcField.value = value;
- const CVC = value;
- const CVCLength = CVC.length;
- const isZipFieldAvailable = this.props.enableZipInput && this.state.showZip;
- const cardType = payment.fns.cardType(this.state.cardNumber);
-
- this.setFieldValid();
- if (CVCLength >= 4) {
- if (!payment.fns.validateCardCVC(CVC, cardType)) {
- this.setFieldInvalid(
- customTextLabels.invalidCvc || 'CVC is invalid',
- 'cardCVC'
- );
- }
- }
-
- if (isZipFieldAvailable && hasCVCReachedMaxLength(cardType, CVCLength)) {
- this.zipField.focus();
- }
-
- const { cardCVCInputProps } = this.props;
- cardCVCInputProps.onChange && cardCVCInputProps.onChange(e);
- onChange && onChange(e);
- };
-
- handleCardCVCKeyPress = (e: any) => {
- const cardType = payment.fns.cardType(this.state.cardNumber);
- const value = e.target.value;
- this.checkIsNumeric(e);
- if (value && !isHighlighted()) {
- const valueLength = value.split(' / ').join('').length;
- if (hasCVCReachedMaxLength(cardType, valueLength)) {
- e.preventDefault();
- }
- }
- };
-
- handleCardZipBlur = (
- { onBlur }: { onBlur?: ?Function } = { onBlur: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
- if (!isZipValid(e.target.value)) {
- this.setFieldInvalid(
- customTextLabels.invalidZipCode || 'Zip code is invalid',
- 'cardZip'
- );
- }
-
- const { cardZipInputProps } = this.props;
- cardZipInputProps.onBlur && cardZipInputProps.onBlur(e);
- onBlur && onBlur(e);
- };
-
- handleCardZipChange = (
- { onChange }: { onChange?: ?Function } = { onChange: null }
- ) => (e: SyntheticInputEvent<*>) => {
- const { customTextLabels } = this.props;
- const zip = e.target.value;
- const zipLength = zip.length;
-
- this.setFieldValid();
-
- if (zipLength >= 5 && !isZipValid(zip)) {
- this.setFieldInvalid(
- customTextLabels.invalidZipCode || 'Zip code is invalid',
- 'cardZip'
- );
- }
-
- const { cardZipInputProps } = this.props;
- cardZipInputProps.onChange && cardZipInputProps.onChange(e);
- onChange && onChange(e);
- };
-
- handleCardZipKeyPress = (e: any) => {
- const cardType = payment.fns.cardType(this.state.cardNumber);
- const value = e.target.value;
- this.checkIsNumeric(e);
- if (value && !isHighlighted()) {
- const valueLength = value.split(' / ').join('').length;
- if (hasZipReachedMaxLength(cardType, valueLength)) {
- e.preventDefault();
- }
- }
- };
-
- handleKeyDown = (ref: any) => {
- return (e: SyntheticInputEvent<*>) => {
- if (e.keyCode === BACKSPACE_KEY_CODE && !e.target.value) {
- ref.focus();
- }
- };
- };
-
- setFieldInvalid = (errorText: string, inputName?: string) => {
- const { invalidClassName, onError } = this.props;
- // $FlowFixMe
- document.getElementById('field-wrapper').classList.add(invalidClassName);
- this.setState({ errorText });
-
- if (inputName) {
- const { onError } = this.props[`${inputName}InputProps`];
- onError && onError(errorText);
- }
-
- if (onError) {
- onError({ inputName, error: errorText });
- }
- };
-
- setFieldValid = () => {
- const { invalidClassName } = this.props;
- // $FlowFixMe
- document.getElementById('field-wrapper').classList.remove(invalidClassName);
- this.setState({ errorText: null });
- };
-
- render = () => {
- const { cardImage, errorText, showZip } = this.state;
- const {
- cardImageClassName,
- cardImageStyle,
- cardCVCInputProps,
- cardZipInputProps,
- cardExpiryInputProps,
- cardNumberInputProps,
- cardCVCInputRenderer,
- cardExpiryInputRenderer,
- cardNumberInputRenderer,
- cardZipInputRenderer,
- containerClassName,
- containerStyle,
- dangerTextClassName,
- dangerTextStyle,
- enableZipInput,
- fieldClassName,
- fieldStyle,
- inputClassName,
- inputStyle,
- invalidStyle,
- customTextLabels
- } = this.props;
-
- return (
-
-
-
-
- {cardNumberInputRenderer({
- handleCardNumberChange: onChange =>
- this.handleCardNumberChange({ onChange }),
- handleCardNumberBlur: onBlur =>
- this.handleCardNumberBlur({ onBlur }),
- props: {
- id: 'card-number',
- ref: cardNumberField => {
- this.cardNumberField = cardNumberField;
- },
- maxLength: '19',
- autoComplete: 'cc-number',
- className: `credit-card-input ${inputClassName}`,
- placeholder:
- customTextLabels.cardNumberPlaceholder || 'Card number',
- type: 'tel',
- ...cardNumberInputProps,
- onBlur: this.handleCardNumberBlur(),
- onChange: this.handleCardNumberChange(),
- onKeyPress: this.handleCardNumberKeyPress
- }
- })}
-
-
- {cardExpiryInputRenderer({
- handleCardExpiryChange: onChange =>
- this.handleCardExpiryChange({ onChange }),
- handleCardExpiryBlur: onBlur =>
- this.handleCardExpiryBlur({ onBlur }),
- props: {
- id: 'card-expiry',
- ref: cardExpiryField => {
- this.cardExpiryField = cardExpiryField;
- },
- autoComplete: 'cc-exp',
- className: `credit-card-input ${inputClassName}`,
- placeholder: customTextLabels.expiryPlaceholder || 'MM/YY',
- type: 'tel',
- ...cardExpiryInputProps,
- onBlur: this.handleCardExpiryBlur(),
- onChange: this.handleCardExpiryChange(),
- onKeyDown: this.handleKeyDown(this.cardNumberField),
- onKeyPress: this.handleCardExpiryKeyPress
- }
- })}
-
-
- {cardCVCInputRenderer({
- handleCardCVCChange: onChange =>
- this.handleCardCVCChange({ onChange }),
- handleCardCVCBlur: onBlur => this.handleCardCVCBlur({ onBlur }),
- props: {
- id: 'cvc',
- ref: cvcField => {
- this.cvcField = cvcField;
- },
- maxLength: '5',
- autoComplete: 'off',
- className: `credit-card-input ${inputClassName}`,
- placeholder: customTextLabels.cvcPlaceholder || 'CVC',
- type: 'tel',
- ...cardCVCInputProps,
- onBlur: this.handleCardCVCBlur(),
- onChange: this.handleCardCVCChange(),
- onKeyDown: this.handleKeyDown(this.cardExpiryField),
- onKeyPress: this.handleCardCVCKeyPress
- }
- })}
-
-
- {cardZipInputRenderer({
- handleCardZipChange: onChange =>
- this.handleCardZipChange({ onChange }),
- handleCardZipBlur: onBlur => this.handleCardZipBlur({ onBlur }),
- props: {
- id: 'zip',
- ref: zipField => {
- this.zipField = zipField;
- },
- maxLength: '6',
- className: `credit-card-input zip-input ${inputClassName}`,
- pattern: '[0-9]*',
- placeholder: customTextLabels.zipPlaceholder || 'Zip',
- type: 'text',
- ...cardZipInputProps,
- onBlur: this.handleCardZipBlur(),
- onChange: this.handleCardZipChange(),
- onKeyDown: this.handleKeyDown(this.cvcField),
- onKeyPress: this.handleCardZipKeyPress
- }
- })}
-
-
- {errorText && (
-
- {errorText}
-
- )}
-
- );
- };
-}
+import CreditCardInput from './credit-card-input';
+import useCreditCardInput from './hooks/use-credit-card-input';
+export { useCreditCardInput };
export default CreditCardInput;
diff --git a/src/index.stories.js b/src/index.stories.js
index 73b6668..ee5c81a 100644
--- a/src/index.stories.js
+++ b/src/index.stories.js
@@ -32,12 +32,20 @@ storiesOf('CreditCardInput', module)
onChange: e => console.log('number change', e),
onError: err => console.log(`number error: ${err}`)
}}
+ onError={err => console.log('onError', err)}
/>
))
.add('with zip field enabled', () => (
-
+ console.log('zip blur', e),
+ onChange: e => console.log('zip change', e),
+ onError: err => console.log(`zip error: ${err}`)
+ }}
+ />
))
.add('with pre-filled values', () => (
@@ -59,7 +67,7 @@ storiesOf('CreditCardInput', module)
images={{
dinersclub: ``
}}
- CARD_TYPES={{
+ cardTypes={{
dinersclub: 'DINERS_CLUB'
}}
/>
diff --git a/src/reducer/actions.js b/src/reducer/actions.js
new file mode 100644
index 0000000..3e58859
--- /dev/null
+++ b/src/reducer/actions.js
@@ -0,0 +1,9 @@
+export const CHANGE_CARD_IMAGE = Symbol('CHANGE_CARD_IMAGE');
+export const CHANGE_CARD_NUMBER = Symbol('CHANGE_CARD_NUMBER');
+export const CHANGE_CARD_TYPES = Symbol('CHANGE_CARD_TYPES');
+export const CHANGE_CUSTOM_TEXT_LABELS = Symbol('CHANGE_CUSTOM_TEXT_LABELS');
+export const CHANGE_ERROR_TEXT = Symbol('CHANGE_ERROR_TEXT');
+export const CHANGE_IMAGES = Symbol('CHANGE_IMAGES');
+export const CHANGE_INPUT_CLASS_NAME = Symbol('CHANGE_INPUT_CLASS_NAME');
+export const ENABLE_ZIP_INPUT = Symbol('ENABLE_ZIP_INPUT');
+export const SHOW_ZIP = Symbol('SHOW_ZIP');
diff --git a/src/reducer/index.js b/src/reducer/index.js
new file mode 100644
index 0000000..e4d0eca
--- /dev/null
+++ b/src/reducer/index.js
@@ -0,0 +1,100 @@
+import cardTypes from '../utils/card-types';
+import images from '../utils/images';
+import events from '../events';
+import {
+ CHANGE_CARD_IMAGE,
+ CHANGE_CARD_NUMBER,
+ CHANGE_CARD_TYPES,
+ CHANGE_CUSTOM_TEXT_LABELS,
+ CHANGE_ERROR_TEXT,
+ CHANGE_IMAGES,
+ CHANGE_INPUT_CLASS_NAME,
+ ENABLE_ZIP_INPUT,
+ SHOW_ZIP
+} from './actions';
+
+type State = {
+ cardTypes: Object,
+ cardImage: string,
+ cardNumber: ?string,
+ customTextLabels: Object,
+ enableZipInput: boolean,
+ errorText: ?string,
+ events: Object,
+ images: Object,
+ inputClassName: string,
+ showZip: boolean
+};
+
+const inititalState: State = {
+ cardImage: images.placeholder,
+ cardNumber: null,
+ cardTypes,
+ customTextLabels: {},
+ enableZipInput: false,
+ errorText: null,
+ events: events(),
+ images,
+ inputClassName: '',
+ showZip: false
+};
+
+const reducer = (state: State, action) => {
+ switch (action.type) {
+ case CHANGE_CARD_IMAGE:
+ return {
+ ...state,
+ cardImage: action.cardImage
+ };
+ case CHANGE_CARD_NUMBER:
+ return {
+ ...state,
+ cardNumber: action.cardNumber
+ };
+ case CHANGE_CARD_TYPES:
+ return {
+ ...state,
+ cardTypes: {
+ ...state.cardTypes,
+ ...action.cardTypes
+ }
+ };
+ case CHANGE_CUSTOM_TEXT_LABELS:
+ return {
+ ...state,
+ customTextLabels: action.customTextLabels
+ };
+ case CHANGE_ERROR_TEXT:
+ return {
+ ...state,
+ errorText: action.errorText
+ };
+ case CHANGE_IMAGES:
+ return {
+ ...state,
+ images: {
+ ...state.images,
+ ...action.images
+ }
+ };
+ case CHANGE_INPUT_CLASS_NAME:
+ return {
+ ...state,
+ inputClassName: action.inputClassName
+ };
+ case ENABLE_ZIP_INPUT:
+ return {
+ ...state,
+ enableZipInput: action.enableZipInput
+ };
+ case SHOW_ZIP:
+ return {
+ ...state,
+ showZip: !!action.showZip
+ };
+ default:
+ throw new Error();
+ }
+};
+
+export { inititalState, reducer };
diff --git a/src/utils/card-types.js b/src/utils/card-types.js
new file mode 100644
index 0000000..963c525
--- /dev/null
+++ b/src/utils/card-types.js
@@ -0,0 +1,5 @@
+export default {
+ mastercard: 'MASTERCARD',
+ visa: 'VISA',
+ amex: 'AMERICAN_EXPRESS'
+};
diff --git a/src/utils/check-is-numeric.js b/src/utils/check-is-numeric.js
new file mode 100644
index 0000000..fd81a91
--- /dev/null
+++ b/src/utils/check-is-numeric.js
@@ -0,0 +1,9 @@
+// @flow
+
+const checkIsNumeric = (e: any) => {
+ if (!/^\d*$/.test(e.key)) {
+ e.preventDefault();
+ }
+};
+
+export default checkIsNumeric;
diff --git a/src/utils/handle-key-down.js b/src/utils/handle-key-down.js
new file mode 100644
index 0000000..5b94a8e
--- /dev/null
+++ b/src/utils/handle-key-down.js
@@ -0,0 +1,13 @@
+// @flow
+
+const BACKSPACE_KEY_CODE = 8;
+
+const handleKeyDown = (callback: Function) => {
+ return (e: SyntheticInputEvent<*>) => {
+ if (e.keyCode === BACKSPACE_KEY_CODE && !e.target.value) {
+ callback && callback();
+ }
+ };
+};
+
+export default handleKeyDown;
diff --git a/src/utils/is-month-dash-key.js b/src/utils/is-month-dash-key.js
new file mode 100644
index 0000000..811ee47
--- /dev/null
+++ b/src/utils/is-month-dash-key.js
@@ -0,0 +1,7 @@
+// @flow
+
+const isMonthDashKey = ({ key, target: { value } } = {}) => {
+ return !value.match(/[/-]/) && /^[/-]$/.test(key);
+};
+
+export default isMonthDashKey;