Form validation is a crucial part of web development, ensuring that user input meets specific criteria before submission. While built-in HTML5 validation and browser-native checks are useful, custom validation often provides a more tailored and user-friendly experience.
Let’s dive into custom form validation using directives, a powerful feature in many modern frameworks. Directives allow us to extend HTML with custom behavior, making them perfect for reusable validation logic.
One common approach is to create a directive that checks input against a specific pattern or rule. For example, let’s say we want to validate that a username contains only alphanumeric characters:
app.directive('alphanumeric', function() {
return {
require: 'ngModel',
link: function(scope, element, attrs, modelCtrl) {
modelCtrl.$validators.alphanumeric = function(modelValue, viewValue) {
var value = modelValue || viewValue;
return /^[a-zA-Z0-9]+$/.test(value);
};
}
};
});
This directive can be applied to any input field:
<input type="text" ng-model="username" alphanumeric>
Now, the form will only be valid if the username contains alphanumeric characters. Pretty neat, right?
But what if we want something more complex, like validating a password strength? We can create a directive that checks for multiple criteria:
app.directive('passwordStrength', function() {
return {
require: 'ngModel',
link: function(scope, element, attrs, modelCtrl) {
modelCtrl.$validators.passwordStrength = function(modelValue, viewValue) {
var value = modelValue || viewValue;
var hasUpperCase = /[A-Z]/.test(value);
var hasLowerCase = /[a-z]/.test(value);
var hasNumbers = /\d/.test(value);
var hasNonAlphas = /\W/.test(value);
return hasUpperCase && hasLowerCase && hasNumbers && hasNonAlphas;
};
}
};
});
This directive ensures that a password contains uppercase and lowercase letters, numbers, and special characters. We can use it like this:
<input type="password" ng-model="password" password-strength>
Custom validation isn’t just about checking input, though. It’s also about providing meaningful feedback to users. Let’s enhance our password strength directive to give more detailed feedback:
app.directive('passwordStrength', function() {
return {
require: 'ngModel',
link: function(scope, element, attrs, modelCtrl) {
modelCtrl.$validators.passwordStrength = function(modelValue, viewValue) {
var value = modelValue || viewValue;
var strength = 0;
if (/[A-Z]/.test(value)) strength++;
if (/[a-z]/.test(value)) strength++;
if (/\d/.test(value)) strength++;
if (/\W/.test(value)) strength++;
scope.passwordStrength = strength;
return strength >= 3;
};
}
};
});
Now, we can use this information to display a strength meter to the user:
<input type="password" ng-model="password" password-strength>
<div ng-switch="passwordStrength">
<div ng-switch-when="0">Very weak</div>
<div ng-switch-when="1">Weak</div>
<div ng-switch-when="2">Medium</div>
<div ng-switch-when="3">Strong</div>
<div ng-switch-when="4">Very strong</div>
</div>
This provides immediate, visual feedback to users as they type their password, encouraging stronger passwords without frustrating them with unexplained rejections.
But what about more complex validations that might require server-side checks? For instance, checking if a username is already taken. We can create a directive that performs an asynchronous check:
app.directive('uniqueUsername', function($q, $http) {
return {
require: 'ngModel',
link: function(scope, element, attrs, modelCtrl) {
modelCtrl.$asyncValidators.uniqueUsername = function(modelValue, viewValue) {
var value = modelValue || viewValue;
return $http.get('/api/check-username?username=' + value).then(
function(response) {
if (!response.data.available) {
return $q.reject('Username already taken');
}
return true;
}
);
};
}
};
});
This directive makes an API call to check if the username is available. The form will remain in a ‘pending’ state until the server responds.
Custom validation isn’t limited to single fields, either. Sometimes, we need to validate relationships between fields. For example, let’s create a directive to ensure a confirmation password matches the original password:
app.directive('passwordMatch', function() {
return {
require: 'ngModel',
scope: {
otherModelValue: '=passwordMatch'
},
link: function(scope, element, attributes, modelCtrl) {
modelCtrl.$validators.passwordMatch = function(modelValue) {
return modelValue === scope.otherModelValue;
};
scope.$watch('otherModelValue', function() {
modelCtrl.$validate();
});
}
};
});
We can use this directive like so:
<input type="password" ng-model="password">
<input type="password" ng-model="confirmPassword" password-match="password">
This ensures that the confirmation password always matches the original password.
Now, let’s talk about reusability. While these directives are already reusable to some extent, we can take it a step further by creating a validation service that can be used across multiple forms:
app.service('validationService', function() {
this.isAlphanumeric = function(value) {
return /^[a-zA-Z0-9]+$/.test(value);
};
this.isStrongPassword = function(value) {
var strength = 0;
if (/[A-Z]/.test(value)) strength++;
if (/[a-z]/.test(value)) strength++;
if (/\d/.test(value)) strength++;
if (/\W/.test(value)) strength++;
return strength >= 3;
};
// Add more validation methods as needed
});
Now we can use this service in our directives:
app.directive('alphanumeric', function(validationService) {
return {
require: 'ngModel',
link: function(scope, element, attrs, modelCtrl) {
modelCtrl.$validators.alphanumeric = validationService.isAlphanumeric;
}
};
});
This approach makes our validation logic more maintainable and easier to test.
Speaking of testing, it’s crucial to write unit tests for our custom validation logic. Here’s an example using Jasmine:
describe('validationService', function() {
var validationService;
beforeEach(module('myApp'));
beforeEach(inject(function(_validationService_) {
validationService = _validationService_;
}));
it('should validate alphanumeric strings', function() {
expect(validationService.isAlphanumeric('abc123')).toBe(true);
expect(validationService.isAlphanumeric('abc-123')).toBe(false);
});
it('should validate strong passwords', function() {
expect(validationService.isStrongPassword('weakpass')).toBe(false);
expect(validationService.isStrongPassword('StrongP@ss1')).toBe(true);
});
});
Custom form validation is a powerful tool in a developer’s arsenal. It allows us to create user-friendly forms that provide immediate, meaningful feedback. By using directives and services, we can create reusable validation logic that’s easy to maintain and test.
Remember, the goal of form validation isn’t just to prevent invalid data - it’s to guide users towards providing the correct information. Clear error messages, visual cues, and immediate feedback all contribute to a better user experience.
As you implement custom validation in your projects, always consider the user’s perspective. Is the validation helpful or frustrating? Does it provide clear guidance on how to correct errors? Are the rules reasonable and necessary?
In my experience, the best form validation is the kind that users hardly notice. It gently guides them towards providing the correct information without getting in their way. It’s a delicate balance, but when done right, it can significantly improve the usability of your web applications.
So go forth and validate! Your users will thank you for it. And who knows? You might even find yourself enjoying the process of crafting the perfect form experience. After all, there’s a certain satisfaction in creating something that’s both functional and user-friendly. Happy coding!