All notable changes to this project will be documented in this file.
Note - All hash comments refer to the issue number. Eg. #169 refers to #169.
- Minimum supported version is Moodle 5.2 (PHP 8.3+).
- Element System v2 (interfaces + services) to improve stability and long-term extensibility of Custom Certificate elements.
- Base element contract:
mod_customcert\element\element_interface— all v2 elements implement this; the factory returns this type. - New element capability interfaces (implement as needed):
mod_customcert\element\form_buildable_interface— providesbuild_form(MoodleQuickForm $mform)to add element-specific fields to the edit form; replaces the legacyrender_form_elements()hook.mod_customcert\element\persistable_element_interface— providesnormalise_data(stdClass $formdata): arrayto return a typed, element-specific associative array ready for JSON encoding; standard visual fields are merged centrally bypersistence_helper.mod_customcert\element\validatable_element_interface— providesvalidate(array $data): arrayto return field-keyed error messages;validation_servicedetects and invokes this automatically.mod_customcert\element\preparable_form_interface— providesprepare_form(MoodleQuickForm $mform)to populate form fields from the stored JSON payload after the form is built (replacesdefinition_after_data()).mod_customcert\element\stylable_element_interface— typed contract for elements that expose standard visual attributes (get_font(),get_fontsize(),get_colour(),get_width()); values are read from the JSON payload rather than removed DB columns.mod_customcert\element\renderable_element_interface— (Scaffolding) typed render contract:render(pdf, bool, stdClass, ?element_renderer)for PDF output andrender_html(?element_renderer): stringfor the designer preview.mod_customcert\element\restorable_element_interface— (Scaffolding) providesafter_restore_from_backup(restore_customcert_activity_task $restore)to remap internal IDs/references after a backup is restored; replaces the legacyafter_restore()hook.
constructable_element_interface— providesfrom_record(\stdClass $record)to opt in to the preferred factory construction path (see Changed section).unknown_element— safe fallback used by the factory when a requested element type is not registered; renders gracefully rather than throwing.- New persistence + migration helpers:
- JSON payload helpers (
get_payload(),get_value(), safe decode/encode helpers with validation). mod_customcert\service\persistence_helper— centralises JSON normalisation and merging of standard visual fields at save time.- Restore migration helper to merge legacy backup fields into the JSON payload during restore.
- JSON payload helpers (
- Element registry + factory:
mod_customcert\service\element_registry— maps element type keys to class names; replaces hard-coded class resolution.mod_customcert\service\element_factory— instantiates elements via the registry, preferringconstructable_element_interface::from_record()and falling back to legacy constructors.
mod_customcert\service\validation_service— centralises element form-data validation, delegating tovalidatable_element_interfacewhen available.- Expanded automated tests covering Element System v2 behaviour and upgrade/restore data migration paths.
- New service-layer APIs for template/page/element CRUD:
mod_customcert\service\template_service— orchestrates template/page/element create, update, delete, and move operations (plus DTOs such aspage_update)mod_customcert\service\template_repository— low-level DB access forcustomcert_templatesmod_customcert\service\page_repository— low-level DB access forcustomcert_pagesmod_customcert\service\element_repository— low-level DB access forcustomcert_elementsmod_customcert\service\template_duplication_service— transactional copy of a template with all its pages and elementsmod_customcert\service\template_load_service— replaces an existing template's pages/elements with content from another templatemod_customcert\service\item_move_service— handles moving pages and elements by swapping sequencesmod_customcert\service\pdf_generation_service— PDF generation, preview, and filename computation
- New certificate issuance and delivery services:
mod_customcert\service\certificate_issue_service— issues certificates and generates unique codesmod_customcert\service\certificate_download_service— handles bulk certificate download for instances and site-widemod_customcert\service\certificate_time_service— computes course time for certificate conditionsmod_customcert\service\certificate_email_service— sends certificate emails to students and teachersmod_customcert\service\certificate_issuer_service— orchestrates the scheduled issuance and email pipeline
- New repository classes:
mod_customcert\service\issue_repository— queriescustomcert_issuesmod_customcert\service\certificate_repository— queries certificates per usermod_customcert\service\issue_email_repository— tracks email send state per issue
- New rendering infrastructure:
mod_customcert\service\element_renderer— interface for the v2 element rendering pipeline (PDF and HTML surfaces)mod_customcert\service\pdf_renderer—element_rendererimplementation for PDF outputmod_customcert\service\html_renderer—element_rendererimplementation for the designer preview
mod_customcert\template::load(int $id)is the supported entry point for instantiating templates; production code should no longer callnew template($record).mod_customcert\element\legacy_element_adapter— wraps a legacy element (extendingmod_customcert\element) to satisfyelement_interface; returned by the factory for unupgraded element plugins.- Static utility methods on
certificatehave been moved to dedicated service/repository classes:certificate::get_issues()/certificate::get_number_of_issues()/certificate::get_conditional_issues_sql()→issue_repositorycertificate::get_number_of_certificates_for_user()/certificate::get_certificates_for_user()→certificate_repositorycertificate::get_fonts()/certificate::get_font_sizes()→element_helpercertificate::set_protection()→form_servicecertificate::upload_files()→form_service
customcert_elementsno longer hasfont,fontsize,colour, orwidthcolumns; these values are now stored inside the JSON payload incustomcert_elements.data.customcert_elements.datais now treated as a JSON payload:- Legacy scalar values are migrated into a JSON envelope (e.g. legacy
"Hello"→{ "value": "Hello" }). - Upgrade and restore routines automatically migrate existing data.
- Legacy scalar values are migrated into a JSON envelope (e.g. legacy
Migration impact for element plugins:
- Do NOT read/write
$element->font,$element->fontsize,$element->colour,$element->widthfrom DB fields (they no longer exist). - Use element getters (
get_font(),get_fontsize(),get_colour(),get_width()) which now read from the JSON payload. - Element plugins should return only element-specific keys from
persistable_element_interface::normalise_data(); standard visual keys are merged centrally.
Reserved JSON keys (visuals):
- The following keys are used by core as standard visual fields and should be treated as reserved in
customcert_elements.data:width,font,fontsize,colour
- Element plugins should avoid using these keys for element-specific payload data.
- Element edit handlers now normalise element-specific data into a JSON payload and merge standard visual fields (
font,fontsize,colour,width) into the same payload. - Saving elements is hardened to avoid corrupting stored JSON payloads when a caller provides a JSON list/array.
- Factory now prefers
constructable_element_interface::from_record(\stdClass $record)when present on the element class. - Legacy constructors remain supported as a fallback (
new Class($record)). - Factory returns
mod_customcert\element\element_interface; legacy elements may be wrapped bymod_customcert\element\legacy_element_adapter.- Avoid
instanceofchecks against legacy concrete classes; prefer$element->get_type()and/or$element->get_inner().
- Avoid
- Element render methods are now typed and accept an optional renderer.
Before (legacy):
public function render($pdf, $preview, $user) { /* ... */ }
public function render_html() { /* ... */ }
After (5.2+):
use mod_customcert\service\element_renderer;
public function render(
pdf $pdf,
bool $preview,
\stdClass $user,
?element_renderer $renderer = null
): void {}
public function render_html(?element_renderer $renderer = null): string {}
element_helper::render_content()remains unchanged for backward compatibility, but element plugins must update their method signatures as shown above.
- Restoring backups created on older versions is supported: legacy
font/fontsize/colour/widthvalues are merged into the JSON payload during restore. - Backups created on 5.2+ store visual fields inside JSON and are not guaranteed to restore correctly on older plugin versions (no backwards compatibility guarantee).
- Improved robustness of upgrade/restore migrations:
- Better handling of missing/partial legacy fields.
- Safer normalisation of element payloads during restore.
- Legacy border elements that stored thickness as a scalar
datavalue are now migrated so width is preserved correctly.
certificate_issue_service::generate_code()now throws amoodle_exceptionafter 10 failed attempts to generate a unique code, preventing an infinite loop in systems with a very large number of issued certificates.- Web service hardening:
external::save_element()ignores JSON list payloads fordatato prevent numeric-key pollution of stored element JSON.
- Fixed authorization bypass (CVE-2026-30884, CWE-639) where a user with mod/customcert:manage in any single course could read and overwrite certificate elements belonging to other courses by supplying an arbitrary elementid. Element ownership is now validated against the authorised context/template before any read or write operation is performed.
The following template methods are now shims that emit developer debugging and delegate to service classes. Third-party developers should swap legacy calls for the service methods; the shims will be removed in a future release.
template::save()→template_service::update()template::add_page()/template::save_page()→template_service::add_page()/template_service::save_pages()template::delete()/template::delete_page()/template::delete_element()→template_service::delete()/template_service::delete_page()/template_service::delete_element()template::copy_to_template()→template_service::copy_to_template()template::move_item()→template_service::move_item()(service constants are available; raw strings remain supported)template::generate_pdf()/template::create_preview_pdf()/template::compute_filename_for_user()→ correspondingpdf_generation_servicemethods
The following certificate methods are now shims that emit developer debugging and should be replaced with service calls:
certificate::issue_certificate()/certificate::generate_code()→certificate_issue_service::issue_certificate()/::generate_code()certificate::download_all_issues_for_instance()/certificate::download_all_for_site()→certificate_download_serviceequivalentscertificate::get_course_time()→certificate_time_service::get_course_time()certificate::get_issues()/certificate::get_number_of_issues()/certificate::get_conditional_issues_sql()→issue_repositorycertificate::get_number_of_certificates_for_user()/certificate::get_certificates_for_user()→certificate_repositorycertificate::get_fonts()/certificate::get_font_sizes()→element_helpercertificate::set_protection()→form_servicecertificate::upload_files()→form_service
Legacy element APIs are still supported but deprecated as of 5.2:
element::render_form_elements()→ useform_buildable_interface::build_form()andelement_helper::render_common_form_elements()for standard fieldselement::definition_after_data()→ usepreparable_form_interface::prepare_form()element::validate_form_elements()→ usevalidatable_element_interface::validate()element::save_form_elements()/element::save_unique_data()→ usepersistable_element_interface::normalise_data()element::after_restore()→ userestorable_element_interface::after_restore_from_backup()element::delete()→ usemod_customcert\service\element_repository::delete()(elements should not delete themselves; deletion is handled by the repository/service layer)
- Deprecated APIs are expected to continue working during the 5.2 line for most common usage patterns, but some edge cases may break — see "Plugin developer migration risks" below. New development should use Element System v2 interfaces.
- New element plugins should not rely on
customcert_elementslegacy columns or legacy element hooks.
-
DB columns removed from
customcert_elements:fontfontsizecolourwidth
-
Element removed from the plugin distribution:
daterange— if you use this element, install it separately before upgrading:
While deprecated APIs include shims for backward compatibility, the following changes may affect third-party element plugins:
- JSON data format — The
datacolumn now stores a JSON object payload. Plugins that previously treatedget_data()as a plain scalar string must be updated to work with JSON. Useget_payload()to decode, orget_value()for simple scalar elements. - Save path changes — Element saves (both from the edit form and via the web service) now route through
persistence_helper::to_json_data()andelement_repository::save(). Plugins with custom save-time side effects insave_form_elements()orsave_unique_data()should verify that these are still called via the legacy adapter path. - DB columns removed — The
font,fontsize,colour, andwidthcolumns no longer exist incustomcert_elements. Any direct SQL queries referencing these columns will fail. normalise_data()scalar wrapping — If apersistable_element_interfaceelement'snormalise_data()returns a plain string that is not already a JSON object, it will be wrapped as{"value": "..."}before storage. Plugins that return a raw scalar fromnormalise_data()and read it back directly viaget_data()must switch toget_value().
Recommendation: Test your element plugin against 5.2 on a test site before release. Implement the Element System v2 interfaces (Option A below) for the most robust forward compatibility.
Update your element class to implement interfaces as needed:
class element extends \mod_customcert\element implements
\mod_customcert\element\form_buildable_interface,
\mod_customcert\element\persistable_element_interface,
\mod_customcert\element\validatable_element_interface,
\mod_customcert\element\preparable_form_interface {
public function build_form(\MoodleQuickForm $mform): void {
// Add element-specific fields.
$mform->addElement('text', 'myfield', get_string('myfield', 'customcertelement_myplugin'));
$mform->setType('myfield', PARAM_TEXT);
$mform->addHelpButton('myfield', 'myfield', 'customcertelement_myplugin');
// Add standard fields (font, colour, position, width, refpoint, alignment).
\mod_customcert\element_helper::render_common_form_elements($mform);
}
public function normalise_data(\stdClass $formdata): array {
return [
'value' => (string)($formdata->myfield ?? ''),
// NOTE: width/font/colour/fontsize are merged into JSON centrally by the edit handler.
];
}
public function validate(array $data): array {
$errors = [];
if (empty($data['myfield'])) {
$errors['myfield'] = get_string('required');
}
return $errors;
}
public function prepare_form(\MoodleQuickForm $mform): void {
// Example: populate form values by reading from JSON.
$payload = $this->get_payload();
if (is_array($payload) && array_key_exists('value', $payload)) {
$mform->getElement('myfield')->setValue((string)$payload['value']);
}
}
}
Existing element plugins may continue using legacy hooks in 5.2, but must be updated to tolerate JSON data and the removal of DB columns:
- Treat
get_data()as JSON and read your values from the decoded payload (get_payload()). - If you previously depended on DB columns for
font/fontsize/colour/width, switch to getters (get_font(),get_fontsize(),get_colour(),get_width()).
- Do not assume
get_data()returns a scalar string; it may be JSON with multiple keys. - Do not overwrite the whole JSON payload with a single scalar; always normalise to an object payload via Element System v2 interfaces.
- Do not store standard visual fields (
font/fontsize/colour/width) in custom keys; rely on the core merge behaviour.
- Ensure certificates are only emailed to users who can view the certificate and have the 'mod/customcert:receiveissue' capability (#732).
- Fixed emails not being sent for certificates on the site home page (#693).
- Added customisable filename options for certificates (#684).
- Added missing fields to backups (#705).
- Ensure certificates are issued and emailed only once when emailing is enabled (#671).
- Fixed "grade_item" not found error due to missing required file.
- Fixed language handling for certificates and emails (#717).
- Fixed issue with code dashes being stripped on verification page (#727).
- Allow teachers enrolled as students to receive certificates (#690).
- Standardization of spacing in HTML as in the plain text version of emails (#724).
- Hard-coded URL in email is now stored as language string allowing it to be modified/moved (#724).
- Add setting that adds a button to return to course from the certificate page (#655).
- Added awarded on date to verification page (#566).
- Added webservice to retrieve a list of issued certificates (#644).
- Updated subplugins.json file to match Moodle 5.0 format (#688).
- Added choice for code format (#668).
- Added events for adding and deleting certificate issues (#675).
- Fixed enrol start date not showing (#410).
- Pass the user's mailformat preference when emailing certificates (#665).
- Added index to the 'code' column in 'customcert_issues' table (#666).
- A SQL error in the issue certificate task on MSSQL and Oracle (#646).
- Issuing a certificate when there are no issues yet (#659).
- Issue in rearrange JS with Modal not closing (#648).
- ISO 8601 date format (#638).
- A SQL error in the issue certificate task on Oracle (#646).
- Verification error when expiry date element is present (#647).
- Only fetch teachers during the email process when necessary, reducing the number of SQL queries if they are not included (#531).
- Filter users before process to speed up certificate task (#634).
- Mobile app: Stop using deprecated module-description.
- Fixed auto-linking filters moving text element positions if reference point is center (#629).
- Mobile app: Update Mobile template to Ionic 7.
- Mobile app: Remove Ionic 3 template.
- Optimise email certificate task by reducing database reads/writes and introducing configurable settings for task efficiency (#531).
- New element
expirywhich when used will display the expiry date on the list of issued certificates and the verification pages.
Any Custom Certificates that are using thedateelement and selected the expiry dates will automatically be upgraded to use this new element (#499).
- Major issue with the pop-up window not working in the reposition element page (#483).
- Non-editing teachers being able to see the download certificate link for a user which took them to a blank page (#620).
- Added 'Save and Continue' option when editing image elements (#613). This means you can add an image to the filemanager, click 'Save and Continue' and then select it in the drop-down avoiding renavigating to the edit element page.
- Added monologo image (#568).
- Fixed issue when restoring
date,daterange,gradeitemnameandgradethat have been linked to a manual grade item (#582). - Removed unnecessary set_context() call causing a PHP notice to be shown (#443).
- Ensure we use $CFG->emailfromvia setting (#471).
- The downloaded issue report can now be ordered by date sanely (#602).
- Added ability to download all certificates in the report as a teacher and bulk download all site certificates as an administrator (#205).
- Do not make index unique (#601).
- Stopped PHP notice caused by the email certificate task (#443).
- Fixed undefined external_format_error in the mobile app (#565).
- Fixed being unable to reposition the course field element if it is empty (#579).
- Fixed incorrect name of mustache variable in email_certificate_html.mustache (#574).
- Fixed passing incorrect course module id value to \mod_customcert\output\email_certificate (#574).
- Delete the pages after deleting the elements otherwise it was breaking in element_deleted::create_from_element() (#571).
- Do not also show the 'My certificates' profile link when the user can not view the content of that page (#585).
- Added missing foreign key relationship for 'userid' in the 'customcert_issues' table (#537).
- Handle missing gradeitems as gracefully as possible, so we don't break the email task (#592).
- Fixed logic breaking the generation of the QR code URL (#545).
- Do not allow non-editing teachers to manage the certificate (#515).
- Ensure the 'verifyany' column length is valid on all sites (#597).
- Fixed events being triggered incorrectly (#570).
- Added the unique index 'userid-customcertid' to the 'customcert_issues' table (#537).
- Added events on the reposition element page (#599).
- Fix TCPDF error when viewing an example PDF from the manage templates page (#558).
- Fix images not displaying on the reposition element page (#562).
- Added new events (#518).
- An event for when an element is created.
- An event for when an element is updated.
- An event for when an element is deleted.
- An event for when a page is created.
- An event for when a page is updated.
- An event for when a page is deleted.
- An event for when a template is created.
- An event for when a template is updated.
- An event for when a template is deleted.
- Fix course settings error on single activity format (#544).
- Remove debugging message caused by the user field element listing the Skype field (#478).
- Fix deprecated usage of rendering primary buttons (#555).
- Fix usage of deprecated
cron_setup_userfunction (#547). - Fix broken webservice functions used by the mobile app.
- You can now optionally force the language of a certificate (#532).
- Fix problem repositioning elements (#500).
- Fix problem repositioning elements (#513).
- Fixed title and description shown twice (#521).
- Fix places not using the multi-language filter (#433).
- Fix user IDs in the issue table not being mapped during restore (#449).
- Fix emails displaying HTML entities encoded (#457).
- Fix error message when we have custom profile fields (#465).
- Respect multiple languages in manage template page title (#467).
- Add field exist check for alignment field in upgrade script to prevent upgrades from dying.
- Stop using deprecated pipe coreToLocaleString.
- User breadcrumbs on the my_certificates.php page changes when a course is specified (#469).
- You can now choose the course short or full name to display (#415).
- You can now select the alignment for all text elements (#121).
- Ability to add a relative date (#389).
- Usage of deprecated functions (#423)
- Usage of github actions (#407).
- The ability to show the description on the course page (#406).
- The ability to choose how to deliver the certificate (#401).
- Managers are now able to download their students' certificates (#412).
- Users being able to view the certificate before the required time set (#403).
- Fixed the issue with displaying PDF when debugging is ON (#420).
- Using incorrect context when sending emails (#402).
- Use
cron_setup_userwhen sending emails (#414).
- Added ability to select outcomes in the Grade element (#329).
- The Grade Item Name element now works with all grade items, whereas before it was just activities (#346).
- Added enrolment start and end dates to the date element (#328).
- Added username to userfield form element (#390).
- Removed unnecessary and confusing 'exampledata' string.
- Do not email those who can manage the certificate (#376).
- Do not force the PDF to be downloaded, instead send the file inline to the browser (#153).
- Updated the 'emailstudents_help', 'emailteachers_help' and 'emailothers_help' strings to warn users about prematurely emailing the certificate (#276).
- Do not email out certificates that contain no elements (#276).
- Certificates now get marked as viewed via the mobile app (#342).
- Custom fields not displaying properly (#359).
- Fix repositioning elements page when resizing the browser (#343).
- Prevent error when duplicate issues exist when using the code element (#363).
- Implemented get_objectid_mapping for the course_module_viewed.php event to avoid warning (#374).
- Fixed exception being thrown when loading a template that has an image element but no image selected (#369).
- Fixed issue with PDF being generated without a name (#333).
- Added extra Behat steps for new elements (#309).
- When copying a site template the site images are also copied to the course context and then those copied images are used. Before, the elements would simply point to the site images. However, this meant when performing a backup/restore the images were not stored in the backup file (#298).
- Fixed the displaying of names of a custom user field (#326).
- Do not allow '0' as a value for width or height in QR code (#321).
- Fixed foreign key violation (#331).
- Added subplugins.json file (#312).
- Re-added 'code' column to user report (#264).
- Add 'userfullname' variable for email subject (#316).
- Do not fail if multiple certificate issues (#304) and (#295).
- Added new custom course field element (#274).
- Added ability to specify the current date for date related elements (#289).
- String improvements for the 'Date range' element.
- Use negative numbers for constants in the 'Date range' element. The reason being that we may have a module that has an id matching one of these positive values. Sites which are using the 'Date range' element (sites which are not using this element do not have to do anything) will need to re-edit each element, select the date item again and save. An upgrade step was not created because it is impossible to tell if the site does actually want the constant or if they actually want the date for the module.
- Always send emails from the 'noreplyuser' (#165).
- Added QR code element (#146).
- Added Date range element (#185).
- Added the number of certificates issued above the report (#266).
- Added new capability to control who can be issued a certificate (#270).
- Failures when running unit tests for multiple activities (#282).
- Check that a certificate is valid before downloading on 'My certificates' page (#269).
- Make it clear what element values are just an example when previewing the PDF (#144).
- Missing implementation for privacy provider (#260).
- Use course module context when calling format_string/text (#200).
- Exception being thrown when adding the 'teachername' element to site template (#261).
- GDPR: Add support for removal of users from a context (see MDL-62560) (#252).
- Images can be made transparent (#186).
- Set default values of activity instance settings (#180).
- Allow element plugins to control if they can be added to a certificate (#225).
- Allow element plugins to have their own admin settings (#213).
- Added plaintext language variants for email bodies (#231).
- Added possibility to selectively disable activity instance settings (#179).
- Allow verification of deleted users (#159).
- The 'element' field in the 'customcert_elements' table has been changed from a Text field to varchar(255) (#241).
- The 'Completion date' option in the 'date' element is only displayed when completion is enabled (#160).
- Instead of assuming 2 decimal points for percentages, we now make use of the decimal value setting, which the
function
grade_format_gradevaluedoes by default if no decimal value is passed.
- Issue with scales not displaying correctly (#242).
- The report now respects the setting 'Show user identity' (#224).
- Removed incorrect course reset logic (#223).
- Description strings referring to the wrong setting (#254).
- Use custom fonts if present (#211).
- Fix broken SQL on Oracle in the email certificate task (#187).
- Fixed exception when clicking 'Add page' when template has not been saved (#154).
- Only email teachers who are enrolled within the course (#176).
- Only display teachers who are enrolled within the course in the dropdown (#171).
- Multiple UX improvements to both the browser and mobile views (#207).
- One big change here is combining the report and activity view page into one.
- Allow short dates with leading zeros (#210).
- Respect filters in the 'My certificates' and 'Verify certificate' pages (#197).
- Fixed reference to 'mod/certificate' capability.
- Multiple UX improvements to both the browser and mobile views (#203).
- Hotfix to prevent misalignment of 'text' elements after last release (#196).
- Mobile app support (#70).
This allows students to view the activity and download
their certificate. It also allows teachers to view the
list of issued certificates, with the ability to revoke
any.
This is for the soon-to-be released Moodle Mobile v3.5.0
(not to be confused with your Moodle site version) and
will not work on Mobile versions earlier than this.
If you are running a Moodle site on version 3.4 or below
you will need to install the local_mobile plugin in order
for this to work.
If you are running a Moodle site on version 3.0 or below
then you will need to upgrade.
- More font sizes (#148).
- Added new download icon.
This was done because the core 'import' icon was mapped
to the Font Awesome icon 'fa-level-up' which did not look
appropriate. So, a new icon was added and that was mapped
to the 'fa-download' icon.
- No longer display the 'action' column and user picture URL when downloading the user report (#192).
- Elements no longer ignore filters (#170).
- GDPR Compliance (#189).
- Race condition on certificate issues in scheduled task (#173).
- Ensure we backup the 'verifyany' setting (#169).
- Fixed encoding content links used by restore (#166).
- Added capability
mod/customcert:verifyallcertificatesthat provides a user with the ability to verify any certificate on the site by simply visiting themod/customcert/verify_certificate.phppage, rather than having to go to the verification link for each certificate. - Added site setting
customcert/verifyallcertificateswhich when enabled allows any person (including users not logged in) to be able to verify any certificate on the site, rather than having to go to the verification link for each certificate. However, this only applies to certificates whereAllow anyone to verify a certificatehas been set toYesin the certificate settings. - You can now display the grade and date of all grade items, not just the course and course activities.
- Text has been added above the
My certificateslist to explain that it contains certificates that have been issued to avoid confusion as to why certificates may not be appearing.
- The course full name is now used in emails.
- Added missing string used in course reset.
- New digital signature element (uses existing functionality in the TCPDF library).
- Ability to duplicate site templates via the manage templates page.
- Ability to delete issued certificates for individual users on the course report page.
- Removed usage of magic getter and abuse of
$this->element. The variable$this->elementwill still be accessible by any third-party element plugins, though this is discouraged and the appropriateget_xxx()method should be used instead. Using$this->elementindefinition_after_data()will no longer work. Please explicitly set the value of any custom fields you have in the form.
- Added missing
confirm_sesskey()checks. - Minor bug fixes.
- Added much needed Behat test coverage.
- Minor language string changes.
- Made changes to the UI when editing a certificate.
- Moved the 'Add element' submit button below the list of elements.
- Added icon next to the 'Delete page' link.
- Changed the 'Add page' button to a link, added an icon and moved it's location to the right.
- Do not make all submit buttons primary. MDL-59740 needs to be applied to your Moodle install in order to notice the change.
- Issue where the date an activity was graded was not displaying at all.
- Renamed the column 'size' in the table 'customcert_elements' to 'fontsize' due to 'size' being a reserved word in Oracle.