Skip to content

Commit

Permalink
Merge branch 'MDL-79194-403' of https://github.com/laurentdavid/moodle
Browse files Browse the repository at this point in the history
…into MOODLE_403_STABLE
  • Loading branch information
junpataleta committed Oct 27, 2023
2 parents 8bf31f2 + 81cdd8d commit 990b740
Show file tree
Hide file tree
Showing 7 changed files with 57 additions and 11 deletions.
2 changes: 1 addition & 1 deletion course/format/amd/build/local/content.min.js

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion course/format/amd/build/local/content.min.js.map

Large diffs are not rendered by default.

49 changes: 42 additions & 7 deletions course/format/amd/src/local/content.js
Expand Up @@ -238,6 +238,8 @@ export default class Component extends BaseComponent {
{watch: `transaction:start`, handler: this._startProcessing},
{watch: `course.sectionlist:updated`, handler: this._refreshCourseSectionlist},
{watch: `section.cmlist:updated`, handler: this._refreshSectionCmlist},
// Section visibility.
{watch: `section.visible:updated`, handler: this._reloadSection},
// Reindex sections and cms.
{watch: `state:updated`, handler: this._indexContents},
];
Expand Down Expand Up @@ -525,12 +527,12 @@ export default class Component extends BaseComponent {
if (debouncedReload) {
return debouncedReload;
}
const pendingReload = new Pending(pendingKey);
const reload = () => {
const pendingReload = new Pending(pendingKey);
this.debouncedReloads.delete(pendingKey);
const cmitem = this.getElement(this.selectors.CM, cmId);
if (!cmitem) {
return;
return pendingReload.resolve();
}
const promise = Fragment.loadFragment(
'core_courseformat',
Expand All @@ -543,17 +545,45 @@ export default class Component extends BaseComponent {
}
);
promise.then((html, js) => {
// Other state change can reload the CM or the section before this one.
if (!document.contains(cmitem)) {
pendingReload.resolve();
return false;
}
Templates.replaceNode(cmitem, html, js);
this._indexContents();
pendingReload.resolve();
return;
}).catch();
return true;
}).catch(() => {
pendingReload.resolve();
});
return pendingReload;
};
debouncedReload = debounce(reload, 200);
debouncedReload = debounce(
reload,
200,
{
cancel: true, pending: true
}
);
this.debouncedReloads.set(pendingKey, debouncedReload);
return debouncedReload;
}

/**
* Cancel the active reload CM debounced function, if any.
* @param {Number} cmId
*/
_cancelDebouncedReloadCm(cmId) {
const pendingKey = `courseformat/content:reloadCm_${cmId}`;
const debouncedReload = this.debouncedReloads.get(pendingKey);
if (!debouncedReload) {
return;
}
debouncedReload.cancel();
this.debouncedReloads.delete(pendingKey);
}

/**
* Reload a course section contents.
*
Expand All @@ -567,6 +597,10 @@ export default class Component extends BaseComponent {
const pendingReload = new Pending(`courseformat/content:reloadSection_${element.id}`);
const sectionitem = this.getElement(this.selectors.SECTION, element.id);
if (sectionitem) {
// Cancel any pending reload because the section will reload cms too.
for (const cmId of element.cmlist) {
this._cancelDebouncedReloadCm(cmId);
}
const promise = Fragment.loadFragment(
'core_courseformat',
'section',
Expand All @@ -581,8 +615,9 @@ export default class Component extends BaseComponent {
Templates.replaceNode(sectionitem, html, js);
this._indexContents();
pendingReload.resolve();
return;
}).catch();
}).catch(() => {
pendingReload.resolve();
});
}
}

Expand Down
2 changes: 1 addition & 1 deletion lib/amd/build/utils.min.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 1 addition & 1 deletion lib/amd/build/utils.min.js.map

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

10 changes: 10 additions & 0 deletions lib/amd/src/utils.js
Expand Up @@ -87,13 +87,15 @@ const debounceMap = new Map();
* @param {Number} wait The number of milliseconds to wait after the final attempt to execute
* @param {Object} [options]
* @param {boolean} [options.pending=false] Whether to wrap the debounced method in a pending promise
* @param {boolean} [options.cancel=false] Whether to add a cancel method to the debounced function
* @return {Function}
*/
export const debounce = (
func,
wait,
{
pending = false,
cancel = false,
} = {},
) => {
let timeout = null;
Expand All @@ -119,6 +121,14 @@ export const debounce = (
}, wait);
};

if (cancel) {
returnedFunction.cancel = () => {
const pendingPromise = debounceMap.get(returnedFunction);
pendingPromise?.resolve();
clearTimeout(timeout);
};
}

return returnedFunction;
};

Expand Down
1 change: 1 addition & 0 deletions lib/upgrade.txt
Expand Up @@ -4,6 +4,7 @@ information provided here is intended especially for developers.
=== 4.3.1 ===

* Added modalform config object `moduleName` param that can be used to define alternative modal type for the modalform. By default 'core/modal_save_cancel' is used.
* Add a new parameter to the debounce (core/utils) function to allow for cancellation.

=== 4.3 ===

Expand Down

0 comments on commit 990b740

Please sign in to comment.