mirror of
https://github.com/moodle/moodle.git
synced 2025-08-04 16:36:37 +02:00
MDL-50346 core: Allow sub-directories in template names
This commit is contained in:
parent
414eca8923
commit
0261d1effa
15 changed files with 209 additions and 64 deletions
|
@ -1,2 +1,2 @@
|
||||||
define ("tool_templatelibrary/display",["jquery","core/ajax","core/log","core/notification","core/templates","core/config","core/str"],function(a,b,c,d,e,f,g){var h=function(a,b){if(!a){return!1}var c="@template "+b,d=0,e=[];e=a.match(/{{!([\s\S]*?)}}/g);if(null!==e){for(d=0;d<e.length;d++){var f=e[d],g=f.indexOf(c);if(-1!==g){var h=g+c.length+1;f=f.substr(h,f.length-2-h);return f}}}return!1},i=function(b,f,i){g.get_string("templateselected","tool_templatelibrary",b).done(function(b){a("[data-region=\"displaytemplateheader\"]").text(b)}).fail(d.exception);var j=h(f,b);if(!1===j){j=h(i,b)}if(j){f=j}a("[data-region=\"displaytemplatesource\"]").text(f);var k=f.match(/Example context \(json\):([\s\S]*)/),l=!1;if(k){var m=k[1].trim();try{l=a.parseJSON(m)}catch(a){c.debug("Could not parse json example context for template.");c.debug(a)}}if(l){e.render(b,l).done(function(b,c){e.replaceNodeContents(a("[data-region=\"displaytemplateexample\"]"),b,c)}).fail(d.exception)}else{g.get_string("templatehasnoexample","tool_templatelibrary").done(function(b){a("[data-region=\"displaytemplateexample\"]").text(b)}).fail(d.exception)}},j=function(c){var e=c.split("/"),g=e.shift(),h=e.shift(),j=b.call([{methodname:"core_output_load_template",args:{component:g,template:h,themename:f.theme,includecomments:!0}},{methodname:"tool_templatelibrary_load_canonical_template",args:{component:g,template:h}}],!0,!1);a.when.apply(a,j).done(function(a,b){i(c,a,b)}).fail(d.exception)};a("[data-region=\"list-templates\"]").on("click","[data-templatename]",function(b){var c=a(this).data("templatename");b.preventDefault();j(c)});return{}});
|
define ("tool_templatelibrary/display",["jquery","core/ajax","core/log","core/notification","core/templates","core/config","core/str"],function(a,b,c,d,e,f,g){var h=function(a,b){if(!a){return!1}var c="@template "+b,d=0,e=[];e=a.match(/{{!([\s\S]*?)}}/g);if(null!==e){for(d=0;d<e.length;d++){var f=e[d],g=f.indexOf(c);if(-1!==g){var h=g+c.length+1;f=f.substr(h,f.length-2-h);return f}}}return!1},i=function(b,f,i){g.get_string("templateselected","tool_templatelibrary",b).done(function(b){a("[data-region=\"displaytemplateheader\"]").text(b)}).fail(d.exception);var j=h(f,b);if(!1===j){j=h(i,b)}if(j){f=j}a("[data-region=\"displaytemplatesource\"]").text(f);var k=f.match(/Example context \(json\):([\s\S]*)/),l=!1;if(k){var m=k[1].trim();try{l=a.parseJSON(m)}catch(a){c.debug("Could not parse json example context for template.");c.debug(a)}}if(l){e.render(b,l).done(function(b,c){e.replaceNodeContents(a("[data-region=\"displaytemplateexample\"]"),b,c)}).fail(d.exception)}else{g.get_string("templatehasnoexample","tool_templatelibrary").done(function(b){a("[data-region=\"displaytemplateexample\"]").text(b)}).fail(d.exception)}},j=function(c){var e=c.split("/"),g=e.shift(),h=e.join("/"),j=b.call([{methodname:"core_output_load_template",args:{component:g,template:h,themename:f.theme,includecomments:!0}},{methodname:"tool_templatelibrary_load_canonical_template",args:{component:g,template:h}}],!0,!1);a.when.apply(a,j).done(function(a,b){i(c,a,b)}).fail(d.exception)};a("[data-region=\"list-templates\"]").on("click","[data-templatename]",function(b){var c=a(this).data("templatename");b.preventDefault();j(c)});return{}});
|
||||||
//# sourceMappingURL=display.min.js.map
|
//# sourceMappingURL=display.min.js.map
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -119,7 +119,7 @@ define(['jquery', 'core/ajax', 'core/log', 'core/notification', 'core/templates'
|
||||||
var loadTemplate = function(templateName) {
|
var loadTemplate = function(templateName) {
|
||||||
var parts = templateName.split('/');
|
var parts = templateName.split('/');
|
||||||
var component = parts.shift();
|
var component = parts.shift();
|
||||||
var name = parts.shift();
|
var name = parts.join('/');
|
||||||
|
|
||||||
var promises = ajax.call([{
|
var promises = ajax.call([{
|
||||||
methodname: 'core_output_load_template',
|
methodname: 'core_output_load_template',
|
||||||
|
|
|
@ -99,13 +99,34 @@ class api {
|
||||||
|
|
||||||
foreach ($templatedirs as $templatecomponent => $dirs) {
|
foreach ($templatedirs as $templatecomponent => $dirs) {
|
||||||
foreach ($dirs as $dir) {
|
foreach ($dirs as $dir) {
|
||||||
|
if (!is_dir($dir) || !is_readable($dir)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$dir = realpath($dir);
|
||||||
|
|
||||||
// List it.
|
// List it.
|
||||||
$files = glob($dir . '/*.mustache');
|
$directory = new \RecursiveDirectoryIterator($dir);
|
||||||
|
$files = new \RecursiveIteratorIterator($directory);
|
||||||
|
|
||||||
foreach ($files as $file) {
|
foreach ($files as $file) {
|
||||||
$templatename = basename($file, '.mustache');
|
if (!$file->isFile()) {
|
||||||
if ($search == '' || strpos($templatename, $search) !== false) {
|
continue;
|
||||||
$results[$templatecomponent . '/' . $templatename] = 1;
|
}
|
||||||
|
$filename = substr($file->getRealpath(), strlen($dir) + 1);
|
||||||
|
if (strpos($templatecomponent, 'theme_') === 0) {
|
||||||
|
if (strpos($filename, '/') !== false && strpos($filename, 'local/') !== 0) {
|
||||||
|
// Skip any template in a sub-directory of a theme which is not in a local directory.
|
||||||
|
// These are theme overrides of core templates.
|
||||||
|
// Note: There is a rare edge case where a theme may override a template and then have additional
|
||||||
|
// dependant templates and these will not be shown.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$templatename = str_replace('.mustache', '', $filename);
|
||||||
|
$componenttemplatename = "{$templatecomponent}/{$templatename}";
|
||||||
|
|
||||||
|
if ($search == '' || strpos($componenttemplatename, $search) !== false) {
|
||||||
|
$results[$componenttemplatename] = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -152,5 +173,4 @@ class api {
|
||||||
return $templatestr;
|
return $templatestr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -105,7 +105,7 @@ class external extends external_api {
|
||||||
public static function load_canonical_template_parameters() {
|
public static function load_canonical_template_parameters() {
|
||||||
return new external_function_parameters(
|
return new external_function_parameters(
|
||||||
array('component' => new external_value(PARAM_COMPONENT, 'component containing the template'),
|
array('component' => new external_value(PARAM_COMPONENT, 'component containing the template'),
|
||||||
'template' => new external_value(PARAM_ALPHANUMEXT, 'name of the template'))
|
'template' => new external_value(PARAM_SAFEPATH, 'name of the template'))
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,6 +21,6 @@
|
||||||
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
|
||||||
*/
|
*/
|
||||||
defined('MOODLE_INTERNAL') || die();
|
defined('MOODLE_INTERNAL') || die();
|
||||||
$plugin->version = 2019052000; // The current plugin version (Date: YYYYMMDDXX).
|
$plugin->version = 2019052002; // The current plugin version (Date: YYYYMMDDXX).
|
||||||
$plugin->requires = 2019051100; // Requires this Moodle version.
|
$plugin->requires = 2019051100; // Requires this Moodle version.
|
||||||
$plugin->component = 'tool_templatelibrary'; // Full name of the plugin (used for diagnostics).
|
$plugin->component = 'tool_templatelibrary'; // Full name of the plugin (used for diagnostics).
|
||||||
|
|
2
lib/amd/build/templates.min.js
vendored
2
lib/amd/build/templates.min.js
vendored
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
|
@ -304,7 +304,7 @@ define([
|
||||||
// to be loaded.
|
// to be loaded.
|
||||||
var parts = templateName.split('/');
|
var parts = templateName.split('/');
|
||||||
var component = parts.shift();
|
var component = parts.shift();
|
||||||
var name = parts.shift();
|
var name = parts.join('/');
|
||||||
var deferred = $.Deferred();
|
var deferred = $.Deferred();
|
||||||
|
|
||||||
// Add this template to the buffer to be loaded.
|
// Add this template to the buffer to be loaded.
|
||||||
|
|
|
@ -51,7 +51,7 @@ class external extends external_api {
|
||||||
public static function load_template_parameters() {
|
public static function load_template_parameters() {
|
||||||
return new external_function_parameters(
|
return new external_function_parameters(
|
||||||
array('component' => new external_value(PARAM_COMPONENT, 'component containing the template'),
|
array('component' => new external_value(PARAM_COMPONENT, 'component containing the template'),
|
||||||
'template' => new external_value(PARAM_ALPHANUMEXT, 'name of the template'),
|
'template' => new external_value(PARAM_SAFEPATH, 'name of the template'),
|
||||||
'themename' => new external_value(PARAM_ALPHANUMEXT, 'The current theme.'),
|
'themename' => new external_value(PARAM_ALPHANUMEXT, 'The current theme.'),
|
||||||
'includecomments' => new external_value(PARAM_BOOL, 'Include comments or not', VALUE_DEFAULT, false)
|
'includecomments' => new external_value(PARAM_BOOL, 'Include comments or not', VALUE_DEFAULT, false)
|
||||||
)
|
)
|
||||||
|
@ -102,7 +102,7 @@ class external extends external_api {
|
||||||
public static function load_template_with_dependencies_parameters() {
|
public static function load_template_with_dependencies_parameters() {
|
||||||
return new external_function_parameters([
|
return new external_function_parameters([
|
||||||
'component' => new external_value(PARAM_COMPONENT, 'component containing the template'),
|
'component' => new external_value(PARAM_COMPONENT, 'component containing the template'),
|
||||||
'template' => new external_value(PARAM_ALPHANUMEXT, 'name of the template'),
|
'template' => new external_value(PARAM_SAFEPATH, 'name of the template'),
|
||||||
'themename' => new external_value(PARAM_ALPHANUMEXT, 'The current theme.'),
|
'themename' => new external_value(PARAM_ALPHANUMEXT, 'The current theme.'),
|
||||||
'includecomments' => new external_value(PARAM_BOOL, 'Include comments or not', VALUE_DEFAULT, false),
|
'includecomments' => new external_value(PARAM_BOOL, 'Include comments or not', VALUE_DEFAULT, false),
|
||||||
'lang' => new external_value(PARAM_LANG, 'lang', VALUE_DEFAULT, null),
|
'lang' => new external_value(PARAM_LANG, 'lang', VALUE_DEFAULT, null),
|
||||||
|
|
|
@ -107,11 +107,9 @@ class mustache_template_finder {
|
||||||
throw new coding_exception('Templates names must be specified as "componentname/templatename"' .
|
throw new coding_exception('Templates names must be specified as "componentname/templatename"' .
|
||||||
' (' . s($name) . ' requested) ');
|
' (' . s($name) . ' requested) ');
|
||||||
}
|
}
|
||||||
|
|
||||||
list($component, $templatename) = explode('/', $name, 2);
|
list($component, $templatename) = explode('/', $name, 2);
|
||||||
$component = clean_param($component, PARAM_COMPONENT);
|
$component = clean_param($component, PARAM_COMPONENT);
|
||||||
if (strpos($templatename, '/') !== false) {
|
|
||||||
throw new coding_exception('Templates cannot be placed in sub directories (' . s($name) . ' requested)');
|
|
||||||
}
|
|
||||||
|
|
||||||
$dirs = self::get_template_directories_for_component($component, $themename);
|
$dirs = self::get_template_directories_for_component($component, $themename);
|
||||||
|
|
||||||
|
|
|
@ -33,66 +33,168 @@ use core\output\mustache_template_finder;
|
||||||
*/
|
*/
|
||||||
class core_output_mustache_template_finder_testcase extends advanced_testcase {
|
class core_output_mustache_template_finder_testcase extends advanced_testcase {
|
||||||
|
|
||||||
public function test_get_template_directories_for_component() {
|
/**
|
||||||
global $CFG;
|
* Data provider which reutrns a set of valid template directories to be used when testing
|
||||||
|
* get_template_directories_for_component.
|
||||||
// Test a plugin.
|
*
|
||||||
$dirs = mustache_template_finder::get_template_directories_for_component('mod_assign', 'classic');
|
* @return array
|
||||||
|
*/
|
||||||
$correct = array(
|
public function valid_template_directories_provider(): array {
|
||||||
|
return [
|
||||||
|
'plugin: mod_assign' => [
|
||||||
|
'component' => 'mod_assign',
|
||||||
|
'theme' => '',
|
||||||
|
'paths' => [
|
||||||
|
'theme/boost/templates/mod_assign/',
|
||||||
|
'mod/assign/templates/'
|
||||||
|
],
|
||||||
|
],
|
||||||
|
'plugin: mod_assign with classic' => [
|
||||||
|
'component' => 'mod_assign',
|
||||||
|
'theme' => 'classic',
|
||||||
|
'paths' => [
|
||||||
'theme/classic/templates/mod_assign/',
|
'theme/classic/templates/mod_assign/',
|
||||||
'theme/boost/templates/mod_assign/',
|
'theme/boost/templates/mod_assign/',
|
||||||
'mod/assign/templates/'
|
'mod/assign/templates/'
|
||||||
);
|
],
|
||||||
foreach ($dirs as $index => $dir) {
|
],
|
||||||
$this->assertSame($dir, $CFG->dirroot . '/' . $correct[$index]);
|
'subsystem: core_user' => [
|
||||||
}
|
'component' => 'core_user',
|
||||||
// Test a subsystem.
|
'theme' => 'classic',
|
||||||
$dirs = mustache_template_finder::get_template_directories_for_component('core_user', 'classic');
|
'paths' => [
|
||||||
|
|
||||||
$correct = array(
|
|
||||||
'theme/classic/templates/core_user/',
|
'theme/classic/templates/core_user/',
|
||||||
'theme/boost/templates/core_user/',
|
'theme/boost/templates/core_user/',
|
||||||
'user/templates/'
|
'user/templates/'
|
||||||
);
|
],
|
||||||
foreach ($dirs as $index => $dir) {
|
],
|
||||||
$this->assertSame($dir, $CFG->dirroot . '/' . $correct[$index]);
|
'core' => [
|
||||||
}
|
'component' => 'core',
|
||||||
// Test core.
|
'theme' => 'classic',
|
||||||
$dirs = mustache_template_finder::get_template_directories_for_component('core', 'classic');
|
'paths' => [
|
||||||
|
|
||||||
$correct = array(
|
|
||||||
'theme/classic/templates/core/',
|
'theme/classic/templates/core/',
|
||||||
'theme/boost/templates/core/',
|
'theme/boost/templates/core/',
|
||||||
'lib/templates/'
|
'lib/templates/'
|
||||||
);
|
],
|
||||||
foreach ($dirs as $index => $dir) {
|
],
|
||||||
$this->assertSame($dir, $CFG->dirroot . '/' . $correct[$index]);
|
];
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @expectedException coding_exception
|
* Tests for get_template_directories_for_component.
|
||||||
|
*
|
||||||
|
* @dataProvider valid_template_directories_provider
|
||||||
|
* @param string $component
|
||||||
|
* @param string $theme
|
||||||
|
* @param array $paths
|
||||||
*/
|
*/
|
||||||
public function test_invalid_get_template_directories_for_component() {
|
public function test_get_template_directories_for_component(string $component, string $theme, array $paths): void {
|
||||||
// Test something invalid.
|
|
||||||
$dirs = mustache_template_finder::get_template_directories_for_component('octopus', 'classic');
|
|
||||||
}
|
|
||||||
|
|
||||||
public function test_get_template_filepath() {
|
|
||||||
global $CFG;
|
global $CFG;
|
||||||
|
|
||||||
$filename = mustache_template_finder::get_template_filepath('core/pix_icon', 'classic');
|
// Test a plugin.
|
||||||
$correct = $CFG->dirroot . '/lib/templates/pix_icon.mustache';
|
$dirs = mustache_template_finder::get_template_directories_for_component($component, $theme, $paths);
|
||||||
$this->assertSame($correct, $filename);
|
|
||||||
|
$correct = array_map(function($path) use ($CFG) {
|
||||||
|
return implode('/', [$CFG->dirroot, $path]);
|
||||||
|
}, $paths);
|
||||||
|
|
||||||
|
$this->assertEquals($correct, $dirs);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* Tests for get_template_directories_for_component when dealing with an invalid component.
|
||||||
|
*
|
||||||
|
* @expectedException coding_exception
|
||||||
|
*/
|
||||||
|
public function test_invalid_component_get_template_directories_for_component() {
|
||||||
|
// Test something invalid.
|
||||||
|
mustache_template_finder::get_template_directories_for_component('octopus', 'classic');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Data provider which reutrns a set of valid template directories to be used when testing
|
||||||
|
* get_template_directories_for_component.
|
||||||
|
*
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public function valid_template_filepath_provider(): array {
|
||||||
|
return [
|
||||||
|
'Standard core template' => [
|
||||||
|
'template' => 'core/modal',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'lib/templates/modal.mustache',
|
||||||
|
],
|
||||||
|
'Template overridden by theme' => [
|
||||||
|
'template' => 'core_form/element-float-inline',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'theme/boost/templates/core_form/element-float-inline.mustache',
|
||||||
|
],
|
||||||
|
'Template overridden by theme but child theme selected' => [
|
||||||
|
'template' => 'core_form/element-float-inline',
|
||||||
|
'theme' => 'classic',
|
||||||
|
'location' => 'theme/boost/templates/core_form/element-float-inline.mustache',
|
||||||
|
],
|
||||||
|
'Template overridden by child theme' => [
|
||||||
|
'template' => 'core/full_header',
|
||||||
|
'theme' => 'classic',
|
||||||
|
'location' => 'theme/classic/templates/core/full_header.mustache',
|
||||||
|
],
|
||||||
|
'Template overridden by child theme but tested against defualt theme' => [
|
||||||
|
'template' => 'core/full_header',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'lib/templates/full_header.mustache',
|
||||||
|
],
|
||||||
|
'Standard plugin template' => [
|
||||||
|
'template' => 'mod_assign/grading_panel',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'mod/assign/templates/grading_panel.mustache',
|
||||||
|
],
|
||||||
|
'Subsystem template' => [
|
||||||
|
'template' => 'core_user/status_details',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'user/templates/status_details.mustache',
|
||||||
|
],
|
||||||
|
'Theme own template' => [
|
||||||
|
'template' => 'theme_classic/columns',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'theme/classic/templates/columns.mustache',
|
||||||
|
],
|
||||||
|
'Theme overridden template against that theme' => [
|
||||||
|
'template' => 'theme_classic/navbar',
|
||||||
|
'theme' => 'classic',
|
||||||
|
'location' => 'theme/classic/templates/navbar.mustache',
|
||||||
|
],
|
||||||
|
// Note: This one looks strange but is correct. It is legitimate to request theme's component template in
|
||||||
|
// the context of another theme. For example, this is used by child themes making use of parent theme
|
||||||
|
// templates.
|
||||||
|
'Theme overridden template against the default theme' => [
|
||||||
|
'template' => 'theme_classic/navbar',
|
||||||
|
'theme' => '',
|
||||||
|
'location' => 'theme/classic/templates/navbar.mustache',
|
||||||
|
],
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for get_template_filepath.
|
||||||
|
*
|
||||||
|
* @dataProvider valid_template_filepath_provider
|
||||||
|
* @param string $template
|
||||||
|
* @param string $theme
|
||||||
|
* @param string $location
|
||||||
|
*/
|
||||||
|
public function test_get_template_filepath(string $template, string $theme, string $location) {
|
||||||
|
global $CFG;
|
||||||
|
|
||||||
|
$filename = mustache_template_finder::get_template_filepath($template, $theme);
|
||||||
|
$this->assertEquals("{$CFG->dirroot}/{$location}", $filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for get_template_filepath when dealing with an invalid component.
|
||||||
|
*
|
||||||
* @expectedException moodle_exception
|
* @expectedException moodle_exception
|
||||||
*/
|
*/
|
||||||
public function test_invalid_get_template_filepath() {
|
public function test_invalid_component_get_template_filepath() {
|
||||||
// Test something invalid.
|
mustache_template_finder::get_template_filepath('core/octopus', 'classic');
|
||||||
$dirs = mustache_template_finder::get_template_filepath('core/octopus', 'classic');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,7 +39,17 @@ information provided here is intended especially for developers.
|
||||||
mod_forum/views/post: mod/forum/amd/src/views/post
|
mod_forum/views/post: mod/forum/amd/src/views/post
|
||||||
* The 'xxxx_check_password_policy' method now has an extra parameter: $user. It contains the user object to perform password
|
* The 'xxxx_check_password_policy' method now has an extra parameter: $user. It contains the user object to perform password
|
||||||
validation against and defaults to null (so, no user needed) if not provided.
|
validation against and defaults to null (so, no user needed) if not provided.
|
||||||
|
* It is now possible to use sub-directories when creating mustache templates.
|
||||||
|
The standard rules for Level 2 namespaces also apply to templates.
|
||||||
|
The sub-directory used must be either an valid component, or placed inside a 'local' directory to ensure that it does not conflict with other components.
|
||||||
|
|
||||||
|
The following are all valid template names and locations in your plugin:
|
||||||
|
mod_forum/forum_post: mod/forum/templates/forum_post.mustache
|
||||||
|
mod_forum/local/post/user: mod/forum/templates/local/post/user.mustache
|
||||||
|
mod_forum/form/checkbox_toggle: mod/forum/templates/form/checkbox_toggle.mustache
|
||||||
|
|
||||||
|
The following are _invalid_ template names and locations:
|
||||||
|
mod_forum/post/user: mod/forum/templates/local/post/user.mustache
|
||||||
|
|
||||||
=== 3.7 ===
|
=== 3.7 ===
|
||||||
|
|
||||||
|
|
|
@ -5,6 +5,21 @@ information provided here is intended especially for theme designer.
|
||||||
|
|
||||||
* The PHP Less compilier has now been removed from the core library.
|
* The PHP Less compilier has now been removed from the core library.
|
||||||
Please consider migrating your theme to use SCSS.
|
Please consider migrating your theme to use SCSS.
|
||||||
|
* It is now possible to use sub-directories when creating mustache templates.
|
||||||
|
The standard rules for Level 2 namespaces also apply to templates.
|
||||||
|
The sub-directory used must be either an valid component, or placed inside a 'local' directory to ensure that it does not conflict with other components.
|
||||||
|
|
||||||
|
The following are all valid template names and locations in your theme:
|
||||||
|
theme_themename/columns2: theme/[themename]/columns2.mustache
|
||||||
|
theme_themename/local/layouts/columns2: theme/[themename]/local/layouts/columns2.mustache
|
||||||
|
|
||||||
|
The following are core templates, locations, and override locations in your theme:
|
||||||
|
core/modal: lib/templates/modal.mustache => theme/[themename]/core/modal.mustache
|
||||||
|
mod_forum/forum_post: mod/forum/templates/forum_post.mustache => theme/[themename]/mod_forum/forum_post.mustache
|
||||||
|
mod_forum/local/post/user: mod/forum/templates/local/post/user.mustache => theme/[themename]/mod_forum/local/post/user.mustache
|
||||||
|
|
||||||
|
The following are _invalid_ template names and locations:
|
||||||
|
theme_themename/layouts/columns2: theme/[themename]/layouts/columns2.mustache
|
||||||
|
|
||||||
=== 3.7 ===
|
=== 3.7 ===
|
||||||
* The core/form_autocompelte_input template now has a `data-tags` attribute.
|
* The core/form_autocompelte_input template now has a `data-tags` attribute.
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
|
|
||||||
defined('MOODLE_INTERNAL') || die();
|
defined('MOODLE_INTERNAL') || die();
|
||||||
|
|
||||||
$version = 2019080100.00; // YYYYMMDD = weekly release date of this DEV branch.
|
$version = 2019080700.00; // YYYYMMDD = weekly release date of this DEV branch.
|
||||||
// RR = release increments - 00 in DEV branches.
|
// RR = release increments - 00 in DEV branches.
|
||||||
// .XX = incremental changes.
|
// .XX = incremental changes.
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue