From 4e5af8eb9b916de0324cc9b85999cac220be1f87 Mon Sep 17 00:00:00 2001 From: Dani Palou Date: Tue, 2 Jul 2024 11:58:05 +0200 Subject: [PATCH 1/4] Mobile app: Create ionic5 template duplicating latest --- classes/output/mobile.php | 6 +- templates/mobile_view_page_ionic5.mustache | 215 +++++++++++++++++++++ 2 files changed, 218 insertions(+), 3 deletions(-) create mode 100644 templates/mobile_view_page_ionic5.mustache diff --git a/classes/output/mobile.php b/classes/output/mobile.php index df9b42b..cdde087 100644 --- a/classes/output/mobile.php +++ b/classes/output/mobile.php @@ -66,7 +66,7 @@ public static function mobile_course_view($args) { $args = (object) $args; - $foldername = $args->appversioncode >= 3950 ? 'latest' : 'ionic3'; + $versionname = $args->appversioncode >= 44000 ? 'latest' : 'ionic5'; $cm = get_coursemodule_from_id('choicegroup', $args->cmid); $course = $DB->get_record('course', ['id' => $cm->course]); @@ -142,10 +142,10 @@ public static function mobile_course_view($args) { 'templates' => [ [ 'id' => 'main', - 'html' => $OUTPUT->render_from_template("mod_choicegroup/mobile_view_page_$foldername", $data), + 'html' => $OUTPUT->render_from_template("mod_choicegroup/mobile_view_page_$versionname", $data), ], ], - 'javascript' => file_get_contents($CFG->dirroot . "/mod/choicegroup/mobile/js/$foldername/courseview.js"), + 'javascript' => file_get_contents($CFG->dirroot . "/mod/choicegroup/mobile/js/latest/courseview.js"), 'otherdata' => [ 'data' => json_encode($responses), 'allowupdate' => $choicegroup->allowupdate ? 1 : 0, diff --git a/templates/mobile_view_page_ionic5.mustache b/templates/mobile_view_page_ionic5.mustache new file mode 100644 index 0000000..d92e5df --- /dev/null +++ b/templates/mobile_view_page_ionic5.mustache @@ -0,0 +1,215 @@ +{{! + This file is part of Moodle - http://moodle.org/ + + Moodle is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + Moodle is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with Moodle. If not, see . +}} +{{! + @template mod_choicegroup/mobile_view_page + + Template for the mobile view page. + + Classes required for JS: + - + + Data attributes required for JS: + - + + Context variables required for this template: + * cmid + * courseid + * choicegroup + * options + + Example context (json): + { + "cmid": "62", + "courseid": "3", + "choicegroup": { + "id": "4", + "course": "3", + "name": "Group choice activity", + "intro": "

Select your group

", + "introformat": "1", + "publish": "1", + "multipleenrollmentspossible": "1", + "showresults": "3", + "display": "0", + "allowupdate": "0", + "showunanswered": "0", + "limitanswers": "1", + "timeopen": "0", + "timeclose": "0", + "timemodified": "1528114222", + "completionsubmit": "0", + "sortgroupsby": "0", + "option": { + "10": "3", + "11": "4", + "12": "5" + }, + "grpmemberid": { + "6": [ + "3", + "2" + ] + }, + "maxanswers": { + "12": "1", + "11": "2", + "10": "2" + }, + "open": true, + "expired": false, + "alloptionsdisabled": false + }, + "options": [ + { + "id": 10, + "groupid": "3", + "name": "Group 1", + "maxanswers": "1", + "displaylayout": "0", + "countanswers": 2, + "checked": false, + "disabled": true + }, + { + "id": 11, + "groupid": "4", + "name": "Group 2", + "maxanswers": "2", + "displaylayout": "0", + "countanswers": 1, + "checked": true, + "disabled": false + }, + { + "id": 12, + "groupid": "5", + "name": "Group 3", + "maxanswers": "2", + "displaylayout": "0", + "countanswers": 0, + "checked": false, + "disabled": false + } + ] + } +}} +{{=<% %>=}} + + + + + + + + + + + + + + + + + {{ 'core.hasdatatosync' | translate:{$a: moduleName} }} + + + + <%# choicegroup.message %> + + + <% choicegroup.message %> + + + <%/ choicegroup.message %> + + <%# choicegroup.open %> +
+ + + + + {{ 'plugin.mod_choicegroup.group' | translate }} + + + <%^ choicegroup.limitanswers %> + {{ 'plugin.mod_choicegroup.members/' | translate }} + <%/ choicegroup.limitanswers %> + <%# choicegroup.limitanswers %> + {{ 'plugin.mod_choicegroup.members/max' | translate }} + <%/ choicegroup.limitanswers %> + + + + + <%^ choicegroup.multipleenrollmentspossible %> + + <%/ choicegroup.multipleenrollmentspossible %> + <%# options %> + + + + <% name %> + <%^ choicegroup.multipleenrollmentspossible %> + checked="true"<%/ checked %> <%# disabled %>disabled="true"<%/ disabled %> [value]="<% id %>"> + <%/ choicegroup.multipleenrollmentspossible %> + <%# choicegroup.multipleenrollmentspossible %> + checked="true"<%/ checked %> + <%# disabled %>disabled="true"<%/ disabled %> + [value]="<% id %>"> + + <%/ choicegroup.multipleenrollmentspossible %> + + + + + <% countanswers %> + <%# choicegroup.limitanswers %> / <% maxanswers %> <%/ choicegroup.limitanswers %> + + + <%/ options %> + <%^ choicegroup.multipleenrollmentspossible %> + + <%/ choicegroup.multipleenrollmentspossible %> + + + + <%^ choicegroup.expired %> + <%^ choicegroup.alloptionsdisabled %> + + + {{ 'plugin.mod_choicegroup.savemychoicegroup' | translate }} + + + <%^ choicegroup.multipleenrollmentspossible %> + <%# choicegroup.allowupdate %> + + + {{ 'plugin.mod_choicegroup.removemychoicegroup' | translate }} + + <%/ choicegroup.allowupdate %> + <%/ choicegroup.multipleenrollmentspossible %> + + <%/ choicegroup.alloptionsdisabled %> + <%/ choicegroup.expired %> +
+ + + + <%/ choicegroup.open %> +
From 24f039c2d64db1e06d1ee6fc613e343d40d6b756 Mon Sep 17 00:00:00 2001 From: Dani Palou Date: Tue, 2 Jul 2024 11:59:25 +0200 Subject: [PATCH 2/4] Mobile app: Remove ionic3 template and JS code --- classes/output/mobile.php | 4 +- mobile/js/ionic3/courseview.js | 240 --------- mobile/js/ionic3/init.js | 600 --------------------- templates/mobile_view_page_ionic3.mustache | 215 -------- 4 files changed, 1 insertion(+), 1058 deletions(-) delete mode 100644 mobile/js/ionic3/courseview.js delete mode 100644 mobile/js/ionic3/init.js delete mode 100644 templates/mobile_view_page_ionic3.mustache diff --git a/classes/output/mobile.php b/classes/output/mobile.php index cdde087..2475b8c 100644 --- a/classes/output/mobile.php +++ b/classes/output/mobile.php @@ -47,11 +47,9 @@ public static function mobile_init($args) { $args = (object) $args; - $foldername = $args->appversioncode >= 3950 ? 'latest' : 'ionic3'; - return [ 'templates' => [], - 'javascript' => file_get_contents($CFG->dirroot . "/mod/choicegroup/mobile/js/$foldername/init.js"), + 'javascript' => file_get_contents($CFG->dirroot . "/mod/choicegroup/mobile/js/latest/init.js"), ]; } diff --git a/mobile/js/ionic3/courseview.js b/mobile/js/ionic3/courseview.js deleted file mode 100644 index b87fddc..0000000 --- a/mobile/js/ionic3/courseview.js +++ /dev/null @@ -1,240 +0,0 @@ -// This file is part of Moodle - http://moodle.org/ -// -// Moodle is free software: you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. -// -// Moodle is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with Moodle. If not, see . - -/** - * This file is part of the Moodle apps support for the choicegroup plugin. - * Defines the function to be used from the mobile course view template. - * - * @copyright 2019 Dani Palou - * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later - */ - -var that = this; -var allowOffline = this.CoreConfigConstants.versioncode > 3800; // In 3.8.0 and older plugins couldn't add DB schemas. -var multipleEnrol = this.CONTENT_OTHERDATA.multipleenrollmentspossible; - -if (Array.isArray(this.CONTENT_OTHERDATA.data) && this.CONTENT_OTHERDATA.data.length == 0) { - // When there are no responses we receive an empty array instead of an empty object. Fix it. - this.CONTENT_OTHERDATA.data = {}; -} - -var originalData = this.CoreUtilsProvider.clone(this.CONTENT_OTHERDATA.data); - -/** - * Send responses to the site. - */ -this.submitResponses = function() { - var promise; - - if (!that.CONTENT_OTHERDATA.allowupdate) { - // Ask the user to confirm. - that.CoreDomUtilsProvider.showConfirm(that.TranslateService.instant('core.areyousure')); - } else { - // No need to confirm. - promise = Promise.resolve(); - } - - promise.then(function() { - // Submit the responses now. - var modal = that.CoreDomUtilsProvider.showModalLoading('core.sending', true); - var data = that.CoreUtilsProvider.objectToArrayOfObjects(that.CONTENT_OTHERDATA.data, 'name', 'value'); - - if (multipleEnrol) { - // In multiple enrol, the WS expects to receive 'true' as a string instead of 1 or 0. - data.forEach(function(entry) { - entry.value = String(entry.value); - }); - } - - that.choiceGroupProvider.submitResponses(that.module.instance, that.module.name, that.courseId, that.module.id, data, - allowOffline).then(function(online) { - - // Responses have been sent to server or stored to be sent later. - that.CoreDomUtilsProvider.showToast(that.TranslateService.instant('plugin.mod_choicegroup.choicegroupsaved')); - - if (online) { - // Check completion since it could be configured to complete once the user answers the choice. - that.CoreCourseProvider.checkModuleCompletion(that.courseId, that.module.completiondata); - - // Data has been sent, refresh the content. - return that.refreshContent(true); - } else { - // Data stored in offline. - return that.loadOfflineData(); - } - - }).catch((message) => { - that.CoreDomUtilsProvider.showErrorModalDefault(message, 'Error submitting responses.', true); - }).finally(() => { - modal.dismiss(); - }); - }).catch(() => { - // User cancelled, ignore. - }); -}; - -/** - * Delete the responses. Only if multiple enrol is not allowed. - */ -this.deleteResponses = function() { - var modal = that.CoreDomUtilsProvider.showModalLoading('core.sending', true); - - that.choiceGroupProvider.deleteResponses(that.module.instance, that.module.name, that.courseId, that.module.id, allowOffline) - .then(function(online) { - - // Responses have been sent to server or stored to be sent later. - that.CoreDomUtilsProvider.showToast(that.TranslateService.instant('plugin.mod_choicegroup.choicegroupsaved')); - - if (online) { - // Data has been sent, refresh the content. - return that.refreshContent(true); - } else { - // Data stored in offline. - return that.loadOfflineData(); - } - - }).catch((message) => { - that.CoreDomUtilsProvider.showErrorModalDefault(message, 'Error deleting responses.', true); - }).finally(() => { - modal.dismiss(); - }); -}; - -/** - * Check if the activity has offline data to be sent. - * - * @return Promise resolved when done. - */ -this.loadOfflineData = function() { - // Get the offline response if it exists. - return that.choiceGroupOffline.getResponse(that.module.instance).then(function(response) { - that.hasOffline = true; - - if (response.deleting) { - // Uncheck selected option. Delete is only possible if there is no multiple enrolment. - delete that.CONTENT_OTHERDATA.data.responses; - that.showDelete = false; - } else { - // Load the offline options into the model. - that.CONTENT_OTHERDATA.data = {}; - - response.data.forEach(function(entry) { - that.CONTENT_OTHERDATA.data[entry.name] = entry.value; - }); - - that.showDelete = !multipleEnrol; // Show delete if there is offline data and is not multiple enrol. - } - }).catch(function() { - // Offline data not found. Use the original data. - that.hasOffline = false; - that.showDelete = that.CONTENT_OTHERDATA.answergiven; - that.CONTENT_OTHERDATA.data = that.CoreUtilsProvider.clone(originalData); - }); -} - -/** - * Tries to synchronize the activity. - * - * @param showErrors If show errors to the user of hide them. - * @param done Function to call when done. - * @return Promise resolved with true if sync succeed, or false if failed. - */ -this.synchronize = function(showErrors, done) { - that.refreshIcon = 'spinner'; - that.syncIcon = 'spinner'; - - // Try to synchronize the group choice. - return that.choiceGroupSync.syncChoiceGroup(that.module.instance).then(function(result) { - if (result.warnings && result.warnings.length) { - that.CoreDomUtilsProvider.showErrorModal(result.warnings[0]); - } - - return result.updated; - }).catch(function(error) { - if (showErrors) { - that.CoreDomUtilsProvider.showErrorModalDefault(error, 'core.errorsync', true); - } - - return false; - }).then(function(updated) { - if (updated) { - // Data has been sent, fetch the content (WS data has already been updated in the sync process). - return that.fetchContent(false); - } - - // Check if the group choice has offline data. - return that.loadOfflineData(); - }).finally(function() { - done && done(); - that.refreshIcon = 'refresh'; - that.syncIcon = 'sync'; - }); -}; - -/** - * Refresh data. - * - * @param done Function to call when done. - * @return Promise resolved when done. - */ -this.doRefresh = function(done) { - that.refreshIcon = 'spinner'; - that.syncIcon = 'spinner'; - - return that.refreshContent(false).finally(function() { - done && done(); - that.refreshIcon = 'refresh'; - that.syncIcon = 'sync'; - }); -}; - -this.moduleName = this.TranslateService.instant('plugin.mod_choicegroup.modulename'); -this.isOnline = this.CoreAppProvider.isOnline(); - -// Refresh online status when changes. -var onlineObserver = this.Network.onchange().subscribe(function() { - that.isOnline = that.CoreAppProvider.isOnline(); -}); - -var syncObserver; - -if (allowOffline) { - // Try to synchronize the choice. - this.synchronize(false).finally(function() { - that.loaded = true; - }); - - // Update the view if the group choice is synchronized automatically. - syncObserver = this.CoreEventsProvider.on(this.choiceGroupSync.AUTO_SYNCED, function(data) { - if (data.choiceGroupId == that.module.instance) { - // This group choice has been synchronized, fetch the content (WS data has already been updated in the sync process). - return that.fetchContent(false); - } - }, this.CoreSitesProvider.getCurrentSiteId()); -} else { - // No offline allowed, just display the data. - this.loaded = true; - that.refreshIcon = 'refresh'; - that.hasOffline = false; - that.showDelete = that.CONTENT_OTHERDATA.answergiven; -} - -/** - * Component being destroyed. - */ -this.ngOnDestroy = function() { - onlineObserver && onlineObserver.unsubscribe(); - syncObserver && syncObserver.off(); -}; diff --git a/mobile/js/ionic3/init.js b/mobile/js/ionic3/init.js deleted file mode 100644 index 30875bc..0000000 --- a/mobile/js/ionic3/init.js +++ /dev/null @@ -1,600 +0,0 @@ -// This file is part of Moodle - http://moodle.org/ -// -// Moodle is free software: you can redistribute it and/or modify -// it under the terms of the GNU General Public License as published by -// the Free Software Foundation, either version 3 of the License, or -// (at your option) any later version. -// -// Moodle is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -// GNU General Public License for more details. -// -// You should have received a copy of the GNU General Public License -// along with Moodle. If not, see . - -/** - * This file is part of the Moodle apps support for the choicegroup plugin. - * Defines some "providers" in the app init process so they can be used by all group choices. - * - * @copyright 2019 Dani Palou - * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later - */ - -var that = this; - -/** - * Offline provider. - */ - -var CHOICEGROUP_TABLE = 'addon_mod_choicegroup_responses'; - -// Define the database tables. -var siteSchema = { - name: 'AddonModChoiceGroupOfflineProvider', - version: 1, - onlyCurrentSite: true, - tables: [ - { - name: CHOICEGROUP_TABLE, - columns: [ - { - name: 'choicegroupid', - type: 'INTEGER', - primaryKey: true - }, - { - name: 'name', - type: 'TEXT' - }, - { - name: 'courseid', - type: 'INTEGER' - }, - { - name: 'cmid', - type: 'INTEGER' - }, - { - name: 'data', - type: 'TEXT' - }, - { - name: 'deleting', - type: 'INTEGER' - }, - { - name: 'timecreated', - type: 'INTEGER' - } - ] - } - ] -}; - -/** - * Class to handle offline group choices. - */ -function AddonModChoiceGroupOfflineProvider() { - // Register the schema so the tables are created. - that.CoreSitesProvider.registerSiteSchema(siteSchema); -} - -/** - * Delete a response stored in DB. - * - * @param id Group choice ID to remove. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved if deleted, rejected if failure. - */ -AddonModChoiceGroupOfflineProvider.prototype.deleteResponse = function(id, siteId) { - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - - return site.getDb().deleteRecords(CHOICEGROUP_TABLE, {choicegroupid: id}); - }); -}; - -/** - * Get all offline responses. - * - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved with responses. - */ -AddonModChoiceGroupOfflineProvider.prototype.getResponses = function(siteId) { - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - return site.getDb().getRecords(CHOICEGROUP_TABLE).then(function(records) { - // Parse the data of each record. - records.forEach(function(record) { - record.data = that.CoreTextUtilsProvider.parseJSON(record.data, []); - }); - - return records; - }); - }); -}; - -/** - * Check if there are offline responses to send. - * - * @param id Group choice ID. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved with boolean: true if has offline answers, false otherwise. - */ -AddonModChoiceGroupOfflineProvider.prototype.hasResponse = function(id, siteId) { - return this.getResponse(id, siteId).then(function(response) { - return !!response.choicegroupid; - }).catch(function() { - // No offline data found, return false. - return false; - }); -}; - -/** - * Get an offline response. - * - * @param id Group choice ID to get. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved with the stored data. - */ -AddonModChoiceGroupOfflineProvider.prototype.getResponse = function(id, siteId) { - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - - return site.getDb().getRecord(CHOICEGROUP_TABLE, {choicegroupid: id}).then(function(record) { - // Parse the data. - record.data = that.CoreTextUtilsProvider.parseJSON(record.data, []); - - return record; - }); - }); -}; - -/** - * Store a response to a group choice. - * - * @param id Group choice ID. - * @param name Group choice name. - * @param courseId Course ID the group choice belongs to. - * @param cmId Course module ID. - * @param data List of selected options. - * @param deleting If true, the user is deleting responses, if false, submitting. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved when data is successfully stored. - */ -AddonModChoiceGroupOfflineProvider.prototype.saveResponses = function(id, name, courseId, cmId, data, deleting, siteId) { - data = data || []; - - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - var entry = { - choicegroupid: id, - name: name, - courseid: courseId, - cmid: cmId, - data: JSON.stringify(data), - deleting: deleting ? 1 : 0, - timecreated: Date.now() - }; - - return site.getDb().insertRecord(CHOICEGROUP_TABLE, entry); - }); -}; - -var choiceGroupOffline = new AddonModChoiceGroupOfflineProvider(); - -/** - * Group choice provider. - */ - -/** - * Class to handle group choices. - */ -function AddonModChoiceGroupProvider() { } - -/** - * Delete responses from a group choice. - * - * @param id Group choice ID to remove. - * @param name The group choice name. - * @param courseId Course ID the group choice belongs to. - * @param cmId Course module ID. - * @param allowOffline Whether to allow storing the data in offline. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved with boolean: true if deleted in server, false if stored in offline. Rejected if failure. - */ -AddonModChoiceGroupProvider.prototype.deleteResponses = function(id, name, courseId, cmId, allowOffline, siteId) { - siteId = siteId || that.CoreSitesProvider.getCurrentSiteId(); - - var self = this; - - // Convenience function to store the delete to be synchronized later. - var storeOffline = function() { - return choiceGroupOffline.saveResponses(id, name, courseId, cmId, undefined, true, siteId).then(function() { - return false; - }); - }; - - if (!that.CoreAppProvider.isOnline() && allowOffline) { - // App is offline, store the action. - return storeOffline(); - } - - // If there's already some data to be sent to the server, discard it first. - return choiceGroupOffline.deleteResponse(id, siteId).catch(function() { - // Nothing was stored already. - }).then(function() { - // Now try to delete the responses in the server. - return self.deleteResponsesOnline(id, siteId).then(function() { - return true; - }).catch(function(error) { - if (!allowOffline || that.CoreUtilsProvider.isWebServiceError(error)) { - // The WebService has thrown an error, this means that responses cannot be submitted. - return Promise.reject(error); - } - - // Couldn't connect to server, store in offline. - return storeOffline(); - }); - }); -}; - -/** - * Delete responses from a group choice. It will fail if offline or cannot connect. - * - * @param id Group choice ID to remove. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved if deleted, rejected if failure. - */ -AddonModChoiceGroupProvider.prototype.deleteResponsesOnline = function(id, siteId) { - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - var params = { - choicegroupid: id - }; - - return site.write('mod_choicegroup_delete_choicegroup_responses', params).then(function(response) { - - if (!response || response.status === false) { - // Couldn't delete the responses. Reject the promise. - var error = response && response.warnings && response.warnings[0] ? - response.warnings[0] : that.CoreUtilsProvider.createFakeWSError(''); - - return Promise.reject(error); - } - }); - }); -}; - -/** - * Send the responses to a group choice. - * - * @param id Group choice ID to submit. - * @param name The group choice name. - * @param courseId Course ID the group choice belongs to. - * @param cmId Course module ID. - * @param data The responses to send. - * @param allowOffline Whether to allow storing the data in offline. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved with boolean: true if responses sent to server, false if stored in offline. Rejected if failure. - */ -AddonModChoiceGroupProvider.prototype.submitResponses = function(id, name, courseId, cmId, data, allowOffline, siteId) { - siteId = siteId || that.CoreSitesProvider.getCurrentSiteId(); - - var self = this; - - // Convenience function to store the delete to be synchronized later. - var storeOffline = function() { - return choiceGroupOffline.saveResponses(id, name, courseId, cmId, data, false, siteId).then(function() { - return false; - }); - }; - - if (!that.CoreAppProvider.isOnline() && allowOffline) { - // App is offline, store the action. - return storeOffline(); - } - - // If there's already some data to be sent to the server, discard it first. - return choiceGroupOffline.deleteResponse(id, siteId).catch(function() { - // Nothing was stored already. - }).then(function() { - // Now try to delete the responses in the server. - return self.submitResponsesOnline(id, data, siteId).then(function() { - return true; - }).catch(function(error) { - if (!allowOffline || that.CoreUtilsProvider.isWebServiceError(error)) { - // The WebService has thrown an error, this means that responses cannot be submitted. - return Promise.reject(error); - } - - // Couldn't connect to server, store in offline. - return storeOffline(); - }); - }); -}; - -/** - * Send responses from a group choice to Moodle. It will fail if offline or cannot connect. - * - * @param id Group choice ID to submit. - * @param data The responses to send. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved if deleted, rejected if failure. - */ -AddonModChoiceGroupProvider.prototype.submitResponsesOnline = function(id, data, siteId) { - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - var params = { - choicegroupid: id, - data: data - }; - - return site.write('mod_choicegroup_submit_choicegroup_response', params).then(function(response) { - - if (!response || response.status === false) { - // Couldn't delete the responses. Reject the promise. - var error = response && response.warnings && response.warnings[0] ? - response.warnings[0] : that.CoreUtilsProvider.createFakeWSError(''); - - return Promise.reject(error); - } - }); - }); -}; - -var choiceGroupProvider = new AddonModChoiceGroupProvider(); - -/** - * Group choice sync provider. - */ - -/** - * Class to handle group choice sync. - */ -function AddonModChoiceGroupSyncProvider() { - // Inherit from sync base provider. - that.CoreSyncBaseProvider.call(this, 'AddonModChoiceGroupSyncProvider', that.CoreLoggerProvider, that.CoreSitesProvider, - that.CoreAppProvider, that.CoreSyncProvider, that.CoreTextUtilsProvider, that.TranslateService, - that.CoreTimeUtilsProvider); - - this.AUTO_SYNCED = 'addon_mod_choicegroup_autom_synced'; -} - -AddonModChoiceGroupSyncProvider.prototype = Object.create(this.CoreSyncBaseProvider.prototype); -AddonModChoiceGroupSyncProvider.prototype.constructor = AddonModChoiceGroupSyncProvider; - -/** - * Try to synchronize all the group choices in a certain site or in all sites. - * - * @param force Wether to force sync not depending on last execution. - * @return Promise resolved if sync is successful, rejected if sync fails. - */ -AddonModChoiceGroupSyncProvider.prototype.syncAllChoiceGroups = function(force) { - return this.syncOnSites('group choices', this.syncAllChoiceGroupsFunc.bind(this), [force], - that.CoreSitesProvider.getCurrentSiteId()); -}; - -/** - * Sync all pending group choices on a site. - * - * @param siteId Site ID to sync. - * @param force Wether to force sync not depending on last execution. - * @return Promise resolved if sync is successful, rejected if sync fails. - */ -AddonModChoiceGroupSyncProvider.prototype.syncAllChoiceGroupsFunc = function(siteId, force) { - var self = this; - - return choiceGroupOffline.getResponses(siteId).then(function(responses) { - // Sync all responses. - var promises = responses.map(function(response) { - var promise = force ? self.syncChoiceGroup(response.choicegroupid, siteId) : - self.syncChoiceGroupIfNeeded(response.choicegroupid, siteId); - - return promise.then(function(result) { - if (result && result.updated) { - // Sync successful, send event. - that.CoreEventsProvider.trigger(self.AUTO_SYNCED, { - choiceGroupId: response.choicegroupid, - warnings: result.warnings - }, siteId); - } - }); - }); - - return Promise.all(promises); - }); -}; - -/** - * Sync a group choice only if a certain time has passed since the last time. - * - * @param id Group choice ID to be synced. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved when the group choice is synced or it doesn't need to be synced. - */ -AddonModChoiceGroupSyncProvider.prototype.syncChoiceGroupIfNeeded = function(id, siteId) { - var self = this; - - return this.isSyncNeeded(id, siteId).then(function(needed) { - if (needed) { - return self.syncChoiceGroup(id, siteId); - } - }); -}; - -/** - * Synchronize a group choice. - * - * @param id Group choice ID to be synced. - * @param siteId Site ID. If not defined, current site. - * @return Promise resolved if sync is successful, rejected otherwise. - */ -AddonModChoiceGroupSyncProvider.prototype.syncChoiceGroup = function(id, siteId) { - var self = this; - - return that.CoreSitesProvider.getSite(siteId).then(function(site) { - siteId = site.getId(); - - if (self.isSyncing(id, siteId)) { - // There's already a sync ongoing for this group choice, return the promise. - return self.getOngoingSync(id, siteId); - } - - self.logger.debug('Try to sync group choice ' + id); - - var courseId; - var cmId; - var result = { - warnings: [], - updated: false - }; - - // Get the data to synchronize. - return choiceGroupOffline.getResponse(id, siteId).catch(function() { - // No offline data found, return empty object. - return {}; - }).then(function(data) { - if (!data.choicegroupid) { - // Nothing to sync. - return; - } - - if (!that.CoreAppProvider.isOnline()) { - // Cannot sync in offline. - return Promise.reject(null); - } - - courseId = data.courseid; - cmId = data.cmid; - - // Send the responses. - var promise; - - if (data.deleting) { - // The user has deleted his responses. - promise = choiceGroupProvider.deleteResponsesOnline(id, siteId); - } else { - // The user has added a response. - promise = choiceGroupProvider.submitResponsesOnline(id, data.data, siteId); - } - - return promise.then(function() { - // Success sending the data. Delete the data stored. - result.updated = true; - - return choiceGroupOffline.deleteResponse(id, siteId); - }).catch(function(error) { - if (that.CoreUtilsProvider.isWebServiceError(error)) { - // The WebService has thrown an error, this means that responses cannot be submitted. Delete them. - result.updated = true; - - return choiceGroupOffline.deleteResponse(id, siteId).then(function() { - // Responses deleted, add a warning. - result.warnings.push(that.TranslateService.instant('core.warningofflinedatadeleted', { - component: that.TranslateService.instant('plugin.mod_choicegroup.modulename'), - name: data.name, - error: that.CoreTextUtilsProvider.getErrorMessageFromError(error) - })); - }); - } - - // Couldn't connect to server, reject. - return Promise.reject(error); - }); - }).then(function() { - if (result.updated) { - // Data has been sent to server, refresh the data. - var args = { - courseid: courseId, - cmid: cmId - }; - var preSets = { - getFromCache: false, - emergencyCache: false - }; - - return that.CoreSitePluginsProvider.getContent('mod_choicegroup', 'mobile_course_view', args, preSets) - .catch(function() { - // Ignore errors. - }); - } - }).then(function() { - // Sync finished, set sync time. - return self.setSyncTime(id, siteId); - }).then(function() { - // All done, return the result. - return result; - }); - - return self.addOngoingSync(id, syncPromise, siteId); - }); -}; - -var choiceGroupSync = new AddonModChoiceGroupSyncProvider(); - -/** - * Group choice sync handler. It will be registered in the cron delegate. - */ - -/** - * Handler to trigger group choice sync. - */ -function AddonModChoiceGroupSyncCronHandler() { - this.name = 'AddonModChoiceGroupSyncCronHandler'; -} - -/** - * Execute the process. - * - * @param siteId ID of the site affected, undefined for all sites. - * @param force Wether the execution is forced (manual sync). - * @return Promise resolved when done, rejected if failure. - */ -AddonModChoiceGroupSyncCronHandler.prototype.execute = function(siteId, force) { - // Only allow synchronizing current site. - if (!siteId || siteId == that.CoreSitesProvider.getCurrentSiteId()) { - return choiceGroupSync.syncAllChoiceGroups(force); - } -}; - -/** - * Get the time between consecutive executions. - * - * @return Time between consecutive executions (in ms). - */ -AddonModChoiceGroupSyncCronHandler.prototype.getInterval = function() { - return choiceGroupSync.syncInterval; -}; - -/** - * Link handler to treat links to a group choice. - */ - -/** - * Handler to treat links to the index page. - */ -function AddonModChoiceGroupLinkHandler() { - that.CoreContentLinksModuleIndexHandler.call(this, that.CoreCourseHelperProvider, 'AddonModChoiceGroup', 'choicegroup'); - - this.name = "AddonModChoiceGroupLinkHandler"; -} - -AddonModChoiceGroupLinkHandler.prototype = Object.create(this.CoreContentLinksModuleIndexHandler.prototype); -AddonModChoiceGroupLinkHandler.prototype.constructor = AddonModChoiceGroupLinkHandler; - -// Register the sync handler. Wait a bit to make sure the DB tables are created. -setTimeout(function() { - that.CoreCronDelegate.register(new AddonModChoiceGroupSyncCronHandler()); -}, 500); - -// Register the link handler. -this.CoreContentLinksDelegate.registerHandler(new AddonModChoiceGroupLinkHandler()); - -var result = { - choiceGroupProvider: choiceGroupProvider, - choiceGroupOffline: choiceGroupOffline, -}; - -if (this.CoreConfigConstants.versioncode > 3800) { - // 3.8.0 and older versions of the app have a bug when returning classes with Angular dependencies. - // Only return the sync provider if the version is newer. - result.choiceGroupSync = choiceGroupSync; -} - -result; diff --git a/templates/mobile_view_page_ionic3.mustache b/templates/mobile_view_page_ionic3.mustache deleted file mode 100644 index 8b3fc96..0000000 --- a/templates/mobile_view_page_ionic3.mustache +++ /dev/null @@ -1,215 +0,0 @@ -{{! - This file is part of Moodle - http://moodle.org/ - - Moodle is free software: you can redistribute it and/or modify - it under the terms of the GNU General Public License as published by - the Free Software Foundation, either version 3 of the License, or - (at your option) any later version. - - Moodle is distributed in the hope that it will be useful, - but WITHOUT ANY WARRANTY; without even the implied warranty of - MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - GNU General Public License for more details. - - You should have received a copy of the GNU General Public License - along with Moodle. If not, see . -}} -{{! - @template mod_choicegroup/mobile_view_page - - Template for the mobile view page. - - Classes required for JS: - - - - Data attributes required for JS: - - - - Context variables required for this template: - * cmid - * courseid - * choicegroup - * options - - Example context (json): - { - "cmid": "62", - "courseid": "3", - "choicegroup": { - "id": "4", - "course": "3", - "name": "Group choice activity", - "intro": "

Select your group

", - "introformat": "1", - "publish": "1", - "multipleenrollmentspossible": "1", - "showresults": "3", - "display": "0", - "allowupdate": "0", - "showunanswered": "0", - "limitanswers": "1", - "timeopen": "0", - "timeclose": "0", - "timemodified": "1528114222", - "completionsubmit": "0", - "sortgroupsby": "0", - "option": { - "10": "3", - "11": "4", - "12": "5" - }, - "grpmemberid": { - "6": [ - "3", - "2" - ] - }, - "maxanswers": { - "12": "1", - "11": "2", - "10": "2" - }, - "open": true, - "expired": false, - "alloptionsdisabled": false - }, - "options": [ - { - "id": 10, - "groupid": "3", - "name": "Group 1", - "maxanswers": "1", - "displaylayout": "0", - "countanswers": 2, - "checked": false, - "disabled": true - }, - { - "id": 11, - "groupid": "4", - "name": "Group 2", - "maxanswers": "2", - "displaylayout": "0", - "countanswers": 1, - "checked": true, - "disabled": false - }, - { - "id": 12, - "groupid": "5", - "name": "Group 3", - "maxanswers": "2", - "displaylayout": "0", - "countanswers": 0, - "checked": false, - "disabled": false - } - ] - } -}} -{{=<% %>=}} - - - - - - - - - - - - - - - {{ 'core.hasdatatosync' | translate:{$a: moduleName} }} - - - <%# choicegroup.message %> - - - <% choicegroup.message %> - - - <%/ choicegroup.message %> - - <%# choicegroup.open %> -
- - - - - {{ 'plugin.mod_choicegroup.group' | translate }} - - - <%^ choicegroup.limitanswers %> - {{ 'plugin.mod_choicegroup.members/' | translate }} - <%/ choicegroup.limitanswers %> - <%# choicegroup.limitanswers %> - {{ 'plugin.mod_choicegroup.members/max' | translate }} - <%/ choicegroup.limitanswers %> - - - - <%^ choicegroup.multipleenrollmentspossible %> - - <%/ choicegroup.multipleenrollmentspossible %> - <%# choicegroup.multipleenrollmentspossible %> - - <%/ choicegroup.multipleenrollmentspossible %> - <%# options %> - - - - <% name %> - <%^ choicegroup.multipleenrollmentspossible %> - checked="true"<%/ checked %> <%# disabled %>disabled="true"<%/ disabled %> value="<% id %>"> - <%/ choicegroup.multipleenrollmentspossible %> - <%# choicegroup.multipleenrollmentspossible %> - checked="true"<%/ checked %> - <%# disabled %>disabled="true"<%/ disabled %> - value="<% id %>"> - - <%/ choicegroup.multipleenrollmentspossible %> - - - - - <% countanswers %> - <%# choicegroup.limitanswers %> / <% maxanswers %> <%/ choicegroup.limitanswers %> - - - <%/ options %> - - - - <%^ choicegroup.expired %> - <%^ choicegroup.alloptionsdisabled %> - - - - - - <%^ choicegroup.multipleenrollmentspossible %> - <%# choicegroup.allowupdate %> - - - - <%/ choicegroup.allowupdate %> - <%/ choicegroup.multipleenrollmentspossible %> - - <%/ choicegroup.alloptionsdisabled %> - <%/ choicegroup.expired %> -
- - - - <%/ choicegroup.open %> -
From 2a757ce1c7220b92b931fafa966f81bc16446b4b Mon Sep 17 00:00:00 2001 From: Dani Palou Date: Tue, 2 Jul 2024 12:15:39 +0200 Subject: [PATCH 3/4] Mobile app: Update template to Ionic 7 --- templates/mobile_view_page_latest.mustache | 20 ++++++++------------ 1 file changed, 8 insertions(+), 12 deletions(-) diff --git a/templates/mobile_view_page_latest.mustache b/templates/mobile_view_page_latest.mustache index d92e5df..59415ae 100644 --- a/templates/mobile_view_page_latest.mustache +++ b/templates/mobile_view_page_latest.mustache @@ -118,15 +118,9 @@ - - - - - - - {{ 'core.hasdatatosync' | translate:{$a: moduleName} }} - - + + <%# choicegroup.message %> @@ -162,16 +156,18 @@ - <% name %> <%^ choicegroup.multipleenrollmentspossible %> - checked="true"<%/ checked %> <%# disabled %>disabled="true"<%/ disabled %> [value]="<% id %>"> + checked="true"<%/ checked %> <%# disabled %>disabled="true"<%/ disabled %> [value]="<% id %>"> + <% name %> + <%/ choicegroup.multipleenrollmentspossible %> <%# choicegroup.multipleenrollmentspossible %> - checked="true"<%/ checked %> <%# disabled %>disabled="true"<%/ disabled %> [value]="<% id %>"> + <% name %> <%/ choicegroup.multipleenrollmentspossible %> From c718c705d26a6c48743404cc0cfdcc5d3c57eeb0 Mon Sep 17 00:00:00 2001 From: Dani Palou Date: Mon, 15 Jul 2024 15:51:33 +0200 Subject: [PATCH 4/4] Mobile app: Use CoreNetwork instead of deprecated methods --- mobile/js/latest/courseview.js | 6 +++--- mobile/js/latest/init.js | 6 +++--- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/mobile/js/latest/courseview.js b/mobile/js/latest/courseview.js index 968616e..11330ea 100644 --- a/mobile/js/latest/courseview.js +++ b/mobile/js/latest/courseview.js @@ -201,11 +201,11 @@ this.doRefresh = (done) => { }; this.moduleName = this.TranslateService.instant('plugin.mod_choicegroup.modulename'); -this.isOnline = this.CoreAppProvider.isOnline(); +this.isOnline = this.CoreNetwork.isOnline(); // Refresh online status when changes. -const onlineObserver = this.Network.onChange().subscribe(() => { - this.isOnline = this.CoreAppProvider.isOnline(); +const onlineObserver = this.CoreNetwork.onChange().subscribe(() => { + this.isOnline = this.CoreNetwork.isOnline(); }); let syncObserver; diff --git a/mobile/js/latest/init.js b/mobile/js/latest/init.js index 13b0cdd..aeb3180 100644 --- a/mobile/js/latest/init.js +++ b/mobile/js/latest/init.js @@ -212,7 +212,7 @@ class AddonModChoiceGroupProvider { }); }; - if (!context.CoreAppProvider.isOnline() && allowOffline) { + if (!context.CoreNetwork.isOnline() && allowOffline) { // App is offline, store the action. return storeOffline(); } @@ -284,7 +284,7 @@ class AddonModChoiceGroupProvider { }); }; - if (!context.CoreAppProvider.isOnline() && allowOffline) { + if (!context.CoreNetwork.isOnline() && allowOffline) { // App is offline, store the action. return storeOffline(); } @@ -445,7 +445,7 @@ class AddonModChoiceGroupSyncProvider extends this.CoreSyncBaseProvider { return; } - if (!context.CoreAppProvider.isOnline()) { + if (!context.CoreNetwork.isOnline()) { // Cannot sync in offline. return Promise.reject(new context.CoreNetworkError()); }