Forms API
Form are created using the Form API. The Form API supports most standard HTML elements, including checkboxes, radio buttons, text boxes, and so on, adding additional accessibility and security features to them.
Highlights
- Tested and optimised for use on major screen-readers like Dragon and JAWS.
- Table-less layout.
- Processes form data securely, with required_param,optional_param, and session key.
- Supports client-side validation.
- Facility to add Moodle help buttons to forms.
- Support for file repository using the File API .
- Support for many custom Moodle specific and non-specific form elements.
- Facility for repeated elements.
- Facility for form elements in advanced groups
Usage
The Moodle forms API separates forms into different areas:
- a form definition, which extends the moodleformclass; and
- uses of that form.
To create a form in Moodle, you create a class that defines the form, including every form element. Your class must extend the moodleform class and overrides the definition member function to describe the form elements.
An example of a form definition
Once the form has been defined it can be instantiated elsewhere in Moodle, for example:
// Instantiate the myform form from within the plugin.
$mform = new \plugintype_pluginname\form\myform();
// Form processing and displaying is done here.
if ($mform->is_cancelled()) {
    // If there is a cancel element on the form, and it was pressed,
    // then the `is_cancelled()` function will return true.
    // You can handle the cancel operation here.
} else if ($fromform = $mform->get_data()) {
    // When the form is submitted, and the data is successfully validated,
    // the `get_data()` function will return the data posted in the form.
} else {
    // This branch is executed if the form is submitted but the data doesn't
    // validate and the form should be redisplayed or on the first display of the form.
    // Set anydefault data (if any).
    $mform->set_data($toform);
    // Display the form.
    $mform->display();
}
If you wish to use the form within a block then you should consider using the render method, as demonstrated below:
Note that the render method does the same as the display method, except returning the HTML rather than outputting it to the browser, as with above make sure you've included the file which contains the class for your Moodle form.
class block_yourblock extends block_base {
    public function init(){
        $this->title = 'Your Block';
    }
    public function get_content(){
        $this->content = (object) [
            'text' => '',
        ];
        $mform = new \plugintype_pluginname\form\myform();
        if ($mform->is_cancelled()) {
            // If there is a cancel element on the form, and it was pressed,
            // then the `is_cancelled()` function will return true.
            // You can handle the cancel operation here.
        } else if ($fromform = $mform->get_data()) {
            // When the form is submitted, and the data is successfully validated,
            // the `get_data()` function will return the data posted in the form.
        } else {
            // This branch is executed if the form is submitted but the data doesn't
            // validate and the form should be redisplayed or on the first display of the form.
            // Set anydefault data (if any).
            $mform->set_data($toform);
            // Display the form.
            $this->content->text = $mform->render();
        }
        return $this->content;
    }
}
Form elements
Moodle provides a number of basic, and advanced, form elements. These are described in more detail below.
Basic form elements
- button
- checkbox
- radio
- select
- multi-select
- password
- hidden
- html - div element
- static - Display a static text.
- text
- textarea
- header
Advanced form elements
- Autocomplete - A select box that allows you to start typing to narrow the list of options, or search for results.
- advcheckbox - Advance checkbox
- float
- passwordunmask - A password element with option to show the password in plaintext.
- recaptcha
- selectyesno
- selectwithlink
- date_selector
- date_time_selector
- duration
- editor
- filepicker - upload single file
- filemanager - upload multiple files
- tags
- addGroup
- modgrade
- modvisible
- choosecoursefile
- grading
- questioncategory
Custom form elements
In addition to the standard form elements, you can register your own custom form elements, for example:
// Register a custom form element.
MoodleQuickForm::registerElementType(
    // The custom element is named `course_competency_rule`.
    // This is the element name used in the `addElement()` function.
    'course_competency_rule',
    // This is where it's definition is defined.
    // This does not currently support class auto-loading.
    "$CFG->dirroot/$CFG->admin/tool/lp/classes/course_competency_rule_form_element.php",
    // The class name of the element.
    'tool_lp_course_competency_rule_form_element'
);
// Add an instance of the custom form element to your form.
$mform->addElement(
    // The name of the custome lement.
    'course_competency_rule',
    'competency_rule',
    get_string('uponcoursemodulecompletion', 'tool_lp'),
    $options
);
For a real-life example, see:
Commonly used functions
add_action_buttons()
Add the standard 'action' buttons to the form - these are the standard Submit, and Cancel buttons on the form.
public function add_action_buttons(
    bool $cancel = true,
    ?string $submitlabel = null
);
- The $cancelparameter can be used to control whether a cancel button is shown.
- The $submitlabelparameter can be used to set the label for the submit button. The default value comes from thesavechangesstring.
The add_action_buttons function is defined on moodlform class, and not a part of $this->_form, for example:
    public function definition() {
        // Add your form elements here.
        $this->_form->addElement(...);
        // When ready, add your action buttons.
        $this->add_action_buttons();
    }
add_sticky_action_buttons()
This method calls add_action_buttons() internally and makes 'action' buttons sticky.
public function add_sticky_action_buttons(
    bool $cancel = true,
    ?string $submitlabel = null
);
setDefault()
The setDefault() function can be used to set the default value for an element.
disabledIf()
The disabledIf() function can be used to conditionally disable a group of elements, or and individual element depending on the state of other form elements.
hideIf()
The hideif() function can be used to conditionally hide a group of elements, or and individual element depending on the state of other form elements.
addRule()
The addRule() function can be used to define both client-side, and server-side validation rules. For example, this can be used to validate that a text-field is required, and has a type of email.
addHelpButton()
The addHelpButton() function can be used to add a pop-up help button to a form element.
setType()
The setType() function can be used to specify how submitted values are cleaned. The PARAM_* constants are used to specify the type of data that will be submitted.
disable_form_change_checker()
Normally, if a user navigate away from any form and changes have been made, a popup will be shown to the user asking them to confirm that they wish to leave the page and that they may lose data.
In some cases this is not the desired behaviour, in which case the disable_form_change_checker() function can be used to disable the form change checker.
For example:
public function definition() {
    // Your definition goes here.
    // Disable the form change checker for this form.
    $this->_form->disable_form_change_checker();
}
filter_shown_headers()
This method adds values to _shownonlyelements array to decide whether a header should be shown or hidden.
Only header names would be accepted and added to _shownonlyelements array.
Headers included in _shownonlyelements will be shown expanded in the form. The rest of the headers will be hidden.
public function filter_shown_headers(array $shownonly): void {
    $this->_shownonlyelements = [];
    if (empty($shownonly)) {
        return;
    }
    foreach ($shownonly as $headername) {
        $element = $this->getElement($headername);
        if ($element->getType() == 'header') {
            $this->_shownonlyelements[] = $headername;
            $this->setExpanded($headername);
        }
    }
}
Empty _shownonlyelements array doesn't affect header's status or visibility.
$showonly = optional_param('showonly', 0, PARAM_TAGLIST);
[...]
$mform = $courseformat->editsection_form($PAGE->url, $customdata);
$initialdata = convert_to_array($sectioninfo);
if (!empty($CFG->enableavailability)) {
    $initialdata['availabilityconditionsjson'] = $sectioninfo->availability;
}
$mform->set_data($initialdata);
if (!empty($showonly)) {
    $mform->filter_shown_headers(explode(',', $showonly));
}
Other features
In some cases you may want to group elements into collections.
Unit testing
In order to test the processing of submitted form contents in unit tests, the Forms API has a mock_submit() function.
This method makes the form behave as if the data supplied to it was submitted by the user via the web interface. The data still passes through all form validation, which means that `get_data() will return all of the parsed values, along with any defaults.