diff --git a/forms/ConfirmedPasswordField.php b/forms/ConfirmedPasswordField.php new file mode 100644 index 000000000..91b244712 --- /dev/null +++ b/forms/ConfirmedPasswordField.php @@ -0,0 +1,205 @@ +children = new FieldSet( + new PasswordField("{$name}[_Password]", 'Password'), + new PasswordField("{$name}[_ConfirmPassword]",'Confirm Password') + ); + + parent::__construct($name, $title, $value, $form); + } + + function Field() { + $content = ''; + foreach($this->children as $field) { + $content.= $field->FieldHolder(); + } + + return $content; + } + + /** + * Can be empty is a flag that turns on/off empty field checking. + * For example, set this to false (the default) when creating a user account, + * and true + */ + function setCanBeEmpty($value) { + $this->canBeEmpty = (bool)$value; + } + + function setRightTitle($title) { + foreach($this->children as $field) { + $field->setRightTitle($title); + } + } + + /** + * Value is sometimes an array, and sometimes a single value, so we need to handle both cases + */ + function setValue($value) { + if(is_array($value)) { + if($value['_Password'] || (!$value['_Password'] && !$this->canBeEmpty)) { + $this->value = $value['_Password']; + } + } else { + if($value || (!$value && !$this->canBeEmpty)) { + $this->value = $value; + } + } + + } + + function jsValidation() + { + $formID = $this->form->FormName(); + + $jsTests = " + if(passEl.value != confEl.value) { + validationError(confEl, \"Passwords have to match.\", \"error\"); + return false; + } + "; + + if(!$this->canBeEmpty) { + $jsTests .= " + if(!passEl.value || !confEl.value) { + validationError(confEl, \"Passwords can't be empty.\", \"error\"); + return false; + } + "; + } + + if(($this->minLength || $this->maxLength) && !$this->canBeEmpty) { + if($this->minLength && $this->maxLength) { + $limit = "{$this->minLength},{$this->maxLength}"; + $errorMsg = "Passwords must be {$this->minLength} to {$this->maxLength} characters long."; + } elseif($this->minLength) { + $limit = "{$this->minLength}"; + $errorMsg = "Passwords must be at least {$this->minLength} characters long."; + } elseif($this->maxLength) { + $limit = "0,{$this->maxLength}"; + $errorMsg = "Passwords must be at most {$this->maxLength} characters long."; + } + $limitRegex = '/^.{' . $limit . '}$/'; + $jsTests .= " + if(!passEl.value.match({$limitRegex})) { + validationError(confEl, \"{$errorMsg}\", \"error\"); + return false; + } + "; + } + + if($this->requireStrongPassword) { + $jsTests .= " + if(!passEl.value.match(/^(([a-zA-Z]+\d+)|(\d+[a-zA-Z]+))[a-zA-Z0-9]*$/)) { + validationError( + confEl, + \"Passwords must have at least one digit and one alphanumeric character.\", + \"error\" + ); + return false; + } + "; + } + + $jsFunc =<<name');"; + return <<name') + $('$formID').validateConfirmedPassword('$this->name'); +}else{ + $('$formID').validateConfirmedPassword('$this->name'); +} +JS; + } + + function validate() { + $validator = $this->form->getValidator(); + $name = $this->name; + $passwordField = $this->children->fieldByName($name.'[_Password]'); + $passwordConfirmField = $this->children->fieldByName($name.'[_ConfirmPassword]'); + $passwordField->setValue($_POST[$name]['_Password']); + $passwordConfirmField->setValue($_POST[$name]['_ConfirmPassword']); + // both password-fields should be the same + if($passwordField->Value() != $passwordConfirmField->Value()) { + $validator->validationError($name, "Passwords don't match", "validation", false); + return false; + } + + if(!$this->canBeEmpty) { + // both password-fields shouldn't be empty + if(!$passwordField->Value() || !$passwordConfirmField->Value()) { + $validator->validationError($name, "Passwords can't be empty", "validation", false); + return false; + } + } + + // lengths + if(($this->minLength || $this->maxLength) && !$this->canBeEmpty) { + if($this->minLength && $this->maxLength) { + $limit = "{$this->minLength},{$this->maxLength}"; + $errorMsg = "Passwords must be {$this->minLength} to {$this->maxLength} characters long."; + } elseif($this->minLength) { + $limit = "{$this->minLength}"; + $errorMsg = "Passwords must be at least {$this->minLength} characters long."; + } elseif($this->maxLength) { + $limit = "0,{$this->maxLength}"; + $errorMsg = "Passwords must be at most {$this->maxLength} characters long."; + } + $limitRegex = '/^.{' . $limit . '}$/'; + if(!preg_match($limitRegex,$passwordField->Value())) { + $validator->validationError('Password', $errorMsg, + "validation", + false + ); + } + } + + if($this->requireStrongPassword) { + if(!preg_match('/^(([a-zA-Z]+\d+)|(\d+[a-zA-Z]+))[a-zA-Z0-9]*$/',$passwordField->Value())) { + $validator->validationError( + 'Password', + "Passwords must have at least one digit and one alphanumeric character.", + "validation", + false + ); + return false; + } + } + return true; + } +} \ No newline at end of file