<link href="//netdna.bootstrapcdn.com/bootstrap/3.0.0/css/bootstrap.min.css" rel="stylesheet" id="bootstrap-css">
<script src="//netdna.bootstrapcdn.com/bootstrap/3.0.0/js/bootstrap.min.js"></script>
<script src="//code.jquery.com/jquery-1.11.1.min.js"></script>
<!------ Include the above in your HEAD tag ---------->
<!DOCTYPE html><html class=''>
<head><script src='//production-assets.codepen.io/assets/editor/live/console_runner-079c09a0e3b9ff743e39ee2d5637b9216b3545af0de366d4b9aad9dc87e26bfd.js'></script><script src='//production-assets.codepen.io/assets/editor/live/events_runner-73716630c22bbc8cff4bd0f07b135f00a0bdc5d14629260c3ec49e5606f98fdd.js'></script><script src='//production-assets.codepen.io/assets/editor/live/css_live_reload_init-2c0dc5167d60a5af3ee189d570b1835129687ea2a61bee3513dee3a50c115a77.js'></script><meta charset='UTF-8'><meta name="robots" content="noindex"><link rel="shortcut icon" type="image/x-icon" href="//production-assets.codepen.io/assets/favicon/favicon-8ea04875e70c4b0bb41da869e81236e54394d63638a1ef12fa558a4a835f1164.ico" /><link rel="mask-icon" type="" href="//production-assets.codepen.io/assets/favicon/logo-pin-f2d2b6d2c61838f7e76325261b7195c27224080bc099486ddd6dccb469b8e8e6.svg" color="#111" /><link rel="canonical" href="https://codepen.io/garyanikin/pen/meMXGK?limit=all&page=62&q=form" />
<link rel='stylesheet prefetch' href='https://cdnjs.cloudflare.com/ajax/libs/normalize/5.0.0/normalize.min.css'>
<style class="cp-pen-styles">.clearfix:after{
content:'';
display:block;
clear: both;
}
.container {
font-family: Arial, Helvetica, sans-serif;
margin: 50px auto;
width: 310px;
}
.payment-form {
padding: 15px 10px;
background-color: #E6E6E6;
}
.payment-form__button{
width: 100%;
height: 40px;
border-radius: 3px;
border: none;
outline: none;
background-color: #2BAD82;
font-size: 14px;
color: #FFFFFF;
line-height: 19px;
cursor: pointer;
margin: 15px 0 0 0;
}
.payment-form__button:disabled{
opacity: 0.5;
cursor: default;
}
.payment-form__agreement{
color: #989898;
font-size: 11px;
line-height: 13px;
margin: 10px 0;
}
.notification {
position: relative;
padding: 15px 0;
border-bottom: solid 1px #DDDDDD;
}
.notification__icon {
position: absolute;
top: 50%;
margin-top: -17px;
}
.notification__text {
margin-left: 44px;
display: block;
color: #989898;
font-size: 11px;
}
.card-type{
padding: 15px 0 0 0;
}
.card-type__label {
font-size: 12px;
color: #999999;
float: left;
width: auto;
height: 20px;
line-height: 20px;
}
.card-type__icons {
text-align: center;
}
.card-type__icon {
vertical-align: middle;
padding: 0 5px;
}
.card-type__icon--disabled {
filter: grayscale(100%);
-ms-filter: grayscale(100%);
-webkit-filter: grayscale(100%);
opacity: 0.5;
}
.card-input {
border: 1px solid #C6C6C6;
font-family: Arial;
font-size: 16px;
line-height: 22px;
height: 38px;
box-sizing: border-box;
padding: 0 10px;
}
.card-input.ng-invalid-first_char, .card-input.ng-invalid-only_latin, .card-input.ng-invalid-date{
border: 1px solid red;
}
.card-input--full {
width: 100%;
margin: 15px 0 0 0;
}
.card-input--date {
width: 80px;
text-align: center;
margin: 5px 0 0 0;
}
.card-input--cvv {
width: 60px;
text-align: center;
margin: 5px 0 0 0;
}
.card-info{
margin: 15px 0 0 0;
}
.card-info__date{
float: left;
width: 50%;
}
.card-info__text{
font-size: 12px;
color: #6D6D6D;
line-height: 17px;
}</style></head><body>
<div ng-app="app" class="container">
<form ng-controller="PaymentFormCtrl" class="payment-form" name="paymentForm">
<div class="notification">
<img src="" class="notification__icon"/>
<span class="notification__text">
Мы не храним ваши платежные данные, а сразу передаем их по зашифрованному каналу платежной системе
</span>
</div>
<div class="card-type clearfix">
<div class="card-type__label">
Введите данные карты
</div>
<div class="card-type__icons">
<img src="" class="card-type__icon card-type__icon--disabled" ng-class="{'card-type__icon--disabled': !isVisa()}" />
<img src="" class="card-type__icon card-type__icon--disabled" ng-class="{'card-type__icon--disabled': !isMasterCard()}"/>
</div>
</div>
<input type="text" class="card-input card-input--full" placeholder="Номер карты" ng-model="card.number" required card-number/>
<input type="text" class="card-input card-input--full" placeholder="Имя держателя (как на карте)" ng-model="card.holder" card-holder required/>
<div class="card-info clearfix">
<div class="card-info__date">
<div class="card-info__text">
Срок действия
</div>
<input type="text" class="card-input card-input--date" placeholder="ММ / ГГ" ng-model="card.date" card-date required/>
</div>
<div class="card-info__cvv">
<div class="card-info__text">
Код безопасности
</div>
<input type="text" class="card-input card-input--cvv" placeholder="CVV" ng-model="card.cvv" card-cvv required/>
</div>
</div>
<button class="payment-form__button" ng-disabled="paymentForm.$invalid" ng-click="sendForm()">
Купить
</button>
<div class="payment-form__agreement">
Нажимая на кнопку «Купить», вы даете согласие на регулярные платежи по подписке. Вы всегда сможете отменить подписку в настройках сервиса.
</div>
</form>
</div>
<script src='//production-assets.codepen.io/assets/common/stopExecutionOnTimeout-b2a7b3fe212eaa732349046d8416e00a9dec26eb7fd347590fbced3ab38af52e.js'></script><script src='//cdnjs.cloudflare.com/ajax/libs/angular.js/1.3.14/angular.min.js'></script><script src='https://raw.githubusercontent.com/candreoliveira/ngMask/master/dist/ngMask.min.js'></script>
<script >var app = angular.module('app', []);
app.controller('PaymentFormCtrl', function($scope) {
$scope.card = {};
$scope.isVisa = function(){
if($scope.card.number && $scope.card.number.charAt(0) == 4){
return true;
}else{
return false;
}
};
$scope.isMasterCard = function(){
if($scope.card.number && $scope.card.number.charAt(0) == 5){
return true;
}else{
return false;
}
};
$scope.sendForm = function(){
alert('Card number: ' + $scope.card.number + "\n" + 'Card holder: ' + $scope.card.holder + "\n" + 'Expiry date: ' + $scope.card.date + "\n" + 'Card CVV: ' + $scope.card.cvv );
};
});
app.directive('cardNumber', function($browser){
return {
require: 'ngModel',
link: function($scope, $element, $attrs, ngModelCtrl) {
var formatter = function() {
var chunks = $element.val().replace(/[^\d]+/g,'').match(/\d{1,4}/g);
if(chunks){
$element.val(chunks.join(' ').slice(0,19));
}else{
$element.val('');
}
};
ngModelCtrl.$parsers.push(function(viewValue) {
viewValue = viewValue.replace(/[^\d]+/g,'').slice(0,16);
if(viewValue.charAt(0) != 4 && viewValue.charAt(0) != 5 && viewValue){
ngModelCtrl.$setValidity('first_char', false);
}else{
ngModelCtrl.$setValidity('first_char', true);
}
if(viewValue.length < 16){
ngModelCtrl.$setValidity('card_length', false);
}else{
ngModelCtrl.$setValidity('card_length', true);
}
return viewValue;
});
ngModelCtrl.$render = function() {
formatter();
};
$element.bind('change', formatter);
$element.bind('keydown', function(event) {
var key = event.keyCode;
if (key == 91 || (15 < key && key < 19) || (37 <= key && key <= 40)){
return;
}
$browser.defer(formatter);
});
$element.bind('paste cut', function() {
$browser.defer(formatter);
});
}
};
});
app.directive('cardHolder', function($browser){
return {
require: 'ngModel',
link: function($scope, $element, $attrs, ngModelCtrl) {
var capitalize = function(inputValue) {
if(inputValue == undefined) inputValue = '';
if(inputValue.search(/[^a-zA-Z\s]+/) === -1){
ngModelCtrl.$setValidity('only_latin', true);
}else{
ngModelCtrl.$setValidity('only_latin', false);
}
var capitalized = inputValue.toUpperCase();
if(capitalized !== inputValue) {
ngModelCtrl.$setViewValue(capitalized);
ngModelCtrl.$render();
}
return capitalized;
};
ngModelCtrl.$parsers.push(capitalize);
capitalize($scope[$attrs.ngModel]);
}
};
});
app.directive('cardDate', function($browser){
return {
require: 'ngModel',
link: function($scope, $element, $attrs, ngModelCtrl) {
var formatter = function() {
var chunks = $element.val().replace(/[^\d]+/g,'').match(/\d{1,2}/g);
if(chunks){
$element.val(chunks.join(' / ').slice(0,7));
}else{
$element.val('');
}
};
ngModelCtrl.$parsers.push(function(viewValue) {
viewValue = viewValue.replace(/[^\d]+/g,'');
var chunks = viewValue.match(/\d{1,2}/g);
if(chunks[0] > 12 || chunks[0] == '00'){
ngModelCtrl.$setValidity('date', false);
}else{
ngModelCtrl.$setValidity('date', true);
}
if(viewValue.length < 4){
ngModelCtrl.$setValidity('date_length', false);
}else{
ngModelCtrl.$setValidity('date_length', true);
}
return chunks.join(' / ').slice(0,7);
});
ngModelCtrl.$render = function() {
formatter();
};
$element.bind('change', formatter);
$element.bind('keydown', function(event) {
var key = event.keyCode;
if (key == 91 || (15 < key && key < 19) || (37 <= key && key <= 40)){
return;
}
$browser.defer(formatter);
});
$element.bind('paste cut', function() {
$browser.defer(formatter);
});
}
};
});
app.directive('cardCvv', function($browser){
return {
require: 'ngModel',
link: function($scope, $element, $attrs, ngModelCtrl) {
var formatter = function() {
$element.val($element.val().replace(/[^\d]+/g,'').slice(0,3));
};
ngModelCtrl.$parsers.push(function(viewValue) {
viewValue = viewValue.replace(/[^\d]+/g,'');
if(viewValue.length < 3){
ngModelCtrl.$setValidity('cvv_length', false);
}else{
ngModelCtrl.$setValidity('cvv_length', true);
}
return viewValue.slice(0,3);
});
ngModelCtrl.$render = function() {
formatter();
};
$element.bind('change', formatter);
$element.bind('keydown', function(event) {
var key = event.keyCode;
if (key == 91 || (15 < key && key < 19) || (37 <= key && key <= 40)){
return;
}
$browser.defer(formatter);
});
$element.bind('paste cut', function() {
$browser.defer(formatter);
});
}
};
});
//# sourceURL=pen.js
</script>
</body></html>