<div class="container">
    <div class="alertSuccess" id="response">
      <p id="response-code" style="padding: 0px 10px"></p>
      <p id="response-message" style="padding: 0px 10px"></p>
    </div>

    <!-- First page -->
      <div class="form-page card-details show">
        <div class="card">
          <div class="flex-x flex-sb">
            <div class="col col-6">
              <h3>Pay with Card</h3>
            </div>
            <div class="logo-container">
              <div class="image">
                <img src="https://upload.wikimedia.org/wikipedia/commons/thumb/2/2a/Mastercard-logo.svg/618px-Mastercard-logo.svg.png" />
              </div>
              <div class="image">
                <img src="" />
              </div>
            </div>
          </div>
    
          <div>
            <div class="form-group">
              <div class="input-label">Card number</div>
              <div class="input" id='cardNumber-container'></div>
            </div>
          </div>
  
          <div class="flex-x flex-sb">
            <div class="col-6 form-group">
              <div class="input-label">Expiry</div>
              <div class="input" id='expirationDate-container'></div>
            </div>
    
            <div class="col-6 form-group">
              <div class="input-label">CVV</div>
              <div class="input" id="cvv-container"></div>
            </div>
          </div>

        </div>

        <button class='btn' id="pay-button">Pay &#8358; <span id='amount'></span></button>   

      </div>

      <!-- Second page -->
      <div class="form-page pin">
        <div class="card">
          <span class="back-control" id="pin-back-button">
            <label>Back</label>
          </span>
  
          <div class="form-control" style="margin-top: 20px">
            <div class="input-label">Please provide your PIN</div>
            <div id="pin-container" class="input"></div>
          </div>
          <div class="button-container">
            <button id="continue-button" class="btn">Continue</button>
          </div>
        </div>   
      </div>

      <!-- Third page -->
      <div class="form-page otp">
        <div class="card">
          <span class="back-control" id="otp-back-button">
            <label>Back</label>
          </span>

          <div class="form-control" style="margin-top: 20px">
            <label>Please input the OTP sent to your mobile number</label>
            <div id="otp-container" class="input"></div>
          </div>
          <div class="button-container">
            <button id="validate-button" class="btn">Validate</button>
          </div>
        </div>
      </div>
      <div class="form-page cardinal">
        <div class="cardinal-container"></div>
      </div>

    </div>

    <script src="https://isw-hosted-fields.k8.isw.la/sdk.js"></script>


*{
    font-family: 'Roboto', sans-serif;
    box-sizing: border-box;
}
body{
    margin: 0
}

.container {
    margin: 0px;
    height: 100vh;
    display: flex;
    justify-content: center;
    align-items: center;
    flex-direction: column;
    background-color: #eee;

}


.flex-x{
    display: flex;
    flex-direction: row;
}

.flex-sb{
    justify-content: space-between;

}

.col-6{
    width: 48%
}

.form-page {
    display: none;
    width: 35%;
}


.card{
    /* background-color: rgb(56, 24, 199); */
    background-color: #fff;
    border-radius: 6px;
    padding: 20px;
    box-shadow: 0 2px 2px rgba(0,0,0,0.2);
}

.btn-container{
    margin-top: 20px
}

.form-group{
    margin-top: 10px;
    margin-bottom: 10px;
}

.input-label{
    margin-right: 10px;
    margin-bottom: 10px;
    width: 100%
}
.input{
    width: 100%;
    /* padding: 10px 0 10px 5px; */
    font-size: 1.2em;
    border: none;
    border-bottom: solid 2px #ccc;
    outline: none;
    height: 45px;
    display: inline-block;
}

input{
    height: 50px !important
}

.btn{
    width: 100%;
    padding-top: 15px;
    padding-bottom: 15px;
    background-color: rgb(29, 9, 119);
    border: none;
    border-radius: 4px;
    color: #fff;
    margin-top: 20px;
    transition: .5s;
    cursor: pointer;
}

.back-control{
    background-color: #ccc;
    padding: 5px 10px;
    border-radius: 3px;
    cursor: pointer;
}

.btn:hover{
    background-color: rgb(21, 6, 85);;
}



.logo-container{
    height: auto;
    display: flex;
    justify-content: center;
    align-items: center;
    flex-direction: row;
    margin-bottom: 20px
}

.logo-container img{
    width: 50px;
    height: auto;
    margin-right: 10px
}

.cardinal-container {
    position: absolute;
    width: 100%;
    height: 100%;
    background: white;
    left: 0;
    top: 0;
    z-index: 10;
}



.form-page.show {
    display: block;
}

.form-page.cardinal {
    min-height: 350px;
}



.showNotification{
    /* height: 50px; */
}

/* .alertSuccess{
    width: 300px;
    overflow: hidden;
    background: #5fdc91;
    margin: 0 auto;
    transition: all 0.4s ease-in-out;
    margin-top: 15px;
    font-size: 20px;
    margin-bottom: 20px;
} */

.alertError{
    color: '#ff0000' !important;
}
let loading = false;

var configuration = {
    fields: {
        cardNumber: {
            selector: '#cardNumber-container',
            placeholder: '0000  0000  0000  0000',
            styles: {
                fontSize: '16px',
                padding: '0px 0px 0px 10px',
                padding: '0px 0px 0px 10px',
            }
        },
        expirationDate: {
            selector: '#expirationDate-container',
            placeholder: 'MM / YY',
            styles: {
                fontSize: '16px',
                padding: '0px 0px 0px 10px',
                padding: '0px 0px 0px 10px',
            }
        },
        cvv: {
            selector: '#cvv-container',
            placeholder: '***',
            styles: {
                fontSize: '16px',
                padding: '0px 0px 0px 10px',
                padding: '0px 0px 0px 10px',
            }
        },
        pin: {
            selector: '#pin-container',
            placeholder: '* * * *',
            styles: {
                fontSize: '16px',
                padding: '0px 0px 0px 10px',
                textAlign: 'center'
            }
        },
        otp: {
            selector: '#otp-container',
            placeholder: '* * * * * *',
            styles: {
                fontSize: '16px',
                padding: '0px 0px 0px 10px',
                textAlign: 'center'
            }
        }
    },
    cardinal: {
        containerSelector: '.cardinal-container',
        activeClass: 'show'
    },
    paymentParameters: {
        amount: 10000,
        currencyCode: "566",
        dateOfPayment: '2019-11-08T00:00:00',
        payableCode: "Default_Payable_MX26070",
        merchantCustomerName: "John Doe",
        merchantCode: 'MX26070',
        transactionReference: "isw_hosted_field_test:" + Date.now(),
    }

}


let instance;



let payButton = document.getElementById('pay-button');
let continueButton = document.getElementById('continue-button');
let validateButton = document.getElementById('validate-button');

let pinBackButton = document.getElementById('pin-back-button');
let otpBackButton = document.getElementById('otp-back-button');

pinBackButton.addEventListener('click', function () {
    setActivePage('card-details');
    instance.clearField('pin');
});

otpBackButton.addEventListener('click', function () {
    setActivePage('pin');
    instance.clearField('otp');
});


function callback(createError, hostedFieldsInstance) {
    if (createError != null) {
        throw createError;
    }

    instance = hostedFieldsInstance;


    instance.on('focus', function (event) {
        let fieldContainer = document.querySelector(event.selector);
        fieldContainer.style.borderBottomColor = '#a0c8e2';
    });

    instance.on('blur', function (event) {
        let fieldContainer = document.querySelector(event.selector);
        fieldContainer.style.borderBottomColor = '#e4e4e4';
    });

    instance.on('validation', function (event) {
    });

    instance.on('cardinal-response', handleCardinalValidateResponse);



    payButton.addEventListener('click', function () {
        instance.getBinConfiguration(handleBinConfigResponse);
    });

    continueButton.addEventListener('click', function () {
        instance.makePayment(handlePayResponse);
    });

    validateButton.addEventListener('click', function () {
        instance.validatePayment(handleValidateResponse);
    });

}

isw.hostedFields.create(configuration, callback);






function handleBinConfigResponse(err, response) {
    loading = true;
    if (err != null && err.validationError === true) {
        let fieldContainer = document.getElementById('cardNumber-container');
        fieldContainer.style.borderBottomColor = '#FF0000';
        return;
    }

    if(err != null && err.networkError === true){
        showNotification(err, null);
        return;
    }

    if (err !== null) {
        showNotification(err, null)
        return;
    }

    if (response.supportsPin) {
        setActivePage('pin');
        return;
    }

    instance.makePayment(handlePayResponse);
}


function handlePayResponse(err, response) {
    loading = false;
    if (err != null && err.validationError === true) {
        showNotification(err, null);
        return;
    }

    if(err != null && err.networkError === true){
        showNotification(err, null);
        return;
    }

    if (err != null) {
        showNotification(err, null)
        return;
    }

    if (response.requiresCentinelAuthorization === true) {
        setActivePage('cardinal');
        return;
    }

    if (response.responseCode === 'T0') {
        setActivePage('otp');
        showNotification(null, response);
        return;
    }
}

function handleValidateResponse(err, response) {

    if (err != null && err.validationError === true) {
        showNotification(err, null);
        return;
    }

    if(err != null && err.networkError === true){
        showNotification(err, null);
        return;
    }

    if (err != null) {
        showNotification(err, null);
        return;
    }
    console.log('rrrr: ', response)

    if(response){
        showNotification(null, response)
        return;
    }
}

let alertDiv = document.getElementById('response')
let responseCodeDiv = document.getElementById('response-code') 
let responseMessageDiv = document.getElementById('response-message') 


function showNotification(error, response){
    if(response){
        let responseCode = response.responseCode
        let responseMessage = response.plainTextSupportMessage
        if(response.plainTextSupportMessage){
            setNotificationText(`${responseCode}`, `${responseMessage}`)
        } else {
            setNotificationText(`${responseCode}`, null)
        }
    } else if(error) {
        alertDiv.classList.add('alertError', true)
        let errorCode = error.responseCode;
        let validationError = error.validationError;
        let networkError = error.networkError;
        if(validationError){
            setNotificationText(null, `There's Validation Error`)
        }
        if(networkError){
            setNotificationText(null, `Network Error`)
        }
        if(errorCode){
            setNotificationText(`${errorCode}`, null)
        }
    }
    if (response.responseCode === '00') {
        setNotificationText(response.responseCode, "Transaction Successful");
        setActivePage('card-details');
        return;
    }
}


function handleCardinalValidateResponse(err, response) {
    
    setActivePage('card-details');

}

function setNotificationText(responseCode, responseMessage){
    responseCodeDiv.innerHTML = responseCode;
    responseMessageDiv.innerHTML = responseMessage
    alertDiv.classList.add('showNotification')
        setTimeout(function(){
            alertDiv.classList.remove('showNotification', 'alertError')
        }, 7000)
}

function setActivePage(pageName) {

    let pages = document.querySelectorAll('.form-page');

    for (let i = 0; i < pages.length; i++) {
        let page = pages[i];
        page.classList.remove('show');
    }

    let activePage = document.querySelector('.form-page.' + pageName);
    activePage.classList.add('show');
}

window.onload = function(){
    //display the amount being processed on page
    document.getElementById('amount').innerHTML = configuration.paymentParameters.amount / 100;
}

External CSS

This Pen doesn't use any external CSS resources.

External JavaScript

This Pen doesn't use any external JavaScript resources.