Merge branch 'MDL-38885-master-int' of git://github.com/FMCorz/moodle

This commit is contained in:
Eloy Lafuente (stronk7) 2013-05-10 03:16:09 +02:00
commit da2cff35c5
2 changed files with 407 additions and 31 deletions

View file

@ -1270,22 +1270,38 @@ abstract class moodleform {
$mform = $this->_form; $mform = $this->_form;
foreach ($mform->_elements as $element) { foreach ($mform->_elements as $element) {
$group = false;
$elements = array($element);
if ($element->getType() == 'group') {
$group = $element;
$elements = $element->getElements();
}
foreach ($elements as $index => $element) {
switch ($element->getType()) { switch ($element->getType()) {
case 'hidden': case 'hidden':
case 'text': case 'text':
case 'url': case 'url':
$key = $element->getName(); if ($group) {
$name = $group->getElementName($index);
} else {
$name = $element->getName();
}
$key = $name;
$found = array_key_exists($key, $mform->_types);
// For repeated elements we need to look for // For repeated elements we need to look for
// the "main" type, not for the one present // the "main" type, not for the one present
// on each repetition. All the stuff in formslib // on each repetition. All the stuff in formslib
// (repeat_elements(), updateSubmission()... seems // (repeat_elements(), updateSubmission()... seems
// to work that way. // to work that way.
$pos = strpos($key, '['); while (!$found && strrpos($key, '[') !== false) {
if ($pos !== false) { $pos = strrpos($key, '[');
$key = substr($key, 0, $pos); $key = substr($key, 0, $pos);
$found = array_key_exists($key, $mform->_types);
} }
if (!array_key_exists($key, $mform->_types)) { if (!$found) {
debugging("Did you remember to call setType() for '$key'? ". debugging("Did you remember to call setType() for '$name'? ".
'Defaulting to PARAM_RAW cleaning.', DEBUG_DEVELOPER); 'Defaulting to PARAM_RAW cleaning.', DEBUG_DEVELOPER);
} }
break; break;
@ -1293,6 +1309,7 @@ abstract class moodleform {
} }
} }
} }
}
/** /**
* MoodleQuickForm implementation * MoodleQuickForm implementation
@ -1673,6 +1690,81 @@ class MoodleQuickForm extends HTML_QuickForm_DHTMLRulesTableless {
$this->_types = $paramtypes + $this->_types; $this->_types = $paramtypes + $this->_types;
} }
/**
* Return the type(s) to use to clean an element.
*
* In the case where the element has an array as a value, we will try to obtain a
* type defined for that specific key, and recursively until done.
*
* This method does not work reverse, you cannot pass a nested element and hoping to
* fallback on the clean type of a parent. This method intends to be used with the
* main element, which will generate child types if needed, not the other way around.
*
* Example scenario:
*
* You have defined a new repeated element containing a text field called 'foo'.
* By default there will always be 2 occurence of 'foo' in the form. Even though
* you've set the type on 'foo' to be PARAM_INT, for some obscure reason, you want
* the first value of 'foo', to be PARAM_FLOAT, which you set using setType:
* $mform->setType('foo[0]', PARAM_FLOAT).
*
* Now if you call this method passing 'foo', along with the submitted values of 'foo':
* array(0 => '1.23', 1 => '10'), you will get an array telling you that the key 0 is a
* FLOAT and 1 is an INT. If you had passed 'foo[1]', along with its value '10', you would
* get the default clean type returned (param $default).
*
* @param string $elementname name of the element.
* @param mixed $value value that should be cleaned.
* @param int $default default constant value to be returned (PARAM_...)
* @return string|array constant value or array of constant values (PARAM_...)
*/
public function getCleanType($elementname, $value, $default = PARAM_RAW) {
$type = $default;
if (array_key_exists($elementname, $this->_types)) {
$type = $this->_types[$elementname];
}
if (is_array($value)) {
$default = $type;
$type = array();
foreach ($value as $subkey => $subvalue) {
$typekey = "$elementname" . "[$subkey]";
if (array_key_exists($typekey, $this->_types)) {
$subtype = $this->_types[$typekey];
} else {
$subtype = $default;
}
if (is_array($subvalue)) {
$type[$subkey] = $this->getCleanType($typekey, $subvalue, $subtype);
} else {
$type[$subkey] = $subtype;
}
}
}
return $type;
}
/**
* Return the cleaned value using the passed type(s).
*
* @param mixed $value value that has to be cleaned.
* @param int|array $type constant value to use to clean (PARAM_...), typically returned by {@link self::getCleanType()}.
* @return mixed cleaned up value.
*/
public function getCleanedValue($value, $type) {
if (is_array($type) && is_array($value)) {
foreach ($type as $key => $param) {
$value[$key] = $this->getCleanedValue($value[$key], $param);
}
} else if (!is_array($type) && !is_array($value)) {
$value = clean_param($value, $type);
} else if (!is_array($type) && is_array($value)) {
$value = clean_param_array($value, $type, true);
} else {
throw new coding_exception('Unexpected type or value received in MoodleQuickForm::getCleanedValue()');
}
return $value;
}
/** /**
* Updates submitted values * Updates submitted values
* *
@ -1686,16 +1778,8 @@ class MoodleQuickForm extends HTML_QuickForm_DHTMLRulesTableless {
$this->_submitValues = array(); $this->_submitValues = array();
} else { } else {
foreach ($submission as $key => $s) { foreach ($submission as $key => $s) {
if (array_key_exists($key, $this->_types)) { $type = $this->getCleanType($key, $s);
$type = $this->_types[$key]; $submission[$key] = $this->getCleanedValue($s, $type);
} else {
$type = PARAM_RAW;
}
if (is_array($s)) {
$submission[$key] = clean_param_array($s, $type, true);
} else {
$submission[$key] = clean_param($s, $type);
}
} }
$this->_submitValues = $submission; $this->_submitValues = $submission;
$this->_flagSubmitted = true; $this->_flagSubmitted = true;

View file

@ -209,9 +209,20 @@ class formslib_testcase extends advanced_testcase {
$mform->display(); $mform->display();
} }
public function test_settype_debugging_url() {
$this->resetAfterTest(true);
$this->setAdminUser();
$mform = new formslib_settype_debugging_url();
$this->assertDebuggingCalled("Did you remember to call setType() for 'urltest'? Defaulting to PARAM_RAW cleaning.");
// Check form still there though
$this->expectOutputRegex('/<input[^>]*name="urltest"[^>]*type="text/');
$mform->display();
}
public function test_settype_debugging_repeat() { public function test_settype_debugging_repeat() {
$mform = new formslib_settype_debugging_repeat(); $mform = new formslib_settype_debugging_repeat();
$this->assertDebuggingCalled("Did you remember to call setType() for 'repeattest'? Defaulting to PARAM_RAW cleaning."); $this->assertDebuggingCalled("Did you remember to call setType() for 'repeattest[0]'? Defaulting to PARAM_RAW cleaning.");
// Check form still there though // Check form still there though
$this->expectOutputRegex('/<input[^>]*name="repeattest[^>]*type="text/'); $this->expectOutputRegex('/<input[^>]*name="repeattest[^>]*type="text/');
@ -225,6 +236,161 @@ class formslib_testcase extends advanced_testcase {
$this->expectOutputRegex('/<input[^>]*name="repeattest[^>]*type="text/'); $this->expectOutputRegex('/<input[^>]*name="repeattest[^>]*type="text/');
$mform->display(); $mform->display();
} }
public function test_settype_debugging_group() {
$mform = new formslib_settype_debugging_group();
$this->assertDebuggingCalled("Did you remember to call setType() for 'groupel1'? Defaulting to PARAM_RAW cleaning.");
$this->expectOutputRegex('/<input[^>]*name="groupel1"[^>]*type="text/');
$this->expectOutputRegex('/<input[^>]*name="groupel2"[^>]*type="text/');
$mform->display();
}
public function test_settype_debugging_namedgroup() {
$mform = new formslib_settype_debugging_namedgroup();
$this->assertDebuggingCalled("Did you remember to call setType() for 'namedgroup[groupel1]'? Defaulting to PARAM_RAW cleaning.");
$this->expectOutputRegex('/<input[^>]*name="namedgroup\[groupel1\]"[^>]*type="text/');
$this->expectOutputRegex('/<input[^>]*name="namedgroup\[groupel2\]"[^>]*type="text/');
$mform->display();
}
public function test_settype_debugging_funky_name() {
$mform = new formslib_settype_debugging_funky_name();
$this->assertDebuggingCalled("Did you remember to call setType() for 'blah[foo][bar][1]'? Defaulting to PARAM_RAW cleaning.");
$this->expectOutputRegex('/<input[^>]*name="blah\[foo\]\[bar\]\[0\]"[^>]*type="text/');
$this->expectOutputRegex('/<input[^>]*name="blah\[foo\]\[bar\]\[1\]"[^>]*type="text/');
$mform->display();
}
public function test_settype_debugging_type_inheritance() {
$mform = new formslib_settype_debugging_type_inheritance();
$this->expectOutputRegex('/<input[^>]*name="blah\[foo\]\[bar\]\[0\]"[^>]*type="text/');
$this->expectOutputRegex('/<input[^>]*name="blah\[bar\]\[foo\]\[1\]"[^>]*type="text/');
$this->expectOutputRegex('/<input[^>]*name="blah\[any\]\[other\]\[2\]"[^>]*type="text/');
$mform->display();
}
public function test_type_cleaning() {
$expectedtypes = array(
'simpleel' => PARAM_INT,
'groupel1' => PARAM_INT,
'groupel2' => PARAM_FLOAT,
'groupel3' => PARAM_INT,
'namedgroup' => array(
'sndgroupel1' => PARAM_INT,
'sndgroupel2' => PARAM_FLOAT,
'sndgroupel3' => PARAM_INT
),
'namedgroupinherit' => array(
'thdgroupel1' => PARAM_INT,
'thdgroupel2' => PARAM_INT
),
'repeatedel' => array(
0 => PARAM_INT,
1 => PARAM_INT
),
'repeatedelinherit' => array(
0 => PARAM_INT,
1 => PARAM_INT
),
'squaretest' => array(
0 => PARAM_INT
),
'nested' => array(
0 => array(
'bob' => array(
123 => PARAM_INT,
'foo' => PARAM_FLOAT
),
'xyz' => PARAM_RAW
),
1 => PARAM_INT
)
);
$valuessubmitted = array(
'simpleel' => '11.01',
'groupel1' => '11.01',
'groupel2' => '11.01',
'groupel3' => '11.01',
'namedgroup' => array(
'sndgroupel1' => '11.01',
'sndgroupel2' => '11.01',
'sndgroupel3' => '11.01'
),
'namedgroupinherit' => array(
'thdgroupel1' => '11.01',
'thdgroupel2' => '11.01'
),
'repeatedel' => array(
0 => '11.01',
1 => '11.01'
),
'repeatedelinherit' => array(
0 => '11.01',
1 => '11.01'
),
'squaretest' => array(
0 => '11.01'
),
'nested' => array(
0 => array(
'bob' => array(
123 => '11.01',
'foo' => '11.01'
),
'xyz' => '11.01'
),
1 => '11.01'
)
);
$expectedvalues = array(
'simpleel' => 11,
'groupel1' => 11,
'groupel2' => 11.01,
'groupel3' => 11,
'namedgroup' => array(
'sndgroupel1' => 11,
'sndgroupel2' => 11.01,
'sndgroupel3' => 11
),
'namedgroupinherit' => array(
'thdgroupel1' => 11,
'thdgroupel2' => 11
),
'repeatable' => 2,
'repeatedel' => array(
0 => 11,
1 => 11
),
'repeatableinherit' => 2,
'repeatedelinherit' => array(
0 => 11,
1 => 11
),
'squaretest' => array(
0 => 11
),
'nested' => array(
0 => array(
'bob' => array(
123 => 11,
'foo' => 11.01
),
'xyz' => '11.01'
),
1 => 11
)
);
$mform = new formslib_clean_value();
$mform->get_form()->updateSubmission($valuessubmitted, null);
foreach ($expectedtypes as $elementname => $expected) {
$actual = $mform->get_form()->getCleanType($elementname, $valuessubmitted[$elementname]);
$this->assertSame($expected, $actual, "Failed validating clean type of '$elementname'");
}
$data = $mform->get_data();
$this->assertSame($expectedvalues, (array) $data);
}
} }
@ -277,6 +443,15 @@ class formslib_settype_debugging_hidden extends moodleform {
} }
} }
// Used to test debugging is called when hidden added without setType.
class formslib_settype_debugging_url extends moodleform {
public function definition() {
$mform = $this->_form;
$mform->addElement('url', 'urltest', 'urltest');
}
}
// Used to test debugging is called when repeated text added without setType. // Used to test debugging is called when repeated text added without setType.
class formslib_settype_debugging_repeat extends moodleform { class formslib_settype_debugging_repeat extends moodleform {
public function definition() { public function definition() {
@ -302,3 +477,120 @@ class formslib_settype_debugging_repeat_ok extends moodleform {
$this->repeat_elements($repeatels, 2, array('repeattest' => array('type' => PARAM_RAW)), 'numtexts', 'addtexts'); $this->repeat_elements($repeatels, 2, array('repeattest' => array('type' => PARAM_RAW)), 'numtexts', 'addtexts');
} }
} }
// Used to test if debugging is called when a group contains elements without type.
class formslib_settype_debugging_group extends moodleform {
public function definition() {
$mform = $this->_form;
$group = array(
$mform->createElement('text', 'groupel1', 'groupel1'),
$mform->createElement('text', 'groupel2', 'groupel2')
);
$mform->addGroup($group);
$mform->setType('groupel2', PARAM_INT);
}
}
// Used to test if debugging is called when a named group contains elements without type.
class formslib_settype_debugging_namedgroup extends moodleform {
public function definition() {
$mform = $this->_form;
$group = array(
$mform->createElement('text', 'groupel1', 'groupel1'),
$mform->createElement('text', 'groupel2', 'groupel2')
);
$mform->addGroup($group, 'namedgroup');
$mform->setType('namedgroup[groupel2]', PARAM_INT);
}
}
// Used to test if debugging is called when has a funky name.
class formslib_settype_debugging_funky_name extends moodleform {
public function definition() {
$mform = $this->_form;
$mform->addElement('text', 'blah[foo][bar][0]', 'test', 'test');
$mform->addElement('text', 'blah[foo][bar][1]', 'test', 'test');
$mform->setType('blah[foo][bar][0]', PARAM_INT);
}
}
// Used to test that debugging is not called with type inheritance.
class formslib_settype_debugging_type_inheritance extends moodleform {
public function definition() {
$mform = $this->_form;
$mform->addElement('text', 'blah[foo][bar][0]', 'test1', 'test');
$mform->addElement('text', 'blah[bar][foo][1]', 'test2', 'test');
$mform->addElement('text', 'blah[any][other][2]', 'test3', 'test');
$mform->setType('blah[foo][bar]', PARAM_INT);
$mform->setType('blah[bar]', PARAM_FLOAT);
$mform->setType('blah', PARAM_TEXT);
}
}
class formslib_clean_value extends moodleform {
public function get_form() {
return $this->_form;
}
public function definition() {
$mform = $this->_form;
// Add a simple int.
$mform->addElement('text', 'simpleel', 'simpleel');
$mform->setType('simpleel', PARAM_INT);
// Add a non-named group.
$group = array(
$mform->createElement('text', 'groupel1', 'groupel1'),
$mform->createElement('text', 'groupel2', 'groupel2'),
$mform->createElement('text', 'groupel3', 'groupel3')
);
$mform->setType('groupel1', PARAM_INT);
$mform->setType('groupel2', PARAM_FLOAT);
$mform->setType('groupel3', PARAM_INT);
$mform->addGroup($group);
// Add a named group.
$group = array(
$mform->createElement('text', 'sndgroupel1', 'sndgroupel1'),
$mform->createElement('text', 'sndgroupel2', 'sndgroupel2'),
$mform->createElement('text', 'sndgroupel3', 'sndgroupel3')
);
$mform->addGroup($group, 'namedgroup');
$mform->setType('namedgroup[sndgroupel1]', PARAM_INT);
$mform->setType('namedgroup[sndgroupel2]', PARAM_FLOAT);
$mform->setType('namedgroup[sndgroupel3]', PARAM_INT);
// Add a named group, with inheritance.
$group = array(
$mform->createElement('text', 'thdgroupel1', 'thdgroupel1'),
$mform->createElement('text', 'thdgroupel2', 'thdgroupel2')
);
$mform->addGroup($group, 'namedgroupinherit');
$mform->setType('namedgroupinherit', PARAM_INT);
// Add a repetition.
$repeat = $mform->createElement('text', 'repeatedel', 'repeatedel');
$this->repeat_elements(array($repeat), 2, array('repeatedel' => array('type' => PARAM_INT)), 'repeatable', 'add', 0);
// Add a repetition, with inheritance.
$repeat = $mform->createElement('text', 'repeatedelinherit', 'repeatedelinherit');
$this->repeat_elements(array($repeat), 2, array(), 'repeatableinherit', 'add', 0);
$mform->setType('repeatedelinherit', PARAM_INT);
// Add an arbitrary named element.
$mform->addElement('text', 'squaretest[0]', 'squaretest[0]');
$mform->setType('squaretest[0]', PARAM_INT);
// Add an arbitrary nested array named element.
$mform->addElement('text', 'nested[0][bob][123]', 'nested[0][bob][123]');
$mform->setType('nested[0][bob][123]', PARAM_INT);
// Add inheritance test cases.
$mform->setType('nested', PARAM_INT);
$mform->setType('nested[0]', PARAM_RAW);
$mform->setType('nested[0][bob]', PARAM_FLOAT);
$mform->addElement('text', 'nested[1]', 'nested[1]');
$mform->addElement('text', 'nested[0][xyz]', 'nested[0][xyz]');
$mform->addElement('text', 'nested[0][bob][foo]', 'nested[0][bob][foo]');
}
}