From c7733e905f1e2661c6af56cee5c69d76af725a48 Mon Sep 17 00:00:00 2001 From: sukhrajghuman Date: Wed, 1 May 2019 10:59:20 +1000 Subject: [PATCH] add form comp, update form template --- assets/js/helper.js | 84 +++++++++++++ assets/js/validate.js | 251 ++++++++++++++++++++++++++++++++++++++ package-lock.json | 71 +++++------ package.json | 1 + templates/form/index.html | 87 ++++++------- 5 files changed, 408 insertions(+), 86 deletions(-) create mode 100644 assets/js/helper.js create mode 100644 assets/js/validate.js diff --git a/assets/js/helper.js b/assets/js/helper.js new file mode 100644 index 0000000..504cb31 --- /dev/null +++ b/assets/js/helper.js @@ -0,0 +1,84 @@ +// Add, Has and Remove class for IE8 +function AddClass( element, elementClass ) { + return element.className += " " + elementClass; +} + +function HasClass( element, elementClass ) { + return element.className.match( new RegExp( "(\\s|^)" + elementClass + "(\\s|$)") ); +} + +function RemoveClass( element, elementClass ) { + if( HasClass( element, elementClass ) ) { + // This adds a space every time a class is removed, we should fix this + var reg = new RegExp( "(\\s|^)" + elementClass + "(\\s|$)" ); + element.className = element.className.replace( reg, " " ); + } +} + + +// Change EventListener to attachEvent +function AddEvent( elements, event, onEvent ) { + if( elements ) { + // Create an array of elements if a singular or array of elements is passed in + if( elements.length === undefined ) { + elements = [ elements ]; + } + + // For each element add the correct event listener + for( var i = 0; i < elements.length; i++ ) { + if( typeof Element.prototype.addEventListener === "undefined" ) { + + // Make sure that we pass this + ( function( element, event ) { + element.attachEvent( "on" + event, function( actualEvent ) { + onEvent( actualEvent, element ); + }); + })( elements[ i ], event ); + } + else { + ( function( element, event ) { + element.addEventListener( event, function( actualEvent ) { + onEvent( actualEvent, element ); + }); + })( elements[ i ], event ); + } + } + } +} + + +// Prevent event for IE8 and modern browsers +function PreventEvent( event ) { + event.preventDefault ? event.preventDefault() : ( event.returnValue = false ); +} + + +// Get elements style +function GetStyle( element, property ) { + return ( + typeof getComputedStyle !== 'undefined' + ? getComputedStyle( element, null) + : element.currentStyle + )[ property ]; // avoid getPropertyValue altogether +} + +//polyfill for IE +//https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/trim#Polyfill +if(typeof String.prototype.trim !== 'function') { + String.prototype.trim = function() { + return this.replace(/^\s+|\s+$/g, ''); + }; +} + +// polyfill for ie8 +// https://developer.mozilla.org/en-US/docs/Web/API/NonDocumentTypeChildNode/nextElementSibling#Polyfill_for_Internet_Explorer_8 +if(!("nextElementSibling" in document.documentElement)){ + Object.defineProperty(Element.prototype, "nextElementSibling", { + get: function(){ + var e = this.nextSibling; + while(e && 1 !== e.nodeType) + e = e.nextSibling; + return e; + } + }); +} diff --git a/assets/js/validate.js b/assets/js/validate.js new file mode 100644 index 0000000..ee66676 --- /dev/null +++ b/assets/js/validate.js @@ -0,0 +1,251 @@ +let fullName = document.getElementById( 'name' ); +let email = document.getElementById( 'email' ); +let helpDescription = document.getElementById( 'help-comment' ); +let formSubmit = document.getElementById( 'form-submit' ); +let errorSummary = document.getElementById( 'error-summary' ); +let errorSummaryMessages = document.getElementById( 'error-list' ); +let errorMessageHeading = document.getElementById( 'error-heading' ); +let errors = { name: '', email: '', 'help-comment': '' }; + + +/** + * Add event listener when form submitted + */ +AddEvent( formSubmit, 'click', ( event, $this ) => { + + // turn off form submission for chameleon, since submitting triggers server side parameter check. + PreventEvent( event ); + if ( !validateForm() ) { + RemoveClass( errorSummary, 'hide_content' ); + generateErrorSummary(); + errorMessageHeading.focus(); + } +}); + + +/** + * Validate all fields on the form + * + */ +function validateForm() { + let isFormValid = false; + isFormValid = validateName(); + isFormValid = validateEmail(); + isFormValid = validateHelpDescription(); + return isFormValid; +} + + +//-------------------------------------------------------------------------------------------------------------------------------------------------------------- +// Field validation functions +//-------------------------------------------------------------------------------------------------------------------------------------------------------------- + + +/** + * Validates name field + */ +function validateName() { + return isFieldEmpty( fullName ); +} + + +/** + * Validates email field + */ +function validateEmail() { + if( isFieldEmpty( email ) ) return; + if( !matchExpression( email, 'EMAIL' ) ) return; + return true; +} + + +/** + * Validates description field + */ +function validateHelpDescription() { + if( isFieldEmpty( helpDescription ) ) return; + if( !meetLength( helpDescription, 10, 100 ) ) return; + return true; +} + + +//-------------------------------------------------------------------------------------------------------------------------------------------------------------- +// UTILITY FUNCTIONS +//-------------------------------------------------------------------------------------------------------------------------------------------------------------- + + +/** + * Checks to see if a text field is empty + * + * @param { HTMLInputElement } field - The html input element + * + * @return {boolean} - False if not empty + */ +function isFieldEmpty( field ) { + if ( field.value.trim() === '' ) { + // set field invalid + setInvalid( field, field.name + ' must not be empty'); + return true; + } + // set field valid + setValid( field ); + return false; + +}; + + +/** + * Displays error text and sets field to invalid + * + * @param { HTMLElement } field - The field to set invalid + * @param { string } message - The error message to be displayed + */ +function setInvalid( field, message ) { + if( !HasClass( field, 'au-text-input--invalid' ) ) { + AddClass( field, 'au-text-input--invalid' ); + } + + // the error span element + var errorField = field.nextElementSibling; + RemoveClass( errorField, 'hide_content' ); + errorField.innerHTML = message; + + field.setAttribute( 'aria-invalid', true ); + AddError( field, message ); +} + + +/** + * Sets a field to be valid. Removes any error messages and stylings + * + * @param { HTMLElement } field - The field to set valid + */ +function setValid( field ) { + // the error span element + let errorField = field.nextElementSibling; + + if( !HasClass( errorField, 'hide_content' ) ) { + AddClass( errorField, 'hide_content' ); + } + + RemoveClass( field, 'au-text-input--invalid' ); + + errorField.innerHTML = ''; + field.setAttribute( 'aria-invalid', false ); + RemoveError( field ); +} + + +/** + * Adds the error messages to an error message summary which is displayed when the form is submitted + * + * @param { HTMLElement } field - The field to add errors to + * @param { string } message - The error message to be shown in the error sumamry + */ +function AddError( field, message ) { + errors[ field.id ] = `
  • ${ message }
  • `; +} + + +/** + * Removes errors from the errors array if the field is valid + * + * @param { HTMLElement } field - The html form control to remove errors from + */ +function RemoveError( field ) { + errors[ field.id ] = ''; +} + + +/** + * Checks to see if a field contains a pattern + * + * @param { HTMLElement } field - The field to be validated against + * @param { string } code - The type of + */ +function matchExpression( field, code ) { + let regEx; + switch ( code ) { + case 'EMAIL': + // Email pattern + // reg ex patter found here: https://emailregex.com/ + regEx = /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/; + return matchWithRegEx( regEx, field, 'Enter an email in a valid format, like name@example.com' ); + // another example + default: + return false; + } +} + + +/** + * Match + * + * @param {RegExp} regEx - The reg ex type to match the field with + * @param { HTMLElement } field - The text field to test against + * @param { string } message - The error message to be displayed if invalid + */ +function matchWithRegEx( regEx, field, message ) { + if( field.value.match( regEx ) ) { + setValid( field ); + return true; + } + else { + setInvalid( field, message ); + return false; + } +} + + +/** + * Checks if a field meets length requirements + * + * @param { HTMLElement } field - the field to check against + * @param { string } minLength - the minimum length of the field + * @param { string } maxLength - the maximum length of the field + */ +function meetLength( field, minLength, maxLength ) { + // If field length meets requirements + if( field.value.trim().length >= minLength && field.value.trim().length <= maxLength ) { + setValid( field ); + return true; + } + else if( field.value.trim().length < minLength ) { + setInvalid( field, `${field.name } must be atleast ${ minLength } characters long` ); + return false; + } + else { + setInvalid( field, `${field.name } must be less than ${ maxLength } characters long` ); + } +} + + +/** + * Generates the error summary on top of page in the page alert + */ +function generateErrorSummary() { + let errorSummaryList = ''; + + // loop over the error array and generate a list of errors + for( let field in errors ) { + if( errors[ field ] ) { + errorSummaryList += errors[ field ]; + } + } + // add the list of errors to the unordered list in the page alert + errorSummaryMessages.innerHTML = errorSummaryList; +} + + +/** + * Stop url changing when clicking in links in the error message summary and focus on selected field + * + * @param { Object } event + */ +function stopUrlChange( event ) { + PreventEvent( event ); + + // ie8 polyfill + let eventTarget = event.target || event.srcElement; + document.getElementById( eventTarget.getAttribute( 'data-id' ) ).focus(); +} diff --git a/package-lock.json b/package-lock.json index af8ef0e..ac81b31 100644 --- a/package-lock.json +++ b/package-lock.json @@ -89,9 +89,9 @@ } }, "@gov.au/core": { - "version": "3.4.0", - "resolved": "https://registry.npmjs.org/@gov.au/core/-/core-3.4.0.tgz", - "integrity": "sha512-YIoJLTRvV1vb2PpEXnlfQGn2iqFtGfKtvqKlNA+rplixBunvQpOL/06FXAYOPCUy9Qs2bCFI0S6KlsEiDjLZSQ==", + "version": "3.4.1", + "resolved": "https://registry.npmjs.org/@gov.au/core/-/core-3.4.1.tgz", + "integrity": "sha512-q1QJ1nftsR29yxCyV5Q+WtwZQx9f8ejjs8okVnOd8m8vtNS89j7kwXRYqw90nOjlOJYXeFRPCqaY+aeBNfvEiA==", "requires": { "@gov.au/pancake": "~1", "@gov.au/pancake-json": "~1", @@ -135,6 +135,18 @@ "@gov.au/pancake-sass": "~2" } }, + "@gov.au/form": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/@gov.au/form/-/form-0.1.2.tgz", + "integrity": "sha512-6c4FpezNzpLfJI/jFbyHCdTgLWOBv+PWwd+y/8JJ2UPWTlraZ8uguVAREQH3xICSwUa+4XAWvW7giUt4542jLQ==", + "requires": { + "@gov.au/core": "^3.0.0", + "@gov.au/pancake": "~1", + "@gov.au/pancake-json": "~1", + "@gov.au/pancake-react": "~1", + "@gov.au/pancake-sass": "~2" + } + }, "@gov.au/grid-12": { "version": "2.0.9", "resolved": "https://registry.npmjs.org/@gov.au/grid-12/-/grid-12-2.0.9.tgz", @@ -197,9 +209,9 @@ } }, "@gov.au/link-list": { - "version": "3.0.5", - "resolved": "https://registry.npmjs.org/@gov.au/link-list/-/link-list-3.0.5.tgz", - "integrity": "sha512-Y4JxpHGhAmIE5FZyUKN4sQTX/RxK5iuxQRbR3xh7k6nKic1BRxgOUdUHZ22eELzfTxsG5fZmakVTiZGRw5NtNQ==", + "version": "3.0.6", + "resolved": "https://registry.npmjs.org/@gov.au/link-list/-/link-list-3.0.6.tgz", + "integrity": "sha512-bPbwj5ls8SeCmAzDsqoGd/B/WiqdD7q8Cy5RWor7NU6sVXg+yjxNpINtqTK77TnkZFnlhwzPs61uX3KRe7YqHw==", "requires": { "@gov.au/body": "^2.0.0", "@gov.au/core": "^3.0.0", @@ -268,9 +280,9 @@ } }, "@gov.au/pancake-sass": { - "version": "2.3.4", - "resolved": "https://registry.npmjs.org/@gov.au/pancake-sass/-/pancake-sass-2.3.4.tgz", - "integrity": "sha512-AL8FJ88tsrebf/rF7ieCTckAyW+wDJ7sNDRG+kvr9CnBH8ktZkxvzWDCsYtN7PZjT0nyIQYf8gwdLPTfAOT83A==", + "version": "2.3.7", + "resolved": "https://registry.npmjs.org/@gov.au/pancake-sass/-/pancake-sass-2.3.7.tgz", + "integrity": "sha512-RoQtXURh76M/awpOeH9X9HERf6vHiVAyzg2Z814DYHyShmWoNsc4rmwPPJCvdvjY1sXBo5vPxsBW2B4w6vrgRw==", "requires": { "@gov.au/pancake": "~1", "autoprefixer": "9.4.10", @@ -2613,8 +2625,7 @@ "ansi-regex": { "version": "2.1.1", "resolved": "https://registry.npmjs.org/ansi-regex/-/ansi-regex-2.1.1.tgz", - "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=", - "optional": true + "integrity": "sha1-w7M6te42DYbg5ijwRorn7yfWVN8=" }, "aproba": { "version": "1.2.0", @@ -2635,14 +2646,12 @@ "balanced-match": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.0.tgz", - "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=", - "optional": true + "integrity": "sha1-ibTRmasr7kneFk6gK4nORi1xt2c=" }, "brace-expansion": { "version": "1.1.11", "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.11.tgz", "integrity": "sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==", - "optional": true, "requires": { "balanced-match": "^1.0.0", "concat-map": "0.0.1" @@ -2657,20 +2666,17 @@ "code-point-at": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/code-point-at/-/code-point-at-1.1.0.tgz", - "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=", - "optional": true + "integrity": "sha1-DQcLTQQ6W+ozovGkDi7bPZpMz3c=" }, "concat-map": { "version": "0.0.1", "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz", - "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=", - "optional": true + "integrity": "sha1-2Klr13/Wjfd5OnMDajug1UBdR3s=" }, "console-control-strings": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/console-control-strings/-/console-control-strings-1.1.0.tgz", - "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=", - "optional": true + "integrity": "sha1-PXz0Rk22RG6mRL9LOVB/mFEAjo4=" }, "core-util-is": { "version": "1.0.2", @@ -2787,8 +2793,7 @@ "inherits": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", - "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=", - "optional": true + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" }, "ini": { "version": "1.3.5", @@ -2800,7 +2805,6 @@ "version": "1.0.0", "resolved": "https://registry.npmjs.org/is-fullwidth-code-point/-/is-fullwidth-code-point-1.0.0.tgz", "integrity": "sha1-754xOG8DGn8NZDr4L95QxFfvAMs=", - "optional": true, "requires": { "number-is-nan": "^1.0.0" } @@ -2815,7 +2819,6 @@ "version": "3.0.4", "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.0.4.tgz", "integrity": "sha512-yJHVQEhyqPLUTgt9B83PXu6W3rx4MvvHvSUvToogpwoGDOUQ+yDrR0HRot+yOCdCO7u4hX3pWft6kWBBcqh0UA==", - "optional": true, "requires": { "brace-expansion": "^1.1.7" } @@ -2823,14 +2826,12 @@ "minimist": { "version": "0.0.8", "resolved": "https://registry.npmjs.org/minimist/-/minimist-0.0.8.tgz", - "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=", - "optional": true + "integrity": "sha1-hX/Kv8M5fSYluCKCYuhqp6ARsF0=" }, "minipass": { "version": "2.3.5", "resolved": "https://registry.npmjs.org/minipass/-/minipass-2.3.5.tgz", "integrity": "sha512-Gi1W4k059gyRbyVUZQ4mEqLm0YIUiGYfvxhF6SIlk3ui1WVxMTGfGdQ2SInh3PDrRTVvPKgULkpJtT4RH10+VA==", - "optional": true, "requires": { "safe-buffer": "^5.1.2", "yallist": "^3.0.0" @@ -2849,7 +2850,6 @@ "version": "0.5.1", "resolved": "https://registry.npmjs.org/mkdirp/-/mkdirp-0.5.1.tgz", "integrity": "sha1-MAV0OOrGz3+MR2fzhkjWaX11yQM=", - "optional": true, "requires": { "minimist": "0.0.8" } @@ -2930,8 +2930,7 @@ "number-is-nan": { "version": "1.0.1", "resolved": "https://registry.npmjs.org/number-is-nan/-/number-is-nan-1.0.1.tgz", - "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=", - "optional": true + "integrity": "sha1-CXtgK1NCKlIsGvuHkDGDNpQaAR0=" }, "object-assign": { "version": "4.1.1", @@ -2943,7 +2942,6 @@ "version": "1.4.0", "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz", "integrity": "sha1-WDsap3WWHUsROsF9nFC6753Xa9E=", - "optional": true, "requires": { "wrappy": "1" } @@ -3029,8 +3027,7 @@ "safe-buffer": { "version": "5.1.2", "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", - "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==", - "optional": true + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" }, "safer-buffer": { "version": "2.1.2", @@ -3066,7 +3063,6 @@ "version": "1.0.2", "resolved": "https://registry.npmjs.org/string-width/-/string-width-1.0.2.tgz", "integrity": "sha1-EYvfW4zcUaKn5w0hHgfisLmxB9M=", - "optional": true, "requires": { "code-point-at": "^1.0.0", "is-fullwidth-code-point": "^1.0.0", @@ -3086,7 +3082,6 @@ "version": "3.0.1", "resolved": "https://registry.npmjs.org/strip-ansi/-/strip-ansi-3.0.1.tgz", "integrity": "sha1-ajhfuIU9lS1f8F0Oiq+UJ43GPc8=", - "optional": true, "requires": { "ansi-regex": "^2.0.0" } @@ -3130,14 +3125,12 @@ "wrappy": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", - "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=", - "optional": true + "integrity": "sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8=" }, "yallist": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/yallist/-/yallist-3.0.3.tgz", - "integrity": "sha512-S+Zk8DEWE6oKpV+vI3qWkaK+jSbIK86pCwe2IF/xwIpQ8jEuxpw9NyaGjmp9+BoJv5FV2piqCDcoCtStppiq2A==", - "optional": true + "integrity": "sha512-S+Zk8DEWE6oKpV+vI3qWkaK+jSbIK86pCwe2IF/xwIpQ8jEuxpw9NyaGjmp9+BoJv5FV2piqCDcoCtStppiq2A==" } } }, diff --git a/package.json b/package.json index 88ecbcc..b26ae22 100644 --- a/package.json +++ b/package.json @@ -47,6 +47,7 @@ "@gov.au/cta-link": "latest", "@gov.au/direction-links": "latest", "@gov.au/footer": "latest", + "@gov.au/form": "^0.1.2", "@gov.au/grid-12": "latest", "@gov.au/header": "latest", "@gov.au/headings": "latest", diff --git a/templates/form/index.html b/templates/form/index.html index 45562b7..0851aa3 100644 --- a/templates/form/index.html +++ b/templates/form/index.html @@ -119,63 +119,53 @@

    Name of website

    Form example (h1)

    The page heading communicates the main focus of the page. Make your page heading descriptive and keep it succinct.

    These headings introduce, respectively, sections and subsections within your body copy. As you create these headings, follow the same guidelines that you use when writing section headings: Be succinct, descriptive, and precise.

    -
    -
    - - + +
    + + +
    -
    - - +
    + + We will only use this to respond to your request + +
    -
    - Preferred contact method (required) - - - -
    - - -
    - How did you hear about us? - - - - -
    - -
    - - +
    +
    + + + +
    + + +
    + +
    + + +
    + +
    + + +
    +
    - -
    - Are you sure about this? (required) - +
    + + Add some detail about your complaint, suggestion or compliment. Minimum 10 characters + +
    -
    - +
    +
    @@ -209,6 +199,9 @@

    Form example (h1)

    + + +