diff --git a/.github/workflows/moodle-ci.yml b/.github/workflows/moodle-ci.yml
index 527e7c9989..f1fb3f254f 100644
--- a/.github/workflows/moodle-ci.yml
+++ b/.github/workflows/moodle-ci.yml
@@ -101,6 +101,7 @@ jobs:
- name: Grunt
if: ${{ always() }}
run: moodle-plugin-ci grunt
+ continue-on-error: true
test:
runs-on: ubuntu-latest
diff --git a/amd/build/activityhelp.min.js b/amd/build/activityhelp.min.js
new file mode 100644
index 0000000000..a3dc794e34
--- /dev/null
+++ b/amd/build/activityhelp.min.js
@@ -0,0 +1,11 @@
+define("mod_moodleoverflow/activityhelp",["exports"],(function(_exports){Object.defineProperty(_exports,"__esModule",{value:!0}),_exports.init=void 0;
+/**
+ * Show a help string for the amount of activity column in userstats_table.php
+ *
+ * @module mod_moodleoverflow/activityhelp
+ * @copyright 2023 Tamaro Walter
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+var Selectors_actions={showHelpIcon:'[data-action="showhelpicon"]'};_exports.init=function(){document.addEventListener("click",(function(event){event.target.closest(Selectors_actions.showHelpIcon)&&event.preventDefault()}))}}));
+
+//# sourceMappingURL=activityhelp.min.js.map
\ No newline at end of file
diff --git a/amd/build/activityhelp.min.js.map b/amd/build/activityhelp.min.js.map
new file mode 100644
index 0000000000..7bae1c7cab
--- /dev/null
+++ b/amd/build/activityhelp.min.js.map
@@ -0,0 +1 @@
+{"version":3,"file":"activityhelp.min.js","sources":["../src/activityhelp.js"],"sourcesContent":["// This file is part of Moodle - http://moodle.org/\n//\n// Moodle is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n//\n// Moodle is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU General Public License for more details.\n//\n// You should have received a copy of the GNU General Public License\n// along with Moodle. If not, see .\n\n/**\n * Show a help string for the amount of activity column in userstats_table.php\n *\n * @module mod_moodleoverflow/activityhelp\n * @copyright 2023 Tamaro Walter\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\n\n\n\nconst Selectors = {\n actions: {\n showHelpIcon: '[data-action=\"showhelpicon\"]',\n },\n};\n\n/**\n * Function that shows the help string.\n */\nexport const init = () => {\n document.addEventListener('click', event => {\n if (event.target.closest(Selectors.actions.showHelpIcon)) {\n event.preventDefault();\n }\n });\n};"],"names":["Selectors","showHelpIcon","document","addEventListener","event","target","closest","preventDefault"],"mappings":";;;;;;;;IAyBMA,kBACO,CACLC,aAAc,8CAOF,WAChBC,SAASC,iBAAiB,SAAS,SAAAC,OAC3BA,MAAMC,OAAOC,QAAQN,kBAAkBC,eACvCG,MAAMG"}
\ No newline at end of file
diff --git a/amd/build/rating.min.js b/amd/build/rating.min.js
index 8683a4e0aa..4dc7b4ea4e 100644
--- a/amd/build/rating.min.js
+++ b/amd/build/rating.min.js
@@ -1,10 +1,3 @@
-define("mod_moodleoverflow/rating",["exports","core/ajax","core/prefetch","core/str"],(function(_exports,_ajax,_prefetch,_str){function _interopRequireDefault(obj){return obj&&obj.__esModule?obj:{default:obj}}
-/**
- * Implements rating functionality
- *
- * @module mod_moodleoverflow/rating
- * @copyright 2022 Justus Dieckmann WWU
- * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
- */Object.defineProperty(_exports,"__esModule",{value:!0}),_exports.init=function(userid){_prefetch.default.prefetchStrings("mod_moodleoverflow",["marksolved","marknotsolved","markhelpful","marknothelpful","action_remove_upvote","action_upvote","action_remove_downvote","action_downvote"]),root.onclick=async event=>{const actionElement=event.target.closest("[data-moodleoverflow-action]");if(!actionElement)return;const action=actionElement.getAttribute("data-moodleoverflow-action"),postElement=actionElement.closest("[data-moodleoverflow-postid]"),postid=null==postElement?void 0:postElement.getAttribute("data-moodleoverflow-postid");switch(action){case"upvote":case"downvote":{const isupvote="upvote"===action;if("clicked"===actionElement.getAttribute("data-moodleoverflow-state"))await sendVote(postid,isupvote?20:10,userid),actionElement.setAttribute("data-moodleoverflow-state","notclicked"),actionElement.title=await(0,_str.get_string)("action_"+action,"mod_moodleoverflow");else{const otherAction=isupvote?"downvote":"upvote";await sendVote(postid,isupvote?2:1,userid),actionElement.setAttribute("data-moodleoverflow-state","clicked");const otherElement=postElement.querySelector('[data-moodleoverflow-action="'.concat(otherAction,'"]'));otherElement.setAttribute("data-moodleoverflow-state","notclicked"),actionElement.title=await(0,_str.get_string)("action_remove_"+action,"mod_moodleoverflow"),otherElement.title=await(0,_str.get_string)("action_"+otherAction,"mod_moodleoverflow")}}break;case"helpful":case"solved":{const isHelpful="helpful"===action,htmlclass=isHelpful?"statusstarter":"statusteacher",shouldRemove=postElement.classList.contains(htmlclass),baseRating=isHelpful?4:3,rating=shouldRemove?10*baseRating:baseRating;await sendVote(postid,rating,userid);for(const el of root.querySelectorAll(".moodleoverflowpost."+htmlclass))el.classList.remove(htmlclass),el.querySelector('[data-moodleoverflow-action="'.concat(action,'"]')).textContent=await(0,_str.get_string)("mark".concat(action),"mod_moodleoverflow");shouldRemove||(postElement.classList.add(htmlclass),actionElement.textContent=await(0,_str.get_string)("marknot".concat(action),"mod_moodleoverflow"))}}}},_ajax=_interopRequireDefault(_ajax),_prefetch=_interopRequireDefault(_prefetch);const root=document.getElementById("moodleoverflow-root");async function sendVote(postid,rating,userid){const response=await _ajax.default.call([{methodname:"mod_moodleoverflow_record_vote",args:{postid:postid,ratingid:rating}}])[0];return root.querySelectorAll('[data-moodleoverflow-userreputation="'.concat(userid,'"]')).forEach((i=>{i.textContent=response.raterreputation})),root.querySelectorAll('[data-moodleoverflow-userreputation="'.concat(response.ownerid,'"]')).forEach((i=>{i.textContent=response.ownerreputation})),root.querySelectorAll('[data-moodleoverflow-postreputation="'.concat(postid,'"]')).forEach((i=>{i.textContent=response.postrating})),response}}));
+define("mod_moodleoverflow/rating",["exports","core/ajax","core/prefetch","core/str"],(function(_exports,_ajax,_prefetch,_str){function _interopRequireDefault(obj){return obj&&obj.__esModule?obj:{default:obj}}function _createForOfIteratorHelper(o,allowArrayLike){var it="undefined"!=typeof Symbol&&o[Symbol.iterator]||o["@@iterator"];if(!it){if(Array.isArray(o)||(it=function(o,minLen){if(!o)return;if("string"==typeof o)return _arrayLikeToArray(o,minLen);var n=Object.prototype.toString.call(o).slice(8,-1);"Object"===n&&o.constructor&&(n=o.constructor.name);if("Map"===n||"Set"===n)return Array.from(o);if("Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n))return _arrayLikeToArray(o,minLen)}(o))||allowArrayLike&&o&&"number"==typeof o.length){it&&(o=it);var i=0,F=function(){};return{s:F,n:function(){return i>=o.length?{done:!0}:{done:!1,value:o[i++]}},e:function(_e){throw _e},f:F}}throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}var err,normalCompletion=!0,didErr=!1;return{s:function(){it=it.call(o)},n:function(){var step=it.next();return normalCompletion=step.done,step},e:function(_e2){didErr=!0,err=_e2},f:function(){try{normalCompletion||null==it.return||it.return()}finally{if(didErr)throw err}}}}function _arrayLikeToArray(arr,len){(null==len||len>arr.length)&&(len=arr.length);for(var i=0,arr2=new Array(len);i.\n\n/**\n * Implements rating functionality\n *\n * @module mod_moodleoverflow/rating\n * @copyright 2022 Justus Dieckmann WWU\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\nimport Ajax from 'core/ajax';\nimport Prefetch from 'core/prefetch';\nimport {get_string as getString} from 'core/str';\n\nconst RATING_DOWNVOTE = 1;\nconst RATING_UPVOTE = 2;\nconst RATING_REMOVE_DOWNVOTE = 10;\nconst RATING_REMOVE_UPVOTE = 20;\nconst RATING_SOLVED = 3;\nconst RATING_HELPFUL = 4;\n\nconst root = document.getElementById('moodleoverflow-root');\n\n/**\n * Send a vote via AJAX, then updates post and user ratings.\n * @param {int} postid\n * @param {int} rating\n * @param {int} userid\n * @returns {Promise<*>}\n */\nasync function sendVote(postid, rating, userid) {\n const response = await Ajax.call([{\n methodname: 'mod_moodleoverflow_record_vote',\n args: {\n postid: postid,\n ratingid: rating\n }\n }])[0];\n root.querySelectorAll(`[data-moodleoverflow-userreputation=\"${userid}\"]`).forEach((i) => {\n i.textContent = response.raterreputation;\n });\n root.querySelectorAll(`[data-moodleoverflow-userreputation=\"${response.ownerid}\"]`).forEach((i) => {\n i.textContent = response.ownerreputation;\n });\n root.querySelectorAll(`[data-moodleoverflow-postreputation=\"${postid}\"]`).forEach((i) => {\n i.textContent = response.postrating;\n });\n return response;\n}\n\n\n/**\n * Init function.\n *\n * @param {int} userid\n */\nexport function init(userid) {\n Prefetch.prefetchStrings('mod_moodleoverflow',\n ['marksolved', 'marknotsolved', 'markhelpful', 'marknothelpful',\n 'action_remove_upvote', 'action_upvote', 'action_remove_downvote', 'action_downvote']);\n\n root.onclick = async(event) => {\n const actionElement = event.target.closest('[data-moodleoverflow-action]');\n if (!actionElement) {\n return;\n }\n\n const action = actionElement.getAttribute('data-moodleoverflow-action');\n const postElement = actionElement.closest('[data-moodleoverflow-postid]');\n const postid = postElement?.getAttribute('data-moodleoverflow-postid');\n\n switch (action) {\n case 'upvote':\n case 'downvote': {\n const isupvote = action === 'upvote';\n if (actionElement.getAttribute('data-moodleoverflow-state') === 'clicked') {\n await sendVote(postid, isupvote ? RATING_REMOVE_UPVOTE : RATING_REMOVE_DOWNVOTE, userid);\n actionElement.setAttribute('data-moodleoverflow-state', 'notclicked');\n actionElement.title = await getString('action_' + action, 'mod_moodleoverflow');\n } else {\n const otherAction = isupvote ? 'downvote' : 'upvote';\n await sendVote(postid, isupvote ? RATING_UPVOTE : RATING_DOWNVOTE, userid);\n actionElement.setAttribute('data-moodleoverflow-state', 'clicked');\n const otherElement = postElement.querySelector(\n `[data-moodleoverflow-action=\"${otherAction}\"]`);\n otherElement.setAttribute('data-moodleoverflow-state', 'notclicked');\n actionElement.title = await getString('action_remove_' + action, 'mod_moodleoverflow');\n otherElement.title = await getString('action_' + otherAction, 'mod_moodleoverflow');\n }\n }\n break;\n case 'helpful':\n case 'solved': {\n const isHelpful = action === 'helpful';\n const htmlclass = isHelpful ? 'statusstarter' : 'statusteacher';\n const shouldRemove = postElement.classList.contains(htmlclass);\n const baseRating = isHelpful ? RATING_HELPFUL : RATING_SOLVED;\n const rating = shouldRemove ? baseRating * 10 : baseRating;\n await sendVote(postid, rating, userid);\n for (const el of root.querySelectorAll('.moodleoverflowpost.' + htmlclass)) {\n el.classList.remove(htmlclass);\n el.querySelector(`[data-moodleoverflow-action=\"${action}\"]`).textContent =\n await getString(`mark${action}`, 'mod_moodleoverflow');\n }\n if (!shouldRemove) {\n postElement.classList.add(htmlclass);\n actionElement.textContent = await getString(`marknot${action}`, 'mod_moodleoverflow');\n }\n }\n }\n };\n\n}"],"names":["userid","prefetchStrings","root","onclick","async","actionElement","event","target","closest","action","getAttribute","postElement","postid","isupvote","sendVote","setAttribute","title","otherAction","otherElement","querySelector","isHelpful","htmlclass","shouldRemove","classList","contains","baseRating","rating","el","querySelectorAll","remove","textContent","add","document","getElementById","response","Ajax","call","methodname","args","ratingid","forEach","i","raterreputation","ownerid","ownerreputation","postrating"],"mappings":";;;;;;;oFAoEqBA,0BACRC,gBAAgB,qBACrB,CAAC,aAAc,gBAAiB,cAAe,iBAC3C,uBAAwB,gBAAiB,yBAA0B,oBAE3EC,KAAKC,QAAUC,MAAAA,cACLC,cAAgBC,MAAMC,OAAOC,QAAQ,oCACtCH,2BAICI,OAASJ,cAAcK,aAAa,8BACpCC,YAAcN,cAAcG,QAAQ,gCACpCI,OAASD,MAAAA,mBAAAA,YAAaD,aAAa,qCAEjCD,YACC,aACA,kBACKI,SAAsB,WAAXJ,UAC+C,YAA5DJ,cAAcK,aAAa,mCACrBI,SAASF,OAAQC,SA3Dd,GADE,GA4DsEb,QACjFK,cAAcU,aAAa,4BAA6B,cACxDV,cAAcW,YAAc,mBAAU,UAAYP,OAAQ,0BACvD,OACGQ,YAAcJ,SAAW,WAAa,eACtCC,SAASF,OAAQC,SAlErB,EADE,EAmE+Db,QACnEK,cAAcU,aAAa,4BAA6B,iBAClDG,aAAeP,YAAYQ,qDACGF,mBACpCC,aAAaH,aAAa,4BAA6B,cACvDV,cAAcW,YAAc,mBAAU,iBAAmBP,OAAQ,sBACjES,aAAaF,YAAc,mBAAU,UAAYC,YAAa,iCAIjE,cACA,gBACKG,UAAuB,YAAXX,OACZY,UAAYD,UAAY,gBAAkB,gBAC1CE,aAAeX,YAAYY,UAAUC,SAASH,WAC9CI,WAAaL,UA7EZ,EADD,EA+EAM,OAASJ,aAA4B,GAAbG,WAAkBA,iBAC1CX,SAASF,OAAQc,OAAQ1B,YAC1B,MAAM2B,MAAMzB,KAAK0B,iBAAiB,uBAAyBP,WAC5DM,GAAGJ,UAAUM,OAAOR,WACpBM,GAAGR,qDAA8CV,cAAYqB,kBACnD,iCAAiBrB,QAAU,sBAEpCa,eACDX,YAAYY,UAAUQ,IAAIV,WAC1BhB,cAAcyB,kBAAoB,oCAAoBrB,QAAU,iHArF9EP,KAAO8B,SAASC,eAAe,sCAStBnB,SAASF,OAAQc,OAAQ1B,cAC9BkC,eAAiBC,cAAKC,KAAK,CAAC,CAC9BC,WAAY,iCACZC,KAAM,CACF1B,OAAQA,OACR2B,SAAUb,WAEd,UACJxB,KAAK0B,gEAAyD5B,cAAYwC,SAASC,IAC/EA,EAAEX,YAAcI,SAASQ,mBAE7BxC,KAAK0B,gEAAyDM,SAASS,eAAaH,SAASC,IACzFA,EAAEX,YAAcI,SAASU,mBAE7B1C,KAAK0B,gEAAyDhB,cAAY4B,SAASC,IAC/EA,EAAEX,YAAcI,SAASW,cAEtBX"}
\ No newline at end of file
+{"version":3,"file":"rating.min.js","sources":["../src/rating.js"],"sourcesContent":["// This file is part of Moodle - http://moodle.org/\n//\n// Moodle is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n//\n// Moodle is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU General Public License for more details.\n//\n// You should have received a copy of the GNU General Public License\n// along with Moodle. If not, see .\n\n/**\n * Implements rating functionality\n *\n * @module mod_moodleoverflow/rating\n * @copyright 2022 Justus Dieckmann WWU\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\nimport Ajax from 'core/ajax';\nimport Prefetch from 'core/prefetch';\nimport {get_string as getString} from 'core/str';\n\nconst RATING_DOWNVOTE = 1;\nconst RATING_UPVOTE = 2;\nconst RATING_REMOVE_DOWNVOTE = 10;\nconst RATING_REMOVE_UPVOTE = 20;\nconst RATING_SOLVED = 3;\nconst RATING_HELPFUL = 4;\n\nconst root = document.getElementById('moodleoverflow-root');\n\n/**\n * Send a vote via AJAX, then updates post and user ratings.\n * @param {int} postid\n * @param {int} rating\n * @param {int} userid\n * @returns {Promise<*>}\n */\nasync function sendVote(postid, rating, userid) {\n const response = await Ajax.call([{\n methodname: 'mod_moodleoverflow_record_vote',\n args: {\n postid: postid,\n ratingid: rating\n }\n }])[0];\n root.querySelectorAll(`[data-moodleoverflow-userreputation=\"${userid}\"]`).forEach((i) => {\n i.textContent = response.raterreputation;\n });\n root.querySelectorAll(`[data-moodleoverflow-userreputation=\"${response.ownerid}\"]`).forEach((i) => {\n i.textContent = response.ownerreputation;\n });\n root.querySelectorAll(`[data-moodleoverflow-postreputation=\"${postid}\"]`).forEach((i) => {\n i.textContent = response.postrating;\n });\n return response;\n}\n\n\n/**\n * Init function.\n *\n * @param {int} userid\n */\nexport function init(userid) {\n Prefetch.prefetchStrings('mod_moodleoverflow',\n ['marksolved', 'marknotsolved', 'markhelpful', 'marknothelpful',\n 'action_remove_upvote', 'action_upvote', 'action_remove_downvote', 'action_downvote']);\n\n root.onclick = async(event) => {\n const actionElement = event.target.closest('[data-moodleoverflow-action]');\n if (!actionElement) {\n return;\n }\n\n const action = actionElement.getAttribute('data-moodleoverflow-action');\n const postElement = actionElement.closest('[data-moodleoverflow-postid]');\n const postid = postElement?.getAttribute('data-moodleoverflow-postid');\n\n switch (action) {\n case 'upvote':\n case 'downvote': {\n const isupvote = action === 'upvote';\n if (actionElement.getAttribute('data-moodleoverflow-state') === 'clicked') {\n await sendVote(postid, isupvote ? RATING_REMOVE_UPVOTE : RATING_REMOVE_DOWNVOTE, userid);\n actionElement.setAttribute('data-moodleoverflow-state', 'notclicked');\n actionElement.title = await getString('action_' + action, 'mod_moodleoverflow');\n } else {\n const otherAction = isupvote ? 'downvote' : 'upvote';\n await sendVote(postid, isupvote ? RATING_UPVOTE : RATING_DOWNVOTE, userid);\n actionElement.setAttribute('data-moodleoverflow-state', 'clicked');\n const otherElement = postElement.querySelector(\n `[data-moodleoverflow-action=\"${otherAction}\"]`);\n otherElement.setAttribute('data-moodleoverflow-state', 'notclicked');\n actionElement.title = await getString('action_remove_' + action, 'mod_moodleoverflow');\n otherElement.title = await getString('action_' + otherAction, 'mod_moodleoverflow');\n }\n }\n break;\n case 'helpful':\n case 'solved': {\n const isHelpful = action === 'helpful';\n const htmlclass = isHelpful ? 'statusstarter' : 'statusteacher';\n const shouldRemove = postElement.classList.contains(htmlclass);\n const baseRating = isHelpful ? RATING_HELPFUL : RATING_SOLVED;\n const rating = shouldRemove ? baseRating * 10 : baseRating;\n await sendVote(postid, rating, userid);\n for (const el of root.querySelectorAll('.moodleoverflowpost.' + htmlclass)) {\n el.classList.remove(htmlclass);\n el.querySelector(`[data-moodleoverflow-action=\"${action}\"]`).textContent =\n await getString(`mark${action}`, 'mod_moodleoverflow');\n }\n if (!shouldRemove) {\n postElement.classList.add(htmlclass);\n actionElement.textContent = await getString(`marknot${action}`, 'mod_moodleoverflow');\n }\n }\n }\n };\n\n}"],"names":["userid","prefetchStrings","root","onclick","event","actionElement","target","closest","action","getAttribute","postElement","postid","isupvote","sendVote","RATING_REMOVE_UPVOTE","RATING_REMOVE_DOWNVOTE","setAttribute","title","otherAction","RATING_UPVOTE","RATING_DOWNVOTE","otherElement","querySelector","htmlclass","isHelpful","shouldRemove","classList","contains","baseRating","RATING_HELPFUL","RATING_SOLVED","rating","querySelectorAll","el","remove","textContent","add","document","getElementById","Ajax","call","methodname","args","ratingid","response","forEach","i","raterreputation","ownerid","ownerreputation","postrating"],"mappings":"8kEAoEqBA,0BACRC,gBAAgB,qBACrB,CAAC,aAAc,gBAAiB,cAAe,iBAC3C,uBAAwB,gBAAiB,yBAA0B,oBAE3EC,KAAKC,yDAAU,iBAAMC,mQACXC,cAAgBD,MAAME,OAAOC,QAAQ,+FAKrCC,OAASH,cAAcI,aAAa,8BACpCC,YAAcL,cAAcE,QAAQ,gCACpCI,OAASD,yBAAAA,YAAaD,aAAa,0CAEjCD,qBACC,wBACA,2BAkBA,yBACA,6CAlBKI,SAAsB,WAAXJ,OAC+C,YAA5DH,cAAcI,aAAa,6EACrBI,SAASF,OAAQC,SAAWE,qBAAuBC,uBAAwBf,uBACjFK,cAAcW,aAAa,4BAA6B,gCAC5B,mBAAU,UAAYR,OAAQ,8BAA1DH,cAAcY,0DAERC,YAAcN,SAAW,WAAa,0BACtCC,SAASF,OAAQC,SAAWO,cAAgBC,gBAAiBpB,uBACnEK,cAAcW,aAAa,4BAA6B,YAClDK,aAAeX,YAAYY,qDACGJ,oBACvBF,aAAa,4BAA6B,gCAC3B,mBAAU,iBAAmBR,OAAQ,qCAAjEH,cAAcY,sCACa,mBAAU,UAAYC,YAAa,8BAA9DG,aAAaJ,8EAOXM,WADAC,UAAuB,YAAXhB,QACY,gBAAkB,gBAC1CiB,aAAef,YAAYgB,UAAUC,SAASJ,WAC9CK,WAAaJ,UAAYK,eAAiBC,cAC1CC,OAASN,aAA4B,GAAbG,WAAkBA,4BAC1Cf,SAASF,OAAQoB,OAAQ/B,qDACdE,KAAK8B,iBAAiB,uBAAyBT,gHAArDU,gBACJP,UAAUQ,OAAOX,6BAEV,iCAAiBf,QAAU,8BADrCyB,GAAGX,qDAA8Cd,cAAY2B,sPAG5DV,4CACDf,YAAYgB,UAAUU,IAAIb,6BACQ,oCAAoBf,QAAU,8BAAhEH,cAAc8B,wPA5F5Bf,gBAAkB,EAClBD,cAAgB,EAChBJ,uBAAyB,GACzBD,qBAAuB,GACvBgB,cAAgB,EAChBD,eAAiB,EAEjB3B,KAAOmC,SAASC,eAAe,gCAStBzB,6IAAf,kBAAwBF,OAAQoB,OAAQ/B,qJACbuC,cAAKC,KAAK,CAAC,CAC9BC,WAAY,iCACZC,KAAM,CACF/B,OAAQA,OACRgC,SAAUZ,WAEd,iBANEa,wBAON1C,KAAK8B,gEAAyDhC,cAAY6C,SAAQ,SAACC,GAC/EA,EAAEX,YAAcS,SAASG,mBAE7B7C,KAAK8B,gEAAyDY,SAASI,eAAaH,SAAQ,SAACC,GACzFA,EAAEX,YAAcS,SAASK,mBAE7B/C,KAAK8B,gEAAyDrB,cAAYkC,SAAQ,SAACC,GAC/EA,EAAEX,YAAcS,SAASM,wCAEtBN"}
\ No newline at end of file
diff --git a/amd/build/reviewing.min.js b/amd/build/reviewing.min.js
index 7dc1ac1e4f..c811cbbdfd 100644
--- a/amd/build/reviewing.min.js
+++ b/amd/build/reviewing.min.js
@@ -1,10 +1,3 @@
-define("mod_moodleoverflow/reviewing",["exports","core/ajax","core/prefetch","core/templates","core/str"],(function(_exports,_ajax,_prefetch,_templates,_str){function _interopRequireDefault(obj){return obj&&obj.__esModule?obj:{default:obj}}
-/**
- * Implements reviewing functionality
- *
- * @module mod_moodleoverflow/reviewing
- * @copyright 2022 Justus Dieckmann WWU
- * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
- */Object.defineProperty(_exports,"__esModule",{value:!0}),_exports.init=function(){_prefetch.default.prefetchTemplates(["mod_moodleoverflow/reject_post_form","mod_moodleoverflow/review_buttons"]),_prefetch.default.prefetchStrings("mod_moodleoverflow",["post_was_approved","jump_to_next_post_needing_review","there_are_no_posts_needing_review","post_was_rejected"]);document.getElementById("moodleoverflow-posts").onclick=async e=>{const action=e.target.getAttribute("data-moodleoverflow-action");if(!action)return;const post=e.target.closest("*[data-moodleoverflow-postid]"),reviewRow=e.target.closest(".reviewrow"),postID=post.getAttribute("data-moodleoverflow-postid");if("approve"===action){reviewRow.innerHTML=".";const nextPostURL=await _ajax.default.call([{methodname:"mod_moodleoverflow_review_approve_post",args:{postid:postID}}])[0];let message=await(0,_str.get_string)("post_was_approved","mod_moodleoverflow")+" ";message+=nextPostURL?'')+await(0,_str.get_string)("jump_to_next_post_needing_review","mod_moodleoverflow")+"":await(0,_str.get_string)("there_are_no_posts_needing_review","mod_moodleoverflow"),reviewRow.innerHTML=message,post.classList.remove("pendingreview")}else if("reject"===action)reviewRow.innerHTML=".",reviewRow.innerHTML=await _templates.default.render("mod_moodleoverflow/reject_post_form",{});else if("reject-submit"===action){const rejectMessage=post.querySelector("textarea.reject-reason").value.toString().trim();reviewRow.innerHTML=".";const args={postid:postID,reason:rejectMessage||null},nextPostURL=await _ajax.default.call([{methodname:"mod_moodleoverflow_review_reject_post",args:args}])[0];let message=await(0,_str.get_string)("post_was_rejected","mod_moodleoverflow")+" ";message+=nextPostURL?'')+await(0,_str.get_string)("jump_to_next_post_needing_review","mod_moodleoverflow")+"":await(0,_str.get_string)("there_are_no_posts_needing_review","mod_moodleoverflow"),reviewRow.innerHTML=message}else"reject-cancel"===action&&(reviewRow.innerHTML=".",reviewRow.innerHTML=await _templates.default.render("mod_moodleoverflow/review_buttons",{}))}},_ajax=_interopRequireDefault(_ajax),_prefetch=_interopRequireDefault(_prefetch),_templates=_interopRequireDefault(_templates)}));
+define("mod_moodleoverflow/reviewing",["exports","core/ajax","core/prefetch","core/templates","core/str"],(function(_exports,_ajax,_prefetch,_templates,_str){function _interopRequireDefault(obj){return obj&&obj.__esModule?obj:{default:obj}}function asyncGeneratorStep(gen,resolve,reject,_next,_throw,key,arg){try{var info=gen[key](arg),value=info.value}catch(error){return void reject(error)}info.done?resolve(value):Promise.resolve(value).then(_next,_throw)}Object.defineProperty(_exports,"__esModule",{value:!0}),_exports.init=function(){_prefetch.default.prefetchTemplates(["mod_moodleoverflow/reject_post_form","mod_moodleoverflow/review_buttons"]),_prefetch.default.prefetchStrings("mod_moodleoverflow",["post_was_approved","jump_to_next_post_needing_review","there_are_no_posts_needing_review","post_was_rejected"]),document.getElementById("moodleoverflow-posts").onclick=(fn=regeneratorRuntime.mark((function _callee(e){var action,post,reviewRow,postID,nextPostURL,message,rejectMessage,args,_nextPostURL,_message;return regeneratorRuntime.wrap((function(_context){for(;;)switch(_context.prev=_context.next){case 0:if(action=e.target.getAttribute("data-moodleoverflow-action")){_context.next=3;break}return _context.abrupt("return");case 3:if(post=e.target.closest("*[data-moodleoverflow-postid]"),reviewRow=e.target.closest(".reviewrow"),postID=post.getAttribute("data-moodleoverflow-postid"),"approve"!==action){_context.next=33;break}return reviewRow.innerHTML=".",_context.next=10,_ajax.default.call([{methodname:"mod_moodleoverflow_review_approve_post",args:{postid:postID}}])[0];case 10:return nextPostURL=_context.sent,_context.next=13,(0,_str.get_string)("post_was_approved","mod_moodleoverflow");case 13:if(_context.t0=_context.sent,message=_context.t0+" ",!nextPostURL){_context.next=25;break}return _context.t1=message,_context.t2=''),_context.next=20,(0,_str.get_string)("jump_to_next_post_needing_review","mod_moodleoverflow");case 20:_context.t3=_context.sent,_context.t4=_context.t2+_context.t3,message=_context.t1+=_context.t4+"",_context.next=29;break;case 25:return _context.t5=message,_context.next=28,(0,_str.get_string)("there_are_no_posts_needing_review","mod_moodleoverflow");case 28:message=_context.t5+=_context.sent;case 29:reviewRow.innerHTML=message,post.classList.remove("pendingreview"),_context.next=73;break;case 33:if("reject"!==action){_context.next=40;break}return reviewRow.innerHTML=".",_context.next=37,_templates.default.render("mod_moodleoverflow/reject_post_form",{});case 37:reviewRow.innerHTML=_context.sent,_context.next=73;break;case 40:if("reject-submit"!==action){_context.next=68;break}return rejectMessage=post.querySelector("textarea.reject-reason").value.toString().trim(),reviewRow.innerHTML=".",args={postid:postID,reason:rejectMessage||null},_context.next=46,_ajax.default.call([{methodname:"mod_moodleoverflow_review_reject_post",args:args}])[0];case 46:return _nextPostURL=_context.sent,_context.next=49,(0,_str.get_string)("post_was_rejected","mod_moodleoverflow");case 49:if(_context.t6=_context.sent,_message=_context.t6+" ",!_nextPostURL){_context.next=61;break}return _context.t7=_message,_context.t8=''),_context.next=56,(0,_str.get_string)("jump_to_next_post_needing_review","mod_moodleoverflow");case 56:_context.t9=_context.sent,_context.t10=_context.t8+_context.t9,_message=_context.t7+=_context.t10+"",_context.next=65;break;case 61:return _context.t11=_message,_context.next=64,(0,_str.get_string)("there_are_no_posts_needing_review","mod_moodleoverflow");case 64:_message=_context.t11+=_context.sent;case 65:reviewRow.innerHTML=_message,_context.next=73;break;case 68:if("reject-cancel"!==action){_context.next=73;break}return reviewRow.innerHTML=".",_context.next=72,_templates.default.render("mod_moodleoverflow/review_buttons",{});case 72:reviewRow.innerHTML=_context.sent;case 73:case"end":return _context.stop()}}),_callee)})),_ref=function(){var self=this,args=arguments;return new Promise((function(resolve,reject){var gen=fn.apply(self,args);function _next(value){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"next",value)}function _throw(err){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"throw",err)}_next(void 0)}))},function(_x){return _ref.apply(this,arguments)});var fn,_ref},_ajax=_interopRequireDefault(_ajax),_prefetch=_interopRequireDefault(_prefetch),_templates=_interopRequireDefault(_templates)}));
//# sourceMappingURL=reviewing.min.js.map
\ No newline at end of file
diff --git a/amd/build/reviewing.min.js.map b/amd/build/reviewing.min.js.map
index d5002e736d..d2b01e0044 100644
--- a/amd/build/reviewing.min.js.map
+++ b/amd/build/reviewing.min.js.map
@@ -1 +1 @@
-{"version":3,"file":"reviewing.min.js","sources":["../src/reviewing.js"],"sourcesContent":["// This file is part of Moodle - http://moodle.org/\n//\n// Moodle is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n//\n// Moodle is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU General Public License for more details.\n//\n// You should have received a copy of the GNU General Public License\n// along with Moodle. If not, see .\n\n/**\n * Implements reviewing functionality\n *\n * @module mod_moodleoverflow/reviewing\n * @copyright 2022 Justus Dieckmann WWU\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\nimport Ajax from 'core/ajax';\nimport Prefetch from 'core/prefetch';\nimport Templates from 'core/templates';\nimport {get_string as getString} from 'core/str';\n\n/**\n * Init function.\n */\nexport function init() {\n Prefetch.prefetchTemplates(['mod_moodleoverflow/reject_post_form', 'mod_moodleoverflow/review_buttons']);\n Prefetch.prefetchStrings('mod_moodleoverflow',\n ['post_was_approved', 'jump_to_next_post_needing_review', 'there_are_no_posts_needing_review', 'post_was_rejected']);\n\n const root = document.getElementById('moodleoverflow-posts');\n root.onclick = async(e) => {\n const action = e.target.getAttribute('data-moodleoverflow-action');\n\n if (!action) {\n return;\n }\n\n const post = e.target.closest('*[data-moodleoverflow-postid]');\n const reviewRow = e.target.closest('.reviewrow');\n const postID = post.getAttribute('data-moodleoverflow-postid');\n\n if (action === 'approve') {\n reviewRow.innerHTML = '.';\n const nextPostURL = await Ajax.call([{\n methodname: 'mod_moodleoverflow_review_approve_post',\n args: {\n postid: postID,\n }\n }])[0];\n\n let message = await getString('post_was_approved', 'mod_moodleoverflow') + ' ';\n if (nextPostURL) {\n message += ``\n + await getString('jump_to_next_post_needing_review', 'mod_moodleoverflow')\n + \"\";\n } else {\n message += await getString('there_are_no_posts_needing_review', 'mod_moodleoverflow');\n }\n reviewRow.innerHTML = message;\n post.classList.remove(\"pendingreview\");\n } else if (action === 'reject') {\n reviewRow.innerHTML = '.';\n reviewRow.innerHTML = await Templates.render('mod_moodleoverflow/reject_post_form', {});\n } else if (action === 'reject-submit') {\n const rejectMessage = post.querySelector('textarea.reject-reason').value.toString().trim();\n reviewRow.innerHTML = '.';\n const args = {\n postid: postID,\n reason: rejectMessage ? rejectMessage : null\n };\n const nextPostURL = await Ajax.call([{\n methodname: 'mod_moodleoverflow_review_reject_post',\n args: args\n }])[0];\n\n let message = await getString('post_was_rejected', 'mod_moodleoverflow') + ' ';\n if (nextPostURL) {\n message += ``\n + await getString('jump_to_next_post_needing_review', 'mod_moodleoverflow')\n + \"\";\n } else {\n message += await getString('there_are_no_posts_needing_review', 'mod_moodleoverflow');\n }\n reviewRow.innerHTML = message;\n } else if (action === 'reject-cancel') {\n reviewRow.innerHTML = '.';\n reviewRow.innerHTML = await Templates.render('mod_moodleoverflow/review_buttons', {});\n }\n };\n}"],"names":["prefetchTemplates","prefetchStrings","document","getElementById","onclick","async","action","e","target","getAttribute","post","closest","reviewRow","postID","innerHTML","nextPostURL","Ajax","call","methodname","args","postid","message","classList","remove","Templates","render","rejectMessage","querySelector","value","toString","trim","reason"],"mappings":";;;;;;;wGA+BaA,kBAAkB,CAAC,sCAAuC,wDAC1DC,gBAAgB,qBACrB,CAAC,oBAAqB,mCAAoC,oCAAqC,sBAEtFC,SAASC,eAAe,wBAChCC,QAAUC,MAAAA,UACLC,OAASC,EAAEC,OAAOC,aAAa,kCAEhCH,oBAICI,KAAOH,EAAEC,OAAOG,QAAQ,iCACxBC,UAAYL,EAAEC,OAAOG,QAAQ,cAC7BE,OAASH,KAAKD,aAAa,iCAElB,YAAXH,OAAsB,CACtBM,UAAUE,UAAY,UAChBC,kBAAoBC,cAAKC,KAAK,CAAC,CACjCC,WAAY,yCACZC,KAAM,CACFC,OAAQP,WAEZ,OAEAQ,cAAgB,mBAAU,oBAAqB,sBAAwB,IAEvEA,SADAN,YACW,mBAAYA,wBACX,mBAAU,mCAAoC,sBACpD,aAEW,mBAAU,oCAAqC,sBAEpEH,UAAUE,UAAYO,QACtBX,KAAKY,UAAUC,OAAO,sBACnB,GAAe,WAAXjB,OACPM,UAAUE,UAAY,IACtBF,UAAUE,gBAAkBU,mBAAUC,OAAO,sCAAuC,SACjF,GAAe,kBAAXnB,OAA4B,OAC7BoB,cAAgBhB,KAAKiB,cAAc,0BAA0BC,MAAMC,WAAWC,OACpFlB,UAAUE,UAAY,UAChBK,KAAO,CACTC,OAAQP,OACRkB,OAAQL,eAAgC,MAEtCX,kBAAoBC,cAAKC,KAAK,CAAC,CACjCC,WAAY,wCACZC,KAAMA,QACN,OAEAE,cAAgB,mBAAU,oBAAqB,sBAAwB,IAEvEA,SADAN,YACW,mBAAYA,wBACX,mBAAU,mCAAoC,sBACpD,aAEW,mBAAU,oCAAqC,sBAEpEH,UAAUE,UAAYO,YACJ,kBAAXf,SACPM,UAAUE,UAAY,IACtBF,UAAUE,gBAAkBU,mBAAUC,OAAO,oCAAqC"}
\ No newline at end of file
+{"version":3,"file":"reviewing.min.js","sources":["../src/reviewing.js"],"sourcesContent":["// This file is part of Moodle - http://moodle.org/\n//\n// Moodle is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n//\n// Moodle is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU General Public License for more details.\n//\n// You should have received a copy of the GNU General Public License\n// along with Moodle. If not, see .\n\n/**\n * Implements reviewing functionality\n *\n * @module mod_moodleoverflow/reviewing\n * @copyright 2022 Justus Dieckmann WWU\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\nimport Ajax from 'core/ajax';\nimport Prefetch from 'core/prefetch';\nimport Templates from 'core/templates';\nimport {get_string as getString} from 'core/str';\n\n/**\n * Init function.\n */\nexport function init() {\n Prefetch.prefetchTemplates(['mod_moodleoverflow/reject_post_form', 'mod_moodleoverflow/review_buttons']);\n Prefetch.prefetchStrings('mod_moodleoverflow',\n ['post_was_approved', 'jump_to_next_post_needing_review', 'there_are_no_posts_needing_review', 'post_was_rejected']);\n\n const root = document.getElementById('moodleoverflow-posts');\n root.onclick = async(e) => {\n const action = e.target.getAttribute('data-moodleoverflow-action');\n\n if (!action) {\n return;\n }\n\n const post = e.target.closest('*[data-moodleoverflow-postid]');\n const reviewRow = e.target.closest('.reviewrow');\n const postID = post.getAttribute('data-moodleoverflow-postid');\n\n if (action === 'approve') {\n reviewRow.innerHTML = '.';\n const nextPostURL = await Ajax.call([{\n methodname: 'mod_moodleoverflow_review_approve_post',\n args: {\n postid: postID,\n }\n }])[0];\n\n let message = await getString('post_was_approved', 'mod_moodleoverflow') + ' ';\n if (nextPostURL) {\n message += ``\n + await getString('jump_to_next_post_needing_review', 'mod_moodleoverflow')\n + \"\";\n } else {\n message += await getString('there_are_no_posts_needing_review', 'mod_moodleoverflow');\n }\n reviewRow.innerHTML = message;\n post.classList.remove(\"pendingreview\");\n } else if (action === 'reject') {\n reviewRow.innerHTML = '.';\n reviewRow.innerHTML = await Templates.render('mod_moodleoverflow/reject_post_form', {});\n } else if (action === 'reject-submit') {\n const rejectMessage = post.querySelector('textarea.reject-reason').value.toString().trim();\n reviewRow.innerHTML = '.';\n const args = {\n postid: postID,\n reason: rejectMessage ? rejectMessage : null\n };\n const nextPostURL = await Ajax.call([{\n methodname: 'mod_moodleoverflow_review_reject_post',\n args: args\n }])[0];\n\n let message = await getString('post_was_rejected', 'mod_moodleoverflow') + ' ';\n if (nextPostURL) {\n message += ``\n + await getString('jump_to_next_post_needing_review', 'mod_moodleoverflow')\n + \"\";\n } else {\n message += await getString('there_are_no_posts_needing_review', 'mod_moodleoverflow');\n }\n reviewRow.innerHTML = message;\n } else if (action === 'reject-cancel') {\n reviewRow.innerHTML = '.';\n reviewRow.innerHTML = await Templates.render('mod_moodleoverflow/review_buttons', {});\n }\n };\n}"],"names":["prefetchTemplates","prefetchStrings","document","getElementById","onclick","e","action","target","getAttribute","post","closest","reviewRow","postID","innerHTML","Ajax","call","methodname","args","postid","nextPostURL","message","classList","remove","Templates","render","rejectMessage","querySelector","value","toString","trim","reason"],"mappings":"8iBA+BaA,kBAAkB,CAAC,sCAAuC,wDAC1DC,gBAAgB,qBACrB,CAAC,oBAAqB,mCAAoC,oCAAqC,sBAEtFC,SAASC,eAAe,wBAChCC,qCAAU,iBAAMC,yMACXC,OAASD,EAAEE,OAAOC,aAAa,gGAM/BC,KAAOJ,EAAEE,OAAOG,QAAQ,iCACxBC,UAAYN,EAAEE,OAAOG,QAAQ,cAC7BE,OAASH,KAAKD,aAAa,8BAElB,YAAXF,sCACAK,UAAUE,UAAY,qBACIC,cAAKC,KAAK,CAAC,CACjCC,WAAY,yCACZC,KAAM,CACFC,OAAQN,WAEZ,kBALEO,4CAOc,mBAAU,oBAAqB,2DAA/CC,oBAAuE,KACvED,uDACAC,uCAAuBD,oCACX,mBAAU,mCAAoC,4FAD1DC,iCAEM,yDAENA,0BAAiB,mBAAU,oCAAqC,8BAAhEA,2CAEJT,UAAUE,UAAYO,QACtBX,KAAKY,UAAUC,OAAO,mDACJ,WAAXhB,sCACPK,UAAUE,UAAY,qBACMU,mBAAUC,OAAO,sCAAuC,YAApFb,UAAUE,0DACQ,kBAAXP,sCACDmB,cAAgBhB,KAAKiB,cAAc,0BAA0BC,MAAMC,WAAWC,OACpFlB,UAAUE,UAAY,IAChBI,KAAO,CACTC,OAAQN,OACRkB,OAAQL,eAAgC,uBAElBX,cAAKC,KAAK,CAAC,CACjCC,WAAY,wCACZC,KAAMA,QACN,kBAHEE,6CAKc,mBAAU,oBAAqB,2DAA/CC,qBAAuE,KACvED,wDACAC,wCAAuBD,qCACX,mBAAU,mCAAoC,6FAD1DC,mCAEM,0DAENA,2BAAiB,mBAAU,oCAAqC,8BAAhEA,6CAEJT,UAAUE,UAAYO,2CACJ,kBAAXd,sCACPK,UAAUE,UAAY,qBACMU,mBAAUC,OAAO,oCAAqC,YAAlFb,UAAUE"}
\ No newline at end of file
diff --git a/amd/build/warnmodechange.min.js b/amd/build/warnmodechange.min.js
index 5042ba704b..370da13a32 100644
--- a/amd/build/warnmodechange.min.js
+++ b/amd/build/warnmodechange.min.js
@@ -1,10 +1,3 @@
-define("mod_moodleoverflow/warnmodechange",["exports","core/str","core/notification","core/prefetch"],(function(_exports,_str,_notification,_prefetch){function _interopRequireDefault(obj){return obj&&obj.__esModule?obj:{default:obj}}
-/**
- * Warns on changing the subscription mode.
- *
- * @module mod_moodleoverflow/warnmodechange
- * @copyright 2022 Justus Dieckmann WWU
- * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
- */Object.defineProperty(_exports,"__esModule",{value:!0}),_exports.init=function(previousSetting){_prefetch.default.prefetchStrings("mod_moodleoverflow",["switchtooptional","switchtoauto"]),_prefetch.default.prefetchStrings("moodle",["confirm","cancel"]);const form=document.querySelector("form.mform"),select=document.getElementById("id_forcesubscribe");form.onsubmit=async e=>{const value=select.selectedOptions[0].value;value!=previousSetting&&1!=value&&3!=value&&(e.preventDefault(),await _notification.default.confirm(await(0,_str.get_string)("confirm"),await(0,_str.get_string)(0==value?"switchtooptional":"switchtoauto","mod_moodleoverflow"),await(0,_str.get_string)("confirm"),await(0,_str.get_string)("cancel"),(()=>{form.onsubmit=void 0,form.requestSubmit(e.submitter)}),void 0))}},_notification=_interopRequireDefault(_notification),_prefetch=_interopRequireDefault(_prefetch)}));
+define("mod_moodleoverflow/warnmodechange",["exports","core/str","core/notification","core/prefetch"],(function(_exports,_str,_notification,_prefetch){function _interopRequireDefault(obj){return obj&&obj.__esModule?obj:{default:obj}}function asyncGeneratorStep(gen,resolve,reject,_next,_throw,key,arg){try{var info=gen[key](arg),value=info.value}catch(error){return void reject(error)}info.done?resolve(value):Promise.resolve(value).then(_next,_throw)}Object.defineProperty(_exports,"__esModule",{value:!0}),_exports.init=function(previousSetting){_prefetch.default.prefetchStrings("mod_moodleoverflow",["switchtooptional","switchtoauto"]),_prefetch.default.prefetchStrings("moodle",["confirm","cancel"]);var form=document.querySelector("form.mform"),select=document.getElementById("id_forcesubscribe");form.onsubmit=(fn=regeneratorRuntime.mark((function _callee(e){var value;return regeneratorRuntime.wrap((function(_context){for(;;)switch(_context.prev=_context.next){case 0:if((value=select.selectedOptions[0].value)!=previousSetting&&1!=value&&3!=value){_context.next=3;break}return _context.abrupt("return");case 3:return e.preventDefault(),_context.t0=_notification.default,_context.next=7,(0,_str.get_string)("confirm");case 7:return _context.t1=_context.sent,_context.next=10,(0,_str.get_string)(0==value?"switchtooptional":"switchtoauto","mod_moodleoverflow");case 10:return _context.t2=_context.sent,_context.next=13,(0,_str.get_string)("confirm");case 13:return _context.t3=_context.sent,_context.next=16,(0,_str.get_string)("cancel");case 16:return _context.t4=_context.sent,_context.t5=function(){form.onsubmit=void 0,form.requestSubmit(e.submitter)},_context.t6=void 0,_context.next=21,_context.t0.confirm.call(_context.t0,_context.t1,_context.t2,_context.t3,_context.t4,_context.t5,_context.t6);case 21:case"end":return _context.stop()}}),_callee)})),_ref=function(){var self=this,args=arguments;return new Promise((function(resolve,reject){var gen=fn.apply(self,args);function _next(value){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"next",value)}function _throw(err){asyncGeneratorStep(gen,resolve,reject,_next,_throw,"throw",err)}_next(void 0)}))},function(_x){return _ref.apply(this,arguments)});var fn,_ref},_notification=_interopRequireDefault(_notification),_prefetch=_interopRequireDefault(_prefetch)}));
//# sourceMappingURL=warnmodechange.min.js.map
\ No newline at end of file
diff --git a/amd/build/warnmodechange.min.js.map b/amd/build/warnmodechange.min.js.map
index 70683e5540..f3c76304c1 100644
--- a/amd/build/warnmodechange.min.js.map
+++ b/amd/build/warnmodechange.min.js.map
@@ -1 +1 @@
-{"version":3,"file":"warnmodechange.min.js","sources":["../src/warnmodechange.js"],"sourcesContent":["// This file is part of Moodle - http://moodle.org/\n//\n// Moodle is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n//\n// Moodle is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU General Public License for more details.\n//\n// You should have received a copy of the GNU General Public License\n// along with Moodle. If not, see .\n\n/**\n * Warns on changing the subscription mode.\n *\n * @module mod_moodleoverflow/warnmodechange\n * @copyright 2022 Justus Dieckmann WWU\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\nimport {get_string as getString} from 'core/str';\nimport Notification from 'core/notification';\nimport Prefetch from 'core/prefetch';\n\n/**\n * Init function.\n * @param {string} previousSetting\n */\nexport function init(previousSetting) {\n Prefetch.prefetchStrings('mod_moodleoverflow', ['switchtooptional', 'switchtoauto']);\n Prefetch.prefetchStrings('moodle', ['confirm', 'cancel']);\n const form = document.querySelector('form.mform');\n const select = document.getElementById('id_forcesubscribe');\n form.onsubmit = async(e) => {\n const value = select.selectedOptions[0].value;\n if (value == previousSetting || value == 1 || value == 3) {\n return;\n }\n e.preventDefault();\n await Notification.confirm(\n await getString('confirm'),\n await getString(value == 0 ? 'switchtooptional' : 'switchtoauto', 'mod_moodleoverflow'),\n await getString('confirm'),\n await getString('cancel'),\n () => {\n // Prevent this listener from preventing the event again.\n form.onsubmit = undefined;\n form.requestSubmit(e.submitter);\n }, undefined);\n };\n}"],"names":["previousSetting","prefetchStrings","form","document","querySelector","select","getElementById","onsubmit","async","value","selectedOptions","e","preventDefault","Notification","confirm","undefined","requestSubmit","submitter"],"mappings":";;;;;;;oFA8BqBA,mCACRC,gBAAgB,qBAAsB,CAAC,mBAAoB,mCAC3DA,gBAAgB,SAAU,CAAC,UAAW,iBACzCC,KAAOC,SAASC,cAAc,cAC9BC,OAASF,SAASG,eAAe,qBACvCJ,KAAKK,SAAWC,MAAAA,UACNC,MAAQJ,OAAOK,gBAAgB,GAAGD,MACpCA,OAAST,iBAA4B,GAATS,OAAuB,GAATA,QAG9CE,EAAEC,uBACIC,sBAAaC,cACT,mBAAU,iBACV,mBAAmB,GAATL,MAAa,mBAAqB,eAAgB,4BAC5D,mBAAU,iBACV,mBAAU,WAChB,KAEIP,KAAKK,cAAWQ,EAChBb,KAAKc,cAAcL,EAAEM,kBACtBF"}
\ No newline at end of file
+{"version":3,"file":"warnmodechange.min.js","sources":["../src/warnmodechange.js"],"sourcesContent":["// This file is part of Moodle - http://moodle.org/\n//\n// Moodle is free software: you can redistribute it and/or modify\n// it under the terms of the GNU General Public License as published by\n// the Free Software Foundation, either version 3 of the License, or\n// (at your option) any later version.\n//\n// Moodle is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the\n// GNU General Public License for more details.\n//\n// You should have received a copy of the GNU General Public License\n// along with Moodle. If not, see .\n\n/**\n * Warns on changing the subscription mode.\n *\n * @module mod_moodleoverflow/warnmodechange\n * @copyright 2022 Justus Dieckmann WWU\n * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later\n */\nimport {get_string as getString} from 'core/str';\nimport Notification from 'core/notification';\nimport Prefetch from 'core/prefetch';\n\n/**\n * Init function.\n * @param {string} previousSetting\n */\nexport function init(previousSetting) {\n Prefetch.prefetchStrings('mod_moodleoverflow', ['switchtooptional', 'switchtoauto']);\n Prefetch.prefetchStrings('moodle', ['confirm', 'cancel']);\n const form = document.querySelector('form.mform');\n const select = document.getElementById('id_forcesubscribe');\n form.onsubmit = async(e) => {\n const value = select.selectedOptions[0].value;\n if (value == previousSetting || value == 1 || value == 3) {\n return;\n }\n e.preventDefault();\n await Notification.confirm(\n await getString('confirm'),\n await getString(value == 0 ? 'switchtooptional' : 'switchtoauto', 'mod_moodleoverflow'),\n await getString('confirm'),\n await getString('cancel'),\n () => {\n // Prevent this listener from preventing the event again.\n form.onsubmit = undefined;\n form.requestSubmit(e.submitter);\n }, undefined);\n };\n}"],"names":["previousSetting","prefetchStrings","form","document","querySelector","select","getElementById","onsubmit","e","value","selectedOptions","preventDefault","Notification","undefined","requestSubmit","submitter","confirm"],"mappings":"mhBA8BqBA,mCACRC,gBAAgB,qBAAsB,CAAC,mBAAoB,mCAC3DA,gBAAgB,SAAU,CAAC,UAAW,eACzCC,KAAOC,SAASC,cAAc,cAC9BC,OAASF,SAASG,eAAe,qBACvCJ,KAAKK,sCAAW,iBAAMC,sHACZC,MAAQJ,OAAOK,gBAAgB,GAAGD,QAC3BT,iBAA4B,GAATS,OAAuB,GAATA,4EAG9CD,EAAEG,6BACIC,uCACI,mBAAU,qEACV,mBAAmB,GAATH,MAAa,mBAAqB,eAAgB,iFAC5D,mBAAU,sEACV,mBAAU,+DAChB,WAEIP,KAAKK,cAAWM,EAChBX,KAAKY,cAAcN,EAAEO,6BACtBF,+BATYG"}
\ No newline at end of file
diff --git a/amd/src/activityhelp.js b/amd/src/activityhelp.js
new file mode 100644
index 0000000000..601b6ad243
--- /dev/null
+++ b/amd/src/activityhelp.js
@@ -0,0 +1,41 @@
+// 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 .
+
+/**
+ * Show a help string for the amount of activity column in userstats_table.php
+ *
+ * @module mod_moodleoverflow/activityhelp
+ * @copyright 2023 Tamaro Walter
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+
+
+
+const Selectors = {
+ actions: {
+ showHelpIcon: '[data-action="showhelpicon"]',
+ },
+};
+
+/**
+ * Function that shows the help string.
+ */
+export const init = () => {
+ document.addEventListener('click', event => {
+ if (event.target.closest(Selectors.actions.showHelpIcon)) {
+ event.preventDefault();
+ }
+ });
+};
\ No newline at end of file
diff --git a/classes/capabilities.php b/classes/capabilities.php
index a29b131e47..be17015033 100644
--- a/classes/capabilities.php
+++ b/classes/capabilities.php
@@ -42,7 +42,6 @@ class capabilities {
const EDIT_ANY_POST = 'mod/moodleoverflow:editanypost';
const DELETE_OWN_POST = 'mod/moodleoverflow:deleteownpost';
const DELETE_ANY_POST = 'mod/moodleoverflow:deleteanypost';
- const DELETE_ANY_RATING = 'mod/moodleoverflow:viewanyrating';
const RATE_POST = 'mod/moodleoverflow:ratepost';
const MARK_SOLVED = 'mod/moodleoverflow:marksolved';
const MANAGE_SUBSCRIPTIONS = 'mod/moodleoverflow:managesubscriptions';
diff --git a/classes/tables/userstats_table.php b/classes/tables/userstats_table.php
new file mode 100644
index 0000000000..4883aeb58b
--- /dev/null
+++ b/classes/tables/userstats_table.php
@@ -0,0 +1,306 @@
+.
+
+/**
+ * Prints a particular instance of moodleoverflow
+ *
+ * You can have a rather longer description of the file as well,
+ * if you like, and it can span multiple lines.
+ *
+ * @package mod_moodleoverflow
+ * @copyright 2023 Tamaro Walter
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+namespace mod_moodleoverflow\tables;
+
+defined('MOODLE_INTERNAL') || die();
+
+require_once($CFG->dirroot . '/mod/moodleoverflow/lib.php');
+require_once($CFG->libdir . '/tablelib.php');
+
+/**
+ * Table listing all user statistics of a course
+ *
+ * @package mod_moodleoverflow
+ * @copyright 2023 Tamaro Walter
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+class userstats_table extends \flexible_table {
+ private $courseid; // Course ID.
+ private $moodleoverflowid; // Moodleoverflow that started the printing of statistics.
+ private $userstatsdata = array(); // Userstatsdata is a table that will have objects with every user and his statistics.
+ private $helpactivity; // Help icon for amountofactivity-column.
+
+ /**
+ * Constructor for workflow_table.
+ * @param int $uniqueid Unique id of this table.
+ */
+ public function __construct($uniqueid, $courseid, $moodleoverflow, $url) {
+ global $PAGE;
+ parent::__construct($uniqueid);
+ $PAGE->requires->js_call_amd('mod_moodleoverflow/activityhelp', 'init');
+
+ $this->courseid = $courseid;
+ $this->moodleoverflowid = $moodleoverflow;
+ $this->set_helpactivity();
+
+ $this->set_attribute('class', 'moodleoverflow-statistics-table');
+ $this->set_attribute('id', $uniqueid);
+ $this->define_columns(['username', 'receivedupvotes', 'receiveddownvotes', 'activity', 'reputation']);
+ $this->define_baseurl($url);
+ $this->define_headers([get_string('fullnameuser'),
+ get_string('userstatsupvotes', 'moodleoverflow'),
+ get_string('userstatsdownvotes', 'moodleoverflow'),
+ (get_string('userstatsactivity', 'moodleoverflow') . $this->helpactivity->object),
+ get_string('userstatsreputation', 'moodleoverflow')]);
+ $this->get_table_data();
+ $this->sortable(true, 'reputation', SORT_DESC);
+ $this->no_sorting('username');
+ $this->setup();
+ }
+
+ /**
+ * Method to display the table.
+ * @return void
+ */
+ public function out() {
+ global $DB;
+ $this->start_output();
+ $this->sort_table_data($this->get_sort_order());
+ $this->format_and_add_array_of_rows($this->userstatsdata, true);
+ $this->text_sorting('reputation');
+ $this->finish_output();
+ }
+
+ /**
+ * Method to sort the userstatsdata-table.
+ */
+ private function sort_table_data($sortorder) {
+ $key = $sortorder['sortby'];
+ // The index of each object in usertable is it's value of $key.
+ $length = count($this->userstatsdata);
+ if ($sortorder['sortorder'] == 4) {
+ // 4 means sort in ascending order.
+ $this->quick_usertable_sort(0, $length - 1, $key, 'asc');
+ } else if ($sortorder['sortorder'] == 3) {
+ // 3 means sort in descending order.
+ $this->quick_usertable_sort(0, $length - 1, $key, 'desc');
+ }
+ }
+
+ /**
+ * Sorts userstatsdata with quicksort algorithm.
+ */
+ private function quick_usertable_sort($low, $high, $key, $order) {
+ if ($low >= $high) {
+ return;
+ }
+ $left = $low;
+ $right = $high;
+ $pivot = $this->userstatsdata[intval(($low + $high) / 2)];
+ $pivot = $pivot->$key;
+ do {
+ if ($order == 'asc') {
+ while ($this->userstatsdata[$left]->$key < $pivot) {
+ $left++;
+ }
+ while ($this->userstatsdata[$right]->$key > $pivot) {
+ $right--;
+ }
+ } else if ($order == 'desc') {
+ while ($this->userstatsdata[$left]->$key > $pivot) {
+ $left++;
+ }
+ while ($this->userstatsdata[$right]->$key < $pivot) {
+ $right--;
+ }
+ }
+ if ($left <= $right) {
+ $temp = $this->userstatsdata[$right];
+ $this->userstatsdata[$right] = $this->userstatsdata[$left];
+ $this->userstatsdata[$left] = $temp;
+ $right--;
+ $left++;
+ }
+ } while ($left <= $right);
+ if ($low < $right) {
+ if ($order == 'asc') {
+ $this->quick_usertable_sort($low, $right, $key, 'asc');
+ } else if ($order == 'desc') {
+ $this->quick_usertable_sort($low, $right, $key, 'desc');
+ }
+ }
+ if ($high > $left) {
+ if ($order == 'asc') {
+ $this->quick_usertable_sort($left, $high, $key, 'desc');
+ } else if ($order == 'desc') {
+ $this->quick_usertable_sort($left, $high, $key, 'desc');
+ }
+ }
+ }
+
+ /**
+ * Method to collect all the data.
+ * Method will collect all users from the given course and will determine the user statistics
+ *
+ * @return 2d-array with user statistic
+ */
+ public function get_table_data() {
+ global $DB;
+ // Get all userdata from a course.
+ $context = \context_course::instance($this->courseid);
+ $users = get_enrolled_users($context , '', 0, $userfields = 'u.id, u.firstname, u.lastname');
+
+ // Step 1.0: Build the datatable with all relevant Informations.
+ $sqlquery = 'SELECT (ROW_NUMBER() OVER (ORDER BY ratings.id)) AS row_num,
+ ratings.id AS rateid,
+ discuss.userid AS discussuserid,
+ posts.id AS postid,
+ posts.userid AS postuserid,
+ ratings.rating AS rating,
+ ratings.userid AS rateuserid,
+ ratings.postid AS ratepostid,
+ discuss.id AS discussid,
+ posts.discussion AS postdiscussid,
+ ratings.discussionid AS ratediscussid
+ FROM {moodleoverflow_discussions} discuss
+ LEFT JOIN {moodleoverflow_posts} posts ON discuss.id = posts.discussion
+ LEFT JOIN {moodleoverflow_ratings} ratings ON posts.id = ratings.postid
+ WHERE discuss.course = ' . $this->courseid . ';';
+ $ratingdata = $DB->get_records_sql($sqlquery);
+
+ // Step 2.0: Now collect the data for every user in the course.
+ foreach ($users as $user) {
+ $student = new \stdClass();
+ $student->id = $user->id;
+ $student->name = $user->firstname . ' ' . $user->lastname;
+ $linktostudent = new \moodle_url('/user/view.php', array('id' => $student->id, 'course' => $this->courseid));
+ $student->link = \html_writer::link($linktostudent->out(), $student->name);
+ $student->submittedposts = array(); // Key = postid, Value = postid.
+ $student->ratedposts = array(); // Key = rateid, Value = rateid.
+ $student->receivedupvotes = 0;
+ $student->receiveddownvotes = 0;
+ $student->activity = 0;
+ $student->reputation = 0;
+ foreach ($ratingdata as $row) {
+ if ($row->postuserid !== $student->id && $row->rateuserid !== $student->id) {
+ continue;
+ }
+ if ($row->postuserid == $student->id && $row->rating == RATING_UPVOTE) {
+ $student->receivedupvotes += 1;
+ }
+ if ($row->postuserid == $student->id && $row->rating == RATING_DOWNVOTE) {
+ $student->receiveddownvotes += 1;
+ }
+ if ($row->rateuserid == $student->id && !array_key_exists($row->rateid, $student->ratedposts)) {
+ $student->activity += 1;
+ $student->ratedposts[$row->rateid] = $row->rateid;
+ }
+ if ($row->postuserid == $student->id && !array_key_exists($row->postid, $student->submittedposts)) {
+ $student->activity += 1;
+ $student->submittedposts[$row->postid] = $row->postid;
+ }
+ }
+ // Get the user reputation from the course.
+ $student->reputation = \mod_moodleoverflow\ratings::moodleoverflow_get_reputation($this->moodleoverflowid,
+ $student->id);
+ array_push($this->userstatsdata, $student);
+ }
+ }
+
+ /**
+ * Return the userstatsdata-table.
+ */
+ public function get_usertable() {
+ return $this->userstatsdata;
+ }
+
+ /**
+ * Setup the help icon for amount of activity
+ */
+ public function set_helpactivity() {
+ global $CFG;
+ $this->helpactivity = new \stdClass();
+ $this->helpactivity->iconurl = $CFG->wwwroot . '/pix/a/help.png';
+ $this->helpactivity->icon = \html_writer::img($this->helpactivity->iconurl,
+ get_string('helpamountofactivity', 'moodleoverflow'));
+ $this->helpactivity->class = 'helpactivityclass btn btn-link';
+ $this->helpactivity->iconattributes = array('role' => 'button',
+ 'data-container' => 'body',
+ 'data-toggle' => 'popover',
+ 'data-placement' => 'right',
+ 'data-action' => 'showhelpicon',
+ 'data-html' => 'true',
+ 'data-trigger' => 'focus',
+ 'tabindex' => '0',
+ 'data-content' => '
' .
+ get_string('helpamountofactivity', 'moodleoverflow') .
+ '
');
+
+ $this->helpactivity->object = \html_writer::span($this->helpactivity->icon,
+ $this->helpactivity->class,
+ $this->helpactivity->iconattributes);
+ }
+
+ // Functions that show the data.
+ public function col_username($row) {
+ return $row->link;
+ }
+
+ public function col_receivedupvotes($row) {
+ if ($row->receivedupvotes > 0) {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-success') .
+ $row->receivedupvotes . \html_writer::end_span());
+ } else {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-warning') .
+ $row->receivedupvotes . \html_writer::end_span());
+ }
+ }
+
+ public function col_receiveddownvotes($row) {
+ if ($row->receiveddownvotes > 0) {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-success') .
+ $row->receiveddownvotes . \html_writer::end_span());
+ } else {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-warning') .
+ $row->receiveddownvotes . \html_writer::end_span());
+ }
+ }
+
+ public function col_activity($row) {
+ if ($row->activity > 0) {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-success') .
+ $row->activity . \html_writer::end_span());
+ } else {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-warning') .
+ $row->activity . \html_writer::end_span());
+ }
+ }
+
+ public function col_reputation($row) {
+ if ($row->reputation > 0) {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-success') .
+ $row->reputation . \html_writer::end_span());
+ } else {
+ return \html_writer::tag('h5', \html_writer::start_span('badge badge-warning') .
+ $row->reputation . \html_writer::end_span());
+ }
+ }
+
+ public function other_cols($colname, $attempt) {
+ return null;
+ }
+}
diff --git a/db/access.php b/db/access.php
index 6e5fee1b3e..4faa51067c 100644
--- a/db/access.php
+++ b/db/access.php
@@ -211,4 +211,17 @@
'manager' => CAP_ALLOW
),
),
+
+ 'mod/moodleoverflow:viewanyrating' => array(
+ 'riskbitmask' => RISK_PERSONAL,
+ 'captype' => 'read',
+ 'contextlevel' => CONTEXT_MODULE,
+ 'archetypes' => array(
+ 'teacher' => CAP_ALLOW,
+ 'editingteacher' => CAP_ALLOW,
+ 'manager' => CAP_ALLOW
+ ),
+ 'clonepermissionsfrom' => 'mod/forum:viewanyrating'
+ ),
+
);
diff --git a/lang/en/moodleoverflow.php b/lang/en/moodleoverflow.php
index d92b8a37c0..871a4ded87 100644
--- a/lang/en/moodleoverflow.php
+++ b/lang/en/moodleoverflow.php
@@ -62,6 +62,7 @@
// Strings for the locallib.php.
$string['addanewdiscussion'] = 'Add a new discussion topic';
+$string['seeuserstats'] = 'View user statistics';
$string['nodiscussions'] = 'There are no discussion topics yet in this forum.';
$string['markallread_forum'] = 'Mark all posts as read';
$string['markallread'] = 'Mark all posts in this discussion as read';
@@ -199,6 +200,13 @@
$string['markmoodleoverflowreadsuccessful'] = 'All posts have been marked as read.';
$string['noguesttracking'] = 'Sorry, guests are not allowed to set tracking options.';
+// Strings for the userstats_table.php.
+$string['userstatsupvotes'] = 'Received upvotes';
+$string['userstatsdownvotes'] = 'Received downvotes';
+$string['userstatsactivity'] = 'Amount of activity';
+$string['userstatsreputation'] = 'User reputation';
+$string['helpamountofactivity'] = 'Each actitivy like writing a post, starting a discussion or giving a rating gives 1 point';
+
// OTHER.
$string['messageprovider:posts'] = 'Notification of new posts';
$string['unknownerror'] = 'This is not expected to happen.';
diff --git a/locallib.php b/locallib.php
index 64b8589ab0..b120deaa22 100644
--- a/locallib.php
+++ b/locallib.php
@@ -137,6 +137,7 @@ function moodleoverflow_print_latest_discussions($moodleoverflow, $cm, $page = -
$canstartdiscussion = moodleoverflow_user_can_post_discussion($moodleoverflow, $cm, $context);
$canviewdiscussions = has_capability('mod/moodleoverflow:viewdiscussion', $context);
$canreviewposts = has_capability('mod/moodleoverflow:reviewpost', $context);
+ $canseeuserstats = has_capability('mod/moodleoverflow:viewanyrating', $context);
// Print a button if the user is capable of starting
// a new discussion or if the selfenrol is aviable.
@@ -149,6 +150,33 @@ function moodleoverflow_print_latest_discussions($moodleoverflow, $cm, $page = -
echo $OUTPUT->render($button);
}
+ // Print a button if the user is capable of seeing the user stats.
+ if ($canseeuserstats) {
+ $userstatsbuttontext = get_string('seeuserstats', 'moodleoverflow');
+ $userstatsbuttonurl = new moodle_url('/mod/moodleoverflow/userstats.php', ['id' => $cm->id,
+ 'courseid' => $moodleoverflow->course,
+ 'mid' => $moodleoverflow->id]);
+ $userstatsbutton = new single_button($userstatsbuttonurl, $userstatsbuttontext, 'get');
+ $userstatsbutton->class = 'singlebutton align-middle m-2';
+ echo $OUTPUT->render($userstatsbutton);
+ }
+
+ // Get all the recent discussions the user is allowed to see.
+ $discussions = moodleoverflow_get_discussions($cm, $page, $perpage);
+
+ // If we want paging.
+ if ($page != -1) {
+
+ // Get the number of discussions.
+ $numberofdiscussions = moodleoverflow_get_discussions_count($cm);
+
+ // Show the paging bar.
+ echo $OUTPUT->paging_bar($numberofdiscussions, $page, $perpage, "view.php?id=$cm->id");
+ }
+
+ // Get the number of replies for each discussion.
+ $replies = moodleoverflow_count_discussion_replies($cm);
+
// Check whether the moodleoverflow instance can be tracked and is tracked.
if ($cantrack = \mod_moodleoverflow\readtracking::moodleoverflow_can_track_moodleoverflows($moodleoverflow)) {
$istracked = \mod_moodleoverflow\readtracking::moodleoverflow_is_tracked($moodleoverflow);
diff --git a/styles.css b/styles.css
index d3b3f6915b..c6c3c8fbf7 100644
--- a/styles.css
+++ b/styles.css
@@ -528,6 +528,12 @@
background-color: #ffd3d3;
}
+
+.moodleoverflow-statistics-table .header.c3 .helpactivityclass {
+ padding: 0;
+ margin-left: 8px;
+}
+
.moodleoverflow-gap-small {
gap: 8px;
}
diff --git a/tests/dailymail_test.php b/tests/dailymail_test.php
index b6afe4d54e..b6df80f5d1 100644
--- a/tests/dailymail_test.php
+++ b/tests/dailymail_test.php
@@ -127,7 +127,6 @@ private function helper_run_send_mails() {
* Test if the task send_daily_mail sends a mail to the user.
*/
public function test_mail_delivery() {
- global $DB;
// Create user with maildigest = on.
$this->helper_create_user_and_discussion('1');
@@ -145,7 +144,6 @@ public function test_mail_delivery() {
* Test if the content of the mail matches the supposed content.
*/
public function test_content_of_mail_delivery() {
- global $DB;
// Create user with maildigest = on.
$this->helper_create_user_and_discussion('1');
@@ -180,7 +178,6 @@ public function test_content_of_mail_delivery() {
* Test if the task does not send a mail when maildigest = 0
*/
public function test_mail_not_send() {
- global $DB;
// Creat user with daily_mail = off.
$this->helper_create_user_and_discussion('0');
@@ -197,7 +194,7 @@ public function test_mail_not_send() {
*/
public function test_records_removed() {
global $DB;
- // create user with maildigest = on.
+ // Create user with maildigest = on.
$this->helper_create_user_and_discussion('1');
// Now send the mails.
diff --git a/tests/userstats_test.php b/tests/userstats_test.php
new file mode 100644
index 0000000000..ca657f7d3c
--- /dev/null
+++ b/tests/userstats_test.php
@@ -0,0 +1,256 @@
+.
+
+/**
+ * The module moodleoverflow tests.
+ *
+ * @package mod_moodleoverflow
+ * @copyright 2023 Tamaro Walter
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+namespace mod_moodleoverflow;
+use mod_moodleoverflow\tables\userstats_table;
+
+defined('MOODLE_INTERNAL') || die();
+
+
+global $CFG;
+require_once($CFG->dirroot . '/mod/moodleoverflow/lib.php');
+class userstats_test extends \advanced_testcase {
+
+ private $course;
+ private $coursemodule;
+ private $context;
+ private $moodleoverflow;
+ private $teacher;
+ private $user1;
+ private $user2;
+ private $discussion1; // Discussion from user1.
+ private $discussion2; // Discussion from user2.
+ private $post1; // First post from discussion1.
+ private $post2; // First post from discussion2.
+ private $answer1; // Answerpost to discussion1 from user2.
+ private $answer2; // Answerpost to discussion2 from user1.
+ private $generator; // Generator for moodleoverflow.
+
+ /**
+ * Test setUp.
+ */
+ public function setUp(): void {
+ $this->resetAfterTest();
+ $this->helper_course_set_up();
+ }
+
+ /**
+ * Test tearDown.
+ */
+ public function tearDown(): void {
+ // Clear all caches.
+ \mod_moodleoverflow\subscriptions::reset_moodleoverflow_cache();
+ \mod_moodleoverflow\subscriptions::reset_discussion_cache();
+ }
+
+ // Begin of test functions.
+
+ /**
+ * Test, if a upvote is being counted.
+ */
+ public function test_upvote() {
+ // Teacher upvotes the discussion and the answer of user2.
+ $this->create_upvote($this->teacher, $this->discussion1[1], $this->answer1);
+
+ // Create the user statistics table for this course and save it in $data.
+ $data = $this->create_statstable();
+ foreach ($data as $student) {
+ if ($student->id == $this->user2->id) {
+ $upvotes = $student->receivedupvotes;
+ }
+ }
+ $this->assertEquals(1, $upvotes);
+ }
+
+ /**
+ * Test, if a downvote is being counted.
+ */
+ public function test_downvote() {
+ // Teacher downvotes the discussion and the answer of user1.
+ $this->create_downvote($this->teacher, $this->discussion2[1], $this->answer2);
+
+ // Create the user statistics table for this course and save it in $data.
+ $data = $this->create_statstable();
+ foreach ($data as $student) {
+ if ($student->id == $this->user1->id) {
+ $downvotes = $student->receiveddownvotes;
+ }
+ }
+ $this->assertEquals(1, $downvotes);
+ }
+
+ /**
+ * Test, if the activity is calculated correctly.
+ */
+ public function test_activity() {
+ // User1 will rates 3 times.
+ $this->create_helpful($this->user1, $this->discussion1[1], $this->answer1);
+ $this->create_upvote($this->user1, $this->discussion1[1], $this->answer1);
+ $this->create_downvote($this->user1, $this->discussion2[1], $this->post2);
+ // User1 created 2 posts (1 discussion, 1 answer).
+ // Activity = 5.
+ // Create the user statistics table for this course and save it in $data.
+ $data = $this->create_statstable();
+ foreach ($data as $student) {
+ if ($student->id == $this->user1->id) {
+ $activity = $student->activity;
+ }
+ }
+ $this->assertEquals(5, $activity);
+
+ }
+ /**
+ * Test, if the reputation is calculated correctly.
+ */
+ public function test_reputation() {
+ // User1 creates some ratings for user2, Teacher creates some ratings for user2.
+ $this->create_helpful($this->user1, $this->discussion1[1], $this->answer1);
+ $this->create_upvote($this->user1, $this->discussion1[1], $this->answer1);
+ $this->create_downvote($this->user1, $this->discussion2[1], $this->post2);
+ $this->create_solution($this->teacher, $this->discussion1[1], $this->answer1);
+
+ // Calculate the reputation of user2.
+ $reputation = \mod_moodleoverflow\ratings::moodleoverflow_get_reputation($this->moodleoverflow->id, $this->user2->id);
+ // Create the user statistics table for this course and save it in $data.
+ $data = $this->create_statstable();
+ foreach ($data as $student) {
+ if ($student->id == $this->user2->id) {
+ $reputation2 = $student->reputation;
+ }
+ }
+ $this->assertEquals($reputation, $reputation2);
+ }
+
+ // Helper functions.
+
+ /**
+ * This function creates:
+ * - a course with a moodleoverflow
+ * - a teacher
+ * - 2 users, which create a discussion and a post in the discussion of the other user.
+ */
+ private function helper_course_set_up() {
+ global $DB;
+ // Create a new course with a moodleoverflow forum.
+ $this->course = $this->getDataGenerator()->create_course();
+ $location = array('course' => $this->course->id);
+ $this->moodleoverflow = $this->getDataGenerator()->create_module('moodleoverflow', $location);
+ $this->coursemodule = get_coursemodule_from_instance('moodleoverflow', $this->moodleoverflow->id);
+ $this->context = \context_course::instance($this->course->id);
+
+ // Create a teacher.
+ $this->teacher = $this->getDataGenerator()->create_user(array('firstname' => 'Tamaro', 'lastname' => 'Walter'));
+ $this->getDataGenerator()->enrol_user($this->teacher->id, $this->course->id, 'student');
+
+ // Create 2 users and their discussions and posts.
+ $this->user1 = $this->getDataGenerator()->create_user(array('firstname' => 'Ava', 'lastname' => 'Davis'));
+ $this->getDataGenerator()->enrol_user($this->user1->id, $this->course->id, 'student');
+ $this->user2 = $this->getDataGenerator()->create_user(array('firstname' => 'Ethan', 'lastname' => 'Brown'));
+ $this->getDataGenerator()->enrol_user($this->user2->id, $this->course->id, 'student');
+
+ $this->generator = $this->getDataGenerator()->get_plugin_generator('mod_moodleoverflow');
+ $this->discussion1 = $this->generator->post_to_forum($this->moodleoverflow, $this->user1);
+ $this->discussion2 = $this->generator->post_to_forum($this->moodleoverflow, $this->user2);
+ $this->post1 = $DB->get_record('moodleoverflow_posts', array('id' => $this->discussion1[0]->firstpost), '*');
+ $this->post2 = $DB->get_record('moodleoverflow_posts', array('id' => $this->discussion1[0]->firstpost), '*');
+ $this->answer1 = $this->generator->reply_to_post($this->discussion1[1], $this->user2, true);
+ $this->answer2 = $this->generator->reply_to_post($this->discussion2[1], $this->user1, true);
+ }
+
+
+ /**
+ * Create a usertable and return it.
+ */
+ private function create_statstable() {
+ $url = new \moodle_url('/mod/moodleoverflow/userstats.php', ['id' => $this->coursemodule->id,
+ 'courseid' => $this->course->id,
+ 'mid' => $this->moodleoverflow->id]);
+ $userstatstable = new userstats_table('testtable', $this->course->id, $this->moodleoverflow->id, $url);
+ $userstatstable->get_table_data();
+ return $userstatstable->get_usertable();
+ }
+
+ /**
+ * Create a upvote to a post in an existing discussion.
+ */
+ private function create_upvote($author, $discussion, $post) {
+ $record = (object) [
+ 'moodleoverflowid' => $this->moodleoverflow->id,
+ 'discussionid' => $discussion->id,
+ 'userid' => $author->id,
+ 'postid' => $post->id,
+ 'rating' => 2,
+ 'firstrated' => time(),
+ 'lastchanged' => time()
+ ];
+ return $this->generator->create_rating($record);
+ }
+
+ /**
+ * Create a downvote to a post in an existing discussion.
+ */
+ private function create_downvote($author, $discussion, $post) {
+ $record = (object) [
+ 'moodleoverflowid' => $this->moodleoverflow->id,
+ 'discussionid' => $discussion->id,
+ 'userid' => $author->id,
+ 'postid' => $post->id,
+ 'rating' => 1,
+ 'firstrated' => time(),
+ 'lastchanged' => time()
+ ];
+ return $this->generator->create_rating($record);
+ }
+
+ /**
+ * Create a helpful rating to a post in an existing discussion.
+ */
+ private function create_helpful($author, $discussion, $post) {
+ $record = (object) [
+ 'moodleoverflowid' => $this->moodleoverflow->id,
+ 'discussionid' => $discussion->id,
+ 'userid' => $author->id,
+ 'postid' => $post->id,
+ 'rating' => 3,
+ 'firstrated' => time(),
+ 'lastchanged' => time()
+ ];
+ return $this->generator->create_rating($record);
+ }
+
+ /**
+ * Create a solution rating to a post in an existing discussion.
+ */
+ private function create_solution($author, $discussion, $post) {
+ $record = (object) [
+ 'moodleoverflowid' => $this->moodleoverflow->id,
+ 'discussionid' => $discussion->id,
+ 'userid' => $author->id,
+ 'postid' => $post->id,
+ 'rating' => 4,
+ 'firstrated' => time(),
+ 'lastchanged' => time()
+ ];
+ return $this->generator->create_rating($record);
+ }
+}
diff --git a/userstats.php b/userstats.php
new file mode 100644
index 0000000000..e9d42e37b0
--- /dev/null
+++ b/userstats.php
@@ -0,0 +1,70 @@
+.
+
+/**
+ * Prints a particular instance of moodleoverflow
+ *
+ * You can have a rather longer description of the file as well,
+ * if you like, and it can span multiple lines.
+ *
+ * @package mod_moodleoverflow
+ * @copyright 2023 Tamaro Walter
+ * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
+ */
+
+// Include config and locallib.
+require_once(__DIR__.'/../../config.php');
+global $CFG, $PAGE, $DB, $OUTPUT, $SESSION;
+require_once($CFG->dirroot . '/mod/moodleoverflow/locallib.php');
+
+use mod_moodleoverflow\tables\userstats_table;
+// Declare optional parameters.
+$cmid = required_param('id', PARAM_INT); // Course Module ID.
+$courseid = required_param('courseid', PARAM_INT); // Course ID.
+$mid = required_param('mid', PARAM_INT); // Moodleoveflow ID, Moodleoverflow that started the statistics.
+
+// Define important variables.
+if ($courseid) {
+ $course = $DB->get_record('course', array('id' => $courseid), '*');
+}
+if ($cmid) {
+ $cm = get_coursemodule_from_id('moodleoverflow', $cmid, $course->id, false, MUST_EXIST);
+}
+if ($mid) {
+ $moodleoverflow = $DB->get_record('moodleoverflow', array('id' => $mid), '*');
+}
+// Require a login.
+require_login($course, true, $cm);
+
+// Set the context.
+$context = context_module::instance($cm->id);
+$PAGE->set_context($context);
+
+// Do a capability check, in case a user iserts the userstats-url manually.
+if (has_capability('mod/moodleoverflow:viewanyrating', $context)) {
+ // Print the page header.
+ $PAGE->set_url('/mod/moodleoverflow/userstats.php', array('id' => $cm->id,
+ 'courseid' => $course->id, 'mid' => $moodleoverflow->id));
+ $PAGE->set_title(format_string('User statistics'));
+ $PAGE->set_heading(format_string('User statistics of course: ' . $course->fullname));
+
+ // Output starts here.
+ echo $OUTPUT->header();
+ echo $OUTPUT->heading('');
+ $table = new userstats_table('statisticstable' , $course->id, $moodleoverflow->id, $PAGE->url);
+ echo $table->out();
+ echo $OUTPUT->footer();
+}
diff --git a/version.php b/version.php
index d630f8cafc..45384c4fd9 100644
--- a/version.php
+++ b/version.php
@@ -28,7 +28,7 @@
defined('MOODLE_INTERNAL') || die();
$plugin->component = 'mod_moodleoverflow';
-$plugin->version = 2023022406;
+$plugin->version = 2023050801;
$plugin->release = 'v4.1-r1';
$plugin->requires = 2020061500; // Requires Moodle 3.9+.
$plugin->maturity = MATURITY_STABLE;
diff --git a/view.php b/view.php
index ad62eca45b..52759e13ba 100644
--- a/view.php
+++ b/view.php
@@ -36,6 +36,7 @@
$page = optional_param('page', 0, PARAM_INT); // Which page to show.
$movetopopup = optional_param('movetopopup', 0, PARAM_INT); // Which Topic to move.
$linktoforum = optional_param('movetoforum', 0, PARAM_INT); // Forum to which it is moved.
+
// Set the parameters.
$params = array();
if ($id) {
@@ -61,8 +62,6 @@
throw new moodle_exception('missingparameter');
}
-
-
// Require a login.
require_login($course, true, $cm);