From 8233308317c778312eaa27fdf22a537b23edd3b3 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Fri, 1 May 2020 22:58:20 +0200 Subject: [PATCH 01/14] add credit-card-type and luhn libs --- package-lock.json | 10 ++++++++++ package.json | 2 ++ 2 files changed, 12 insertions(+) diff --git a/package-lock.json b/package-lock.json index 40bc07f..2d73cba 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4880,6 +4880,11 @@ "sha.js": "^2.4.8" } }, + "credit-card-type": { + "version": "8.3.0", + "resolved": "https://registry.npmjs.org/credit-card-type/-/credit-card-type-8.3.0.tgz", + "integrity": "sha512-czfZUpQ7W9CDxZL4yFLb1kFtM/q2lTOY975hL2aO+DC8+GRNDVSXVCHXhVFZPxiUKmQCZbFP8vIhxx5TBQaThw==" + }, "cross-env": { "version": "7.0.2", "resolved": "https://registry.npmjs.org/cross-env/-/cross-env-7.0.2.tgz", @@ -11664,6 +11669,11 @@ "yallist": "^3.0.2" } }, + "luhn": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/luhn/-/luhn-2.4.1.tgz", + "integrity": "sha512-p1eEWSb2RJAfv+BzrPEUqbUXV1H3ylHEAOk9yDM1L12ojD6OQQGrE29DqKLa0XYluJTIwXIOFmyzVOme3QSyww==" + }, "make-dir": { "version": "2.1.0", "resolved": "https://registry.npmjs.org/make-dir/-/make-dir-2.1.0.tgz", diff --git a/package.json b/package.json index d7498b9..f80e9c7 100644 --- a/package.json +++ b/package.json @@ -41,6 +41,8 @@ "prop-types": "^15.6.2" }, "dependencies": { + "credit-card-type": "^8.3.0", + "luhn": "^2.4.1", "payment": "^2.3.0" }, "devDependencies": { From 6d3536908c72bcb1dfa4a1ff7dfb63f8c8119766 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Fri, 1 May 2020 23:04:41 +0200 Subject: [PATCH 02/14] deprecate payment - use credit-card-type + luhn to achieve the same functionality previously provided by the payment lib - some types had their names updated to match the new lib pattern (american-express and diners-club) --- src/index.js | 130 ++++++++++++++++++++++++--------------------- src/styles.scss | 20 +++---- src/utils.js | 1 + test/index.spec.js | 18 +++---- 4 files changed, 90 insertions(+), 79 deletions(-) create mode 100644 src/utils.js diff --git a/src/index.js b/src/index.js index ba6ff4a..fa62d3d 100644 --- a/src/index.js +++ b/src/index.js @@ -1,51 +1,24 @@ import React from 'react'; import PropTypes from 'prop-types'; -import Payment from 'payment'; +import creditCardType, { types as cardTypes } from 'credit-card-type'; +import luhn from 'luhn'; + +import { sanitizeNumber } from './utils'; class ReactCreditCards extends React.Component { constructor(props) { super(props); - this.setCards(); + // TODO: Add Dankort, Laser, Visa Electron + // Fix Hipercard + this.state = { + validCardTypes: Object.values(cardTypes), + }; } - static propTypes = { - acceptedCards: PropTypes.array, - callback: PropTypes.func, - cvc: PropTypes.oneOfType([ - PropTypes.string, - PropTypes.number, - ]).isRequired, - expiry: PropTypes.oneOfType([ - PropTypes.string, - PropTypes.number, - ]).isRequired, - focused: PropTypes.string, - issuer: PropTypes.string, - locale: PropTypes.shape({ - valid: PropTypes.string, - }), - name: PropTypes.string.isRequired, - number: PropTypes.oneOfType([ - PropTypes.string, - PropTypes.number, - ]).isRequired, - placeholders: PropTypes.shape({ - name: PropTypes.string, - }), - preview: PropTypes.bool, - }; - - static defaultProps = { - acceptedCards: [], - locale: { - valid: 'valid thru', - }, - placeholders: { - name: 'YOUR NAME HERE', - }, - preview: false, - }; + componentDidMount() { + this.setCards(); + } componentDidUpdate(prevProps) { const { acceptedCards, callback, number } = this.props; @@ -53,7 +26,7 @@ class ReactCreditCards extends React.Component { if (prevProps.number !== number) { /* istanbul ignore else */ if (typeof callback === 'function') { - callback(this.options, Payment.fns.validateCardNumber(number)); + callback(this.options, luhn.validate(number)); } } @@ -90,7 +63,7 @@ class ReactCreditCards extends React.Component { nextNumber += '•'; } - if (['amex', 'dinersclub'].includes(this.issuer)) { + if (['american-express', 'diners-club'].includes(this.issuer)) { const format = [0, 4, 10]; const limit = [4, 6, 5]; nextNumber = `${nextNumber.substr(format[0], limit[0])} ${nextNumber.substr(format[1], limit[1])} ${nextNumber.substr(format[2], limit[2])}`; @@ -140,44 +113,43 @@ class ReactCreditCards extends React.Component { } get options() { - const { number } = this.props; - const issuer = Payment.fns.cardType(number) || 'unknown'; + const { issuer, number, preview } = this.props; + const { validCardTypes } = this.state; + let updatedIssuer = issuer || 'unknown'; + + if (number && !preview) { + const validatedIssuer = creditCardType(sanitizeNumber(number)).length ? creditCardType(sanitizeNumber(number))[0].type : 'unknown'; + + if (validCardTypes.includes(validatedIssuer)) { + updatedIssuer = validatedIssuer; + } + } let maxLength = 16; - if (issuer === 'amex') { + if (updatedIssuer === 'american-express') { maxLength = 15; } - else if (issuer === 'dinersclub') { + else if (updatedIssuer === 'diners-club') { maxLength = 14; } - else if (['hipercard', 'mastercard', 'visa'].includes(issuer)) { + else if (['hipercard', 'mastercard', 'visa'].includes(updatedIssuer)) { maxLength = 19; } return { - issuer, + issuer: updatedIssuer, maxLength, }; } setCards() { const { acceptedCards } = this.props; - let newCardArray = []; if (acceptedCards.length) { - Payment.getCardArray() - .forEach(d => { - if (acceptedCards.includes(d.type)) { - newCardArray.push(d); - } - }); - } - else { - newCardArray = newCardArray.concat(Payment.getCardArray()); + const validCardTypes = Object.values(cardTypes).filter(card => acceptedCards.includes(card)); + this.setState({ validCardTypes }); } - - Payment.setCardArray(newCardArray); } render() { @@ -190,7 +162,7 @@ class ReactCreditCards extends React.Component { className={[ 'rccs__card', `rccs__card--${this.issuer}`, - focused === 'cvc' && this.issuer !== 'amex' ? 'rccs__card--flipped' : '', + focused === 'cvc' && this.issuer !== 'american-express' ? 'rccs__card--flipped' : '', ].join(' ').trim()} >
@@ -255,4 +227,42 @@ class ReactCreditCards extends React.Component { } } +ReactCreditCards.propTypes = { + acceptedCards: PropTypes.array, + callback: PropTypes.func, + cvc: PropTypes.oneOfType([ + PropTypes.string, + PropTypes.number, + ]).isRequired, + expiry: PropTypes.oneOfType([ + PropTypes.string, + PropTypes.number, + ]).isRequired, + focused: PropTypes.string, + issuer: PropTypes.string, + locale: PropTypes.shape({ + valid: PropTypes.string, + }), + name: PropTypes.string.isRequired, + number: PropTypes.oneOfType([ + PropTypes.string, + PropTypes.number, + ]).isRequired, + placeholders: PropTypes.shape({ + name: PropTypes.string, + }), + preview: PropTypes.bool, +}; + +ReactCreditCards.defaultProps = { + acceptedCards: [], + locale: { + valid: 'valid thru', + }, + placeholders: { + name: 'YOUR NAME HERE', + }, + preview: false, +}; + export default ReactCreditCards; diff --git a/src/styles.scss b/src/styles.scss index 630b199..776d46d 100644 --- a/src/styles.scss +++ b/src/styles.scss @@ -26,9 +26,9 @@ $rccs-background-transition: all 0.5s ease-out !default; $rccs-animate-background: true; /** ISSUERS **/ -$rccs-amex-background: linear-gradient(25deg, #308c67, #a3f2cf) !default; +$rccs-american-express-background: linear-gradient(25deg, #308c67, #a3f2cf) !default; $rccs-dankort-background: linear-gradient(25deg, #ccc, #999) !default; -$rccs-dinersclub-background: linear-gradient(25deg, #fff, #eee) !default; +$rccs-diners-club-background: linear-gradient(25deg, #fff, #eee) !default; $rccs-discover-background: linear-gradient(25deg, #fff, #eee) !default; $rccs-mastercard-background: linear-gradient(25deg, #fbfbfb, #e8e9e5) !default; $rccs-visa-background: linear-gradient(25deg, #0f509e, #1399cd) !default; @@ -38,11 +38,11 @@ $rccs-hipercard-background: linear-gradient(25deg, #8b181b, #de1f27) !default; /** Images **/ $rccs-chip-image: '' !default; -$rccs-amex-logo: '' !default; +$rccs-american-express-logo: '' !default; $rccs-dankort-logo: '' !default; -$rccs-dinersclub-logo: '' !default; +$rccs-diners-club-logo: '' !default; $rccs-discover-logo: '' !default; @@ -159,13 +159,13 @@ $rccs-visaelectron-logo: ' } } - &--amex { + &--american-express { .rccs__card__background { - background: $rccs-amex-background; + background: $rccs-american-express-background; } .rccs__issuer { - background-image: url($rccs-amex-logo); + background-image: url($rccs-american-express-logo); } .rccs__cvc__front { @@ -184,17 +184,17 @@ $rccs-visaelectron-logo: ' } } - &--dinersclub { + &--diners-club { > div { color: $rccs-dark-text-color; } .rccs__card__background { - background: $rccs-dinersclub-background; + background: $rccs-diners-club-background; } .rccs__issuer { - background-image: url($rccs-dinersclub-logo); + background-image: url($rccs-diners-club-logo); } } diff --git a/src/utils.js b/src/utils.js new file mode 100644 index 0000000..7065522 --- /dev/null +++ b/src/utils.js @@ -0,0 +1 @@ +export const sanitizeNumber = (number) => number.toString().trim().replace(' ', ''); diff --git a/test/index.spec.js b/test/index.spec.js index de34a2b..c980a6e 100644 --- a/test/index.spec.js +++ b/test/index.spec.js @@ -78,15 +78,15 @@ describe('ReactCreditCards', () => { focused: 'number', }); - expect(wrapper.find('.rccs__card').hasClass('rccs__card--amex')).toBe(true); + expect(wrapper.find('.rccs__card').hasClass('rccs__card--american-express')).toBe(true); expect(wrapper.find('.rccs__number').text()).toBe('3782 822463 10005'); expect(wrapper.find('.rccs__number').hasClass('rccs--focused')).toBe(true); - expect(mockCallback.mock.calls[0][0]).toEqual({ maxLength: 15, issuer: 'amex' }); + expect(mockCallback.mock.calls[0][0]).toEqual({ maxLength: 15, issuer: 'american-express' }); expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it('should handle new number props (Dankort)', () => { + it.skip('should handle new number props (Dankort)', () => { wrapper.setProps({ number: '5019717010103742', focused: 'number', @@ -106,11 +106,11 @@ describe('ReactCreditCards', () => { focused: 'number', }); - expect(wrapper.find('.rccs__card').hasClass('rccs__card--dinersclub')).toBe(true); + expect(wrapper.find('.rccs__card').hasClass('rccs__card--diners-club')).toBe(true); expect(wrapper.find('.rccs__number').text()).toBe('3056 930902 5904'); expect(wrapper.find('.rccs__number').hasClass('rccs--focused')).toBe(true); - expect(mockCallback.mock.calls[0][0]).toEqual({ maxLength: 14, issuer: 'dinersclub' }); + expect(mockCallback.mock.calls[0][0]).toEqual({ maxLength: 14, issuer: 'diners-club' }); expect(mockCallback.mock.calls[0][1]).toEqual(true); }); @@ -142,7 +142,7 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it('should handle new number props (Hipercard)', () => { + it.skip('should handle new number props (Hipercard)', () => { wrapper.setProps({ number: '3841005899088180330', focused: 'number', @@ -170,7 +170,7 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it('should handle new number props (Laser)', () => { + it.skip('should handle new number props (Laser)', () => { wrapper.setProps({ number: '6709359636227382', focused: 'number', @@ -254,7 +254,7 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it('should handle new number props (VisaElectron)', () => { + it.skip('should handle new number props (VisaElectron)', () => { wrapper.setProps({ number: '4508269706217171', focused: 'number', @@ -370,7 +370,7 @@ describe('ReactCreditCards', () => { wrapper.setProps({ number: '**** **** **** 7056', preview: true, - issuer: 'Hipercard', + issuer: 'hipercard', }); expect(wrapper.find('.rccs__number').text()).toBe('**** **** **** 7056'); From b9b091f80e8160437e14c39e5981c9e6d0bd48bd Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sat, 2 May 2020 10:48:42 +0200 Subject: [PATCH 03/14] add support for dankort, laser, and visa electron --- src/cardTypes.js | 35 +++++++++++++++++++++++++++++++++++ src/index.js | 37 ++++++++++++++++++++++++++++++++----- src/styles.scss | 8 ++++---- test/index.spec.js | 12 ++++++------ 4 files changed, 77 insertions(+), 15 deletions(-) create mode 100644 src/cardTypes.js diff --git a/src/cardTypes.js b/src/cardTypes.js new file mode 100644 index 0000000..45d7043 --- /dev/null +++ b/src/cardTypes.js @@ -0,0 +1,35 @@ +export const dankort = { + niceType: 'Dankort', + type: 'dankort', + patterns: [5019], + gaps: [4, 8, 12], + lengths: [16], + code: { + name: 'CVC', + size: 3, + }, +}; + +export const laser = { + niceType: 'Laser', + type: 'laser', + patterns: [6706, 6771, 6709], + gaps: [4, 8, 12], + lengths: [16, 19], + code: { + name: 'CVV', + size: 3, + }, +}; + +export const visaElectron = { + niceType: 'Visa Electron', + type: 'visa-electron', + patterns: [4026, 417500, 4405, 4508, 4844, 49137], + gaps: [4, 8, 12], + lengths: [16], + code: { + name: 'CVV', + size: 3, + }, +}; diff --git a/src/index.js b/src/index.js index fa62d3d..bfcfe37 100644 --- a/src/index.js +++ b/src/index.js @@ -3,16 +3,44 @@ import PropTypes from 'prop-types'; import creditCardType, { types as cardTypes } from 'credit-card-type'; import luhn from 'luhn'; +import { dankort, laser, visaElectron } from './cardTypes'; import { sanitizeNumber } from './utils'; class ReactCreditCards extends React.Component { constructor(props) { super(props); - // TODO: Add Dankort, Laser, Visa Electron - // Fix Hipercard + creditCardType.updateCard(cardTypes.MAESTRO, { + patterns: [ + 493698, + [5000, 5018], + [502000, 506698], + [506779, 508999], + [56, 59], + 63, + 67, + 6, + ], + }); + creditCardType.updateCard(cardTypes.HIPERCARD, { + patterns: [ + 384100, + 384140, + 384160, + 606282, + 637095, + 637568, + ], + }); + creditCardType.addCard(dankort); + creditCardType.addCard(laser); + creditCardType.addCard(visaElectron); + + const initialValidCards = Object.values(cardTypes); + const extendedValidCards = [...initialValidCards, 'dankort', 'laser', 'visa-electron']; + this.state = { - validCardTypes: Object.values(cardTypes), + validCardTypes: extendedValidCards, }; } @@ -147,8 +175,7 @@ class ReactCreditCards extends React.Component { const { acceptedCards } = this.props; if (acceptedCards.length) { - const validCardTypes = Object.values(cardTypes).filter(card => acceptedCards.includes(card)); - this.setState({ validCardTypes }); + this.setState((prevState) => ({ validCardTypes: prevState.validCardTypes.filter(card => acceptedCards.includes(card)) })); } } diff --git a/src/styles.scss b/src/styles.scss index 776d46d..6e4b102 100644 --- a/src/styles.scss +++ b/src/styles.scss @@ -62,7 +62,7 @@ $rccs-unionpay-logo: ' $rccs-visa-logo: '' !default; -$rccs-visaelectron-logo: '' !default; +$rccs-visa-electron-logo: '' !default; .rccs { margin: 0 auto; @@ -274,7 +274,7 @@ $rccs-visaelectron-logo: ' } &--visa, - &--visaelectron { + &--visa-electron { .rccs__card__background { background: $rccs-visa-background; } @@ -290,9 +290,9 @@ $rccs-visaelectron-logo: ' } } - &--visaelectron { + &--visa-electron { .rccs__issuer { - background-image: url($rccs-visaelectron-logo); + background-image: url($rccs-visa-electron-logo); } } } diff --git a/test/index.spec.js b/test/index.spec.js index c980a6e..0078597 100644 --- a/test/index.spec.js +++ b/test/index.spec.js @@ -86,7 +86,7 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it.skip('should handle new number props (Dankort)', () => { + it('should handle new number props (Dankort)', () => { wrapper.setProps({ number: '5019717010103742', focused: 'number', @@ -142,7 +142,7 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it.skip('should handle new number props (Hipercard)', () => { + it('should handle new number props (Hipercard)', () => { wrapper.setProps({ number: '3841005899088180330', focused: 'number', @@ -170,7 +170,7 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it.skip('should handle new number props (Laser)', () => { + it('should handle new number props (Laser)', () => { wrapper.setProps({ number: '6709359636227382', focused: 'number', @@ -254,17 +254,17 @@ describe('ReactCreditCards', () => { expect(mockCallback.mock.calls[0][1]).toEqual(true); }); - it.skip('should handle new number props (VisaElectron)', () => { + it('should handle new number props (Visa Electron)', () => { wrapper.setProps({ number: '4508269706217171', focused: 'number', }); - expect(wrapper.find('.rccs__card').hasClass('rccs__card--visaelectron')).toBe(true); + expect(wrapper.find('.rccs__card').hasClass('rccs__card--visa-electron')).toBe(true); expect(wrapper.find('.rccs__number').text()).toBe('4508 2697 0621 7171'); expect(wrapper.find('.rccs__number').hasClass('rccs--focused')).toBe(true); - expect(mockCallback.mock.calls[0][0]).toEqual({ maxLength: 16, issuer: 'visaelectron' }); + expect(mockCallback.mock.calls[0][0]).toEqual({ maxLength: 16, issuer: 'visa-electron' }); expect(mockCallback.mock.calls[0][1]).toEqual(true); }); From bd94b0b2bfad4f0f243386e9065a7ec87d52e03a Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sat, 2 May 2020 11:12:19 +0200 Subject: [PATCH 04/14] remove unnecessary variables in the constructor --- src/index.js | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/src/index.js b/src/index.js index bfcfe37..0c2413a 100644 --- a/src/index.js +++ b/src/index.js @@ -36,11 +36,8 @@ class ReactCreditCards extends React.Component { creditCardType.addCard(laser); creditCardType.addCard(visaElectron); - const initialValidCards = Object.values(cardTypes); - const extendedValidCards = [...initialValidCards, 'dankort', 'laser', 'visa-electron']; - this.state = { - validCardTypes: extendedValidCards, + validCardTypes: Object.values(cardTypes).concat(['dankort', 'laser', 'visa-electron']), }; } From 21065998d8a2b389258998072b8ac950ea1ff84c Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sat, 2 May 2020 11:37:02 +0200 Subject: [PATCH 05/14] update readme - add new card type information - update local development section --- README.md | 23 +++++++++++++++++++++-- 1 file changed, 21 insertions(+), 2 deletions(-) diff --git a/README.md b/README.md index ab199a0..0f0fb9e 100644 --- a/README.md +++ b/README.md @@ -71,7 +71,8 @@ Or you can import the CSS: ### Features -- We support all credit card issuers available in [Payment](https://github.com/jessepollak/payment) plus Hipercard (a brazilian credit card). +- We support all credit card issuers available in [credit-card-type](https://github.com/braintree/credit-card-type) plus + Dankort, Laser, and Visa Electron. ## Props @@ -136,12 +137,30 @@ Or you can import the CSS: Here's how you can get started developing locally: +1. Clone this repo and create a symlink: + $ git clone https://github.com/amarofashion/react-credit-cards.git $ cd react-credit-cards $ npm install + $ npm link + +2. Download the demo source from [codesandbox](https://codesandbox.io/s/ovvwzkzry9) and install the dependencies: + + $ cd react-credit-cards-demo + $ npm link react-credit-cards + $ npm install + +3. On the `react-credit-cards` directory, start the watcher: + + $ npm run watch + +4. On the `react-credit-cards-demo` directory, start the demo app: + $ npm start + +Done! Your local changes should be automatically reflected on the demo. -Now, if you go to `http://localhost:3000` in your browser, you should see the demo page. +Check [npm-link](https://docs.npmjs.com/cli/link.html) for detailed instructions. ## Contributing From dc34c1bf1298a6f620470ac6ae75a97f20cfb0e0 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sat, 2 May 2020 11:37:50 +0200 Subject: [PATCH 06/14] update my email --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index f80e9c7..b07b804 100644 --- a/package.json +++ b/package.json @@ -6,7 +6,7 @@ "contributors": [ { "name": "Cassio Cardoso", - "email": "cassio.cardoso@amaro.com" + "email": "caugusto.cardoso@gmail.com" }, { "name": "Gil Barbara", From 141c614ac2e50c8fab1a86a19b1fd56cbec22e6b Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sat, 2 May 2020 13:44:24 +0200 Subject: [PATCH 07/14] move configuration to its own file --- src/config.js | 38 ++++++++++++++++++++++++++++++++++++++ src/index.js | 32 ++++---------------------------- 2 files changed, 42 insertions(+), 28 deletions(-) create mode 100644 src/config.js diff --git a/src/config.js b/src/config.js new file mode 100644 index 0000000..68e37d8 --- /dev/null +++ b/src/config.js @@ -0,0 +1,38 @@ +import creditCardType, { types as cardTypes } from 'credit-card-type'; + +import { dankort, laser, visaElectron } from './cardTypes'; + +/** + * Configure credit cards and return an array of valid types + */ +export const configure = () => { + creditCardType.updateCard(cardTypes.MAESTRO, { + patterns: [ + 493698, + [5000, 5018], + [502000, 506698], + [506779, 508999], + [56, 59], + 63, + 67, + 6, + ], + }); + + creditCardType.updateCard(cardTypes.HIPERCARD, { + patterns: [ + 384100, + 384140, + 384160, + 606282, + 637095, + 637568, + ], + }); + + creditCardType.addCard(dankort); + creditCardType.addCard(laser); + creditCardType.addCard(visaElectron); + + return Object.values(cardTypes).concat(['dankort', 'laser', 'visa-electron']); +}; diff --git a/src/index.js b/src/index.js index 0c2413a..5b7306e 100644 --- a/src/index.js +++ b/src/index.js @@ -1,43 +1,19 @@ import React from 'react'; import PropTypes from 'prop-types'; -import creditCardType, { types as cardTypes } from 'credit-card-type'; +import creditCardType from 'credit-card-type'; import luhn from 'luhn'; -import { dankort, laser, visaElectron } from './cardTypes'; +import { configure } from './config'; import { sanitizeNumber } from './utils'; class ReactCreditCards extends React.Component { constructor(props) { super(props); - creditCardType.updateCard(cardTypes.MAESTRO, { - patterns: [ - 493698, - [5000, 5018], - [502000, 506698], - [506779, 508999], - [56, 59], - 63, - 67, - 6, - ], - }); - creditCardType.updateCard(cardTypes.HIPERCARD, { - patterns: [ - 384100, - 384140, - 384160, - 606282, - 637095, - 637568, - ], - }); - creditCardType.addCard(dankort); - creditCardType.addCard(laser); - creditCardType.addCard(visaElectron); + const validCardTypes = configure(); this.state = { - validCardTypes: Object.values(cardTypes).concat(['dankort', 'laser', 'visa-electron']), + validCardTypes, }; } From 9f50659efd2b07d0f5bfde6d77b21d367583104d Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 10:21:46 +0200 Subject: [PATCH 08/14] abstract card helpers logic --- src/config.js | 38 ----------------- src/index.js | 9 ++-- src/utils.js | 1 - src/utils/cardHelpers.js | 79 ++++++++++++++++++++++++++++++++++++ src/{ => utils}/cardTypes.js | 4 ++ 5 files changed, 86 insertions(+), 45 deletions(-) delete mode 100644 src/config.js delete mode 100644 src/utils.js create mode 100644 src/utils/cardHelpers.js rename src/{ => utils}/cardTypes.js (86%) diff --git a/src/config.js b/src/config.js deleted file mode 100644 index 68e37d8..0000000 --- a/src/config.js +++ /dev/null @@ -1,38 +0,0 @@ -import creditCardType, { types as cardTypes } from 'credit-card-type'; - -import { dankort, laser, visaElectron } from './cardTypes'; - -/** - * Configure credit cards and return an array of valid types - */ -export const configure = () => { - creditCardType.updateCard(cardTypes.MAESTRO, { - patterns: [ - 493698, - [5000, 5018], - [502000, 506698], - [506779, 508999], - [56, 59], - 63, - 67, - 6, - ], - }); - - creditCardType.updateCard(cardTypes.HIPERCARD, { - patterns: [ - 384100, - 384140, - 384160, - 606282, - 637095, - 637568, - ], - }); - - creditCardType.addCard(dankort); - creditCardType.addCard(laser); - creditCardType.addCard(visaElectron); - - return Object.values(cardTypes).concat(['dankort', 'laser', 'visa-electron']); -}; diff --git a/src/index.js b/src/index.js index 5b7306e..20439ae 100644 --- a/src/index.js +++ b/src/index.js @@ -1,10 +1,7 @@ import React from 'react'; import PropTypes from 'prop-types'; -import creditCardType from 'credit-card-type'; -import luhn from 'luhn'; -import { configure } from './config'; -import { sanitizeNumber } from './utils'; +import { cardTypesMap, configure, getCardType, validateLuhn } from './utils/cardHelpers'; class ReactCreditCards extends React.Component { constructor(props) { @@ -27,7 +24,7 @@ class ReactCreditCards extends React.Component { if (prevProps.number !== number) { /* istanbul ignore else */ if (typeof callback === 'function') { - callback(this.options, luhn.validate(number)); + callback(this.options, validateLuhn(number)); } } @@ -119,7 +116,7 @@ class ReactCreditCards extends React.Component { let updatedIssuer = issuer || 'unknown'; if (number && !preview) { - const validatedIssuer = creditCardType(sanitizeNumber(number)).length ? creditCardType(sanitizeNumber(number))[0].type : 'unknown'; + const validatedIssuer = getCardType(number); if (validCardTypes.includes(validatedIssuer)) { updatedIssuer = validatedIssuer; diff --git a/src/utils.js b/src/utils.js deleted file mode 100644 index 7065522..0000000 --- a/src/utils.js +++ /dev/null @@ -1 +0,0 @@ -export const sanitizeNumber = (number) => number.toString().trim().replace(' ', ''); diff --git a/src/utils/cardHelpers.js b/src/utils/cardHelpers.js new file mode 100644 index 0000000..3cb5f5a --- /dev/null +++ b/src/utils/cardHelpers.js @@ -0,0 +1,79 @@ +import creditCardType, { types as cardTypes } from 'credit-card-type'; +import luhn from 'luhn'; + +import { dankort, laser, visaElectron } from './cardTypes'; + +/** + * Check if a credit card number is valid using the Luhn algorithm + * @returns {boolean} + */ +export const validateLuhn = luhn.validate; + +/** + * Given a credit card number in the format (XXXX XXXX XXXX...) return it as a string without any spaces + * @param {*} number + * @returns {string} number + */ +export const sanitizeNumber = (number) => number.toString().trim().replace(' ', ''); + +/** + * Return the issuer of a given credit card number or `unknown` if the issuer can't be identified + * @param {string|number} cardNumber + * @returns {string} cardType + */ +export const getCardType = (cardNumber) => { + const potentialCardTypes = creditCardType(sanitizeNumber(cardNumber)); + + if (potentialCardTypes.length === 1) { + const firstResult = potentialCardTypes.shift(); + + return firstResult.type; + } + + return 'unknown'; +}; + +/** + * Configure credit cards and return an array of valid types + * @returns {string[]} validCardTypes + */ +export const configure = () => { + creditCardType.updateCard(cardTypes.MAESTRO, { + patterns: [ + 493698, + [5000, 5018], + [502000, 506698], + [506779, 508999], + [56, 59], + 63, + 67, + 6, + ], + }); + + creditCardType.updateCard(cardTypes.HIPERCARD, { + patterns: [ + 384100, + 384140, + 384160, + 606282, + 637095, + 637568, + ], + }); + + creditCardType.addCard(dankort); + creditCardType.addCard(laser); + creditCardType.addCard(visaElectron); + + return Object.values(cardTypes).concat(['dankort', 'laser', 'visa-electron']); +}; + +/** + * Provides a map of patterns to match for some card types + */ +export const cardTypesMap = { + amex: ['amex', 'americanexpress', 'american-express'], + dinersclub: ['diners', 'dinersclub', 'diners-club'], + visaelectron: ['visaelectron', 'visa-electron'], +}; diff --git a/src/cardTypes.js b/src/utils/cardTypes.js similarity index 86% rename from src/cardTypes.js rename to src/utils/cardTypes.js index 45d7043..01d3fd2 100644 --- a/src/cardTypes.js +++ b/src/utils/cardTypes.js @@ -1,3 +1,7 @@ +/** + * Provides configuration for card types not supported by `credit-card-types` + */ + export const dankort = { niceType: 'Dankort', type: 'dankort', From 26f97705f15c1513c07320b83a02bb7aaa0ca4e1 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 10:22:26 +0200 Subject: [PATCH 09/14] move setCards call back to the constructor --- src/index.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/src/index.js b/src/index.js index 20439ae..6e51a61 100644 --- a/src/index.js +++ b/src/index.js @@ -12,9 +12,7 @@ class ReactCreditCards extends React.Component { this.state = { validCardTypes, }; - } - componentDidMount() { this.setCards(); } From 8d436b7da0c8c456fa1000baac5a3442e0805d27 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 10:23:18 +0200 Subject: [PATCH 10/14] use cardTypesMap to check for amex and diners --- src/index.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/index.js b/src/index.js index 6e51a61..802e5bb 100644 --- a/src/index.js +++ b/src/index.js @@ -59,7 +59,7 @@ class ReactCreditCards extends React.Component { nextNumber += '•'; } - if (['american-express', 'diners-club'].includes(this.issuer)) { + if (cardTypesMap.amex.includes(this.issuer) || cardTypesMap.dinersclub.includes(this.issuer)) { const format = [0, 4, 10]; const limit = [4, 6, 5]; nextNumber = `${nextNumber.substr(format[0], limit[0])} ${nextNumber.substr(format[1], limit[1])} ${nextNumber.substr(format[2], limit[2])}`; @@ -123,10 +123,10 @@ class ReactCreditCards extends React.Component { let maxLength = 16; - if (updatedIssuer === 'american-express') { + if (cardTypesMap.amex.includes(updatedIssuer)) { maxLength = 15; } - else if (updatedIssuer === 'diners-club') { + else if (cardTypesMap.dinersclub.includes(updatedIssuer)) { maxLength = 14; } else if (['hipercard', 'mastercard', 'visa'].includes(updatedIssuer)) { From 0bdc95c77b885ac8a4446f619a3a790942092c67 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 10:29:11 +0200 Subject: [PATCH 11/14] remove preview check on the options getter --- src/index.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/src/index.js b/src/index.js index 802e5bb..08a914a 100644 --- a/src/index.js +++ b/src/index.js @@ -109,11 +109,11 @@ class ReactCreditCards extends React.Component { } get options() { - const { issuer, number, preview } = this.props; + const { number } = this.props; const { validCardTypes } = this.state; - let updatedIssuer = issuer || 'unknown'; + let updatedIssuer = 'unknown'; - if (number && !preview) { + if (number) { const validatedIssuer = getCardType(number); if (validCardTypes.includes(validatedIssuer)) { From 697141c4e05b840e07a415ef2e76cb03914fcf3f Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 10:38:15 +0200 Subject: [PATCH 12/14] update local development instructions --- README.md | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index 0f0fb9e..f4340b2 100644 --- a/README.md +++ b/README.md @@ -137,28 +137,30 @@ Or you can import the CSS: Here's how you can get started developing locally: -1. Clone this repo and create a symlink: +1. Clone this repo and link it to your global `node_modules`: $ git clone https://github.com/amarofashion/react-credit-cards.git $ cd react-credit-cards $ npm install $ npm link -2. Download the demo source from [codesandbox](https://codesandbox.io/s/ovvwzkzry9) and install the dependencies: +2. Download the demo source from [codesandbox](https://codesandbox.io/s/ovvwzkzry9). +3. Unzip it to the desired directory. +4. Install the dependencies $ cd react-credit-cards-demo - $ npm link react-credit-cards $ npm install + $ npm link react-credit-cards -3. On the `react-credit-cards` directory, start the watcher: +5. On the `react-credit-cards` directory, start the watcher: $ npm run watch -4. On the `react-credit-cards-demo` directory, start the demo app: +6. On the `react-credit-cards-demo` directory, start the demo app: $ npm start -Done! Your local changes should be automatically reflected on the demo. +7. 🎉 Done! The demo app will be running on: `http://localhost:3000/`. Your local changes should be automatically reflected there. Check [npm-link](https://docs.npmjs.com/cli/link.html) for detailed instructions. From c73965138af8a599b4f1ec7e551c698dec6728c3 Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 11:27:18 +0200 Subject: [PATCH 13/14] refactor validCardTypes - use class getter and setter to store the validCardTypes value instead of using the component state - refactor config function name to be more explicit about its implementation --- src/index.js | 34 ++++++++++++++++------------------ src/utils/cardHelpers.js | 4 ++-- 2 files changed, 18 insertions(+), 20 deletions(-) diff --git a/src/index.js b/src/index.js index 08a914a..8458bb6 100644 --- a/src/index.js +++ b/src/index.js @@ -1,21 +1,9 @@ import React from 'react'; import PropTypes from 'prop-types'; -import { cardTypesMap, configure, getCardType, validateLuhn } from './utils/cardHelpers'; +import { cardTypesMap, getCardType, setInitialValidCardTypes, validateLuhn } from './utils/cardHelpers'; class ReactCreditCards extends React.Component { - constructor(props) { - super(props); - - const validCardTypes = configure(); - - this.state = { - validCardTypes, - }; - - this.setCards(); - } - componentDidUpdate(prevProps) { const { acceptedCards, callback, number } = this.props; @@ -27,7 +15,7 @@ class ReactCreditCards extends React.Component { } if (prevProps.acceptedCards.toString() !== acceptedCards.toString()) { - this.setCards(); + this.validCardTypes = acceptedCards; } } @@ -110,13 +98,12 @@ class ReactCreditCards extends React.Component { get options() { const { number } = this.props; - const { validCardTypes } = this.state; let updatedIssuer = 'unknown'; if (number) { const validatedIssuer = getCardType(number); - if (validCardTypes.includes(validatedIssuer)) { + if (this.validCardTypes.includes(validatedIssuer)) { updatedIssuer = validatedIssuer; } } @@ -139,12 +126,23 @@ class ReactCreditCards extends React.Component { }; } - setCards() { + get validCardTypes() { const { acceptedCards } = this.props; + const initialValidCardTypes = setInitialValidCardTypes(); if (acceptedCards.length) { - this.setState((prevState) => ({ validCardTypes: prevState.validCardTypes.filter(card => acceptedCards.includes(card)) })); + return initialValidCardTypes.filter(card => acceptedCards.includes(card)); } + + return initialValidCardTypes; + } + + set validCardTypes(acceptedCards) { + if (acceptedCards.length) { + return this.validCardTypes.filter(card => acceptedCards.includes(card)); + } + + return this.validCardTypes; } render() { diff --git a/src/utils/cardHelpers.js b/src/utils/cardHelpers.js index 3cb5f5a..8b1f51c 100644 --- a/src/utils/cardHelpers.js +++ b/src/utils/cardHelpers.js @@ -34,10 +34,10 @@ export const getCardType = (cardNumber) => { }; /** - * Configure credit cards and return an array of valid types + * Configure the credit card types supported and return an array of valid types * @returns {string[]} validCardTypes */ -export const configure = () => { +export const setInitialValidCardTypes = () => { creditCardType.updateCard(cardTypes.MAESTRO, { patterns: [ 493698, From 2e18b7f06b36ec1cca389b241d0447c4931633ca Mon Sep 17 00:00:00 2001 From: Cassio Cardoso Date: Sun, 3 May 2020 11:32:45 +0200 Subject: [PATCH 14/14] fix setter function returning value - fix issue reported by DeepScan --- src/index.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/index.js b/src/index.js index 8458bb6..2a6911c 100644 --- a/src/index.js +++ b/src/index.js @@ -15,7 +15,7 @@ class ReactCreditCards extends React.Component { } if (prevProps.acceptedCards.toString() !== acceptedCards.toString()) { - this.validCardTypes = acceptedCards; + this.updateValidCardTypes(acceptedCards); } } @@ -137,7 +137,7 @@ class ReactCreditCards extends React.Component { return initialValidCardTypes; } - set validCardTypes(acceptedCards) { + updateValidCardTypes(acceptedCards) { if (acceptedCards.length) { return this.validCardTypes.filter(card => acceptedCards.includes(card)); }