blob: 2898a913172931ae07bcb915a3485f1c47b7af6f (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
|
<?php
/**
* TReCaptchaValidator class file
*
* @author Bérczi Gábor <gabor.berczi@devworx.hu>
* @link http://www.devworx.hu/
* @copyright Copyright © 2011 DevWorx
* @license http://www.pradosoft.com/license/
* @package System.Web.UI.WebControls
*/
Prado::using('System.Web.UI.WebControls.TBaseValidator');
Prado::using('System.Web.UI.WebControls.TReCaptcha');
/**
* TReCaptchaValidator class
*
* TReCaptchaValidator validates user input against a reCAPTCHA represented by
* a {@link TReCaptcha} control. The input control fails validation if its value
* is not the same as the token displayed in reCAPTCHA. Note, if the user does
* not enter any thing, it is still considered as failing the validation.
*
* To use TReCaptchaValidator, specify the {@link setControlToValidate ControlToValidate}
* to be the ID path of the {@link TReCaptcha} control.
*
* @author Bérczi Gábor <gabor.berczi@devworx.hu>
* @package System.Web.UI.WebControls
* @since 3.2
*/
class TReCaptchaValidator extends TBaseValidator
{
protected $_isvalid = null;
/**
* Gets the name of the javascript class responsible for performing validation for this control.
* This method overrides the parent implementation.
* @return string the javascript class name
*/
protected function getClientClassName()
{
return 'Prado.WebUI.TReCaptchaValidator';
}
public function getEnableClientScript()
{
return true;
}
protected function getCaptchaControl()
{
$control = $this->getValidationTarget();
if (!$control)
throw new Exception('No target control specified for TReCaptchaValidator');
if (!($control instanceof TReCaptcha))
throw new Exception('TReCaptchaValidator only works with TReCaptcha controls');
return $control;
}
public function getClientScriptOptions()
{
$options = parent::getClientScriptOptions();
$options['ResponseFieldName'] = $this->getCaptchaControl()->getResponseFieldName();
return $options;
}
/**
* This method overrides the parent's implementation.
* The validation succeeds if the input control has the same value
* as the one displayed in the corresponding RECAPTCHA control.
*
* @return boolean whether the validation succeeds
*/
protected function evaluateIsValid()
{
// check validity only once (if trying to evaulate multiple times, all redundant checks would fail)
if (is_null($this->_isvalid))
{
$control = $this->getCaptchaControl();
$this->_isvalid = $control->validate();
}
return ($this->_isvalid==true);
}
public function onPreRender($param)
{
parent::onPreRender($param);
$cs = $this->Page->getClientScript();
// communicate validation status to the client side
$value = $this->_isvalid===false ? '0' : '1';
$cs->registerHiddenField($this->getClientID().'_1',$value);
// check if we need to request a new captcha too
if ($this->Page->IsCallback)
{
// force update of validator display
if ($control = $this->getValidationTarget())
{
$cs->registerEndScript(
$this->getClientID().'::validate',
'$(\''.TJavaScript::quoteString($this->getClientID().'_1').'\').value = \''.TJavaScript::quoteString($value).'\';'.
'Prado.Validation.validateControl(\''.TJavaScript::quoteString($control->ClientID).'\');'
);
if ($control->getVisible(true))
if ($this->_isvalid)
{
// if the challenge has been solved + we're in a callback and we still reach prerender phase,
// that means that some other validator failed and the user will be sent back to the page/form with
// the captcha control. in this case we need to force re-rendering of the control, because once
// solved, the old challenge won't validate anymore anyway
$control->regenerateToken();
}
}
}
}
}
?>
|