{"version":3,"file":"cm.min.js","sources":["https:\/\/moodle-test3.univ-paris1.fr\/course\/format\/amd\/src\/local\/courseindex\/cm.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 * Course index cm component.\n *\n * This component is used to control specific course modules interactions like drag and drop.\n *\n * @module core_courseformat\/local\/courseindex\/cm\n * @class core_courseformat\/local\/courseindex\/cm\n * @copyright 2021 Ferran Recio \n * @license http:\/\/www.gnu.org\/copyleft\/gpl.html GNU GPL v3 or later\n *\/\n\nimport DndCmItem from 'core_courseformat\/local\/courseeditor\/dndcmitem';\nimport Templates from 'core\/templates';\nimport Prefetch from 'core\/prefetch';\nimport Config from 'core\/config';\nimport Pending from \"core\/pending\";\n\n\/\/ Prefetch the completion icons template.\nconst completionTemplate = 'core_courseformat\/local\/courseindex\/cmcompletion';\nPrefetch.prefetchTemplate(completionTemplate);\n\nexport default class Component extends DndCmItem {\n\n \/**\n * Constructor hook.\n *\/\n create() {\n \/\/ Optional component name for debugging.\n this.name = 'courseindex_cm';\n \/\/ Default query selectors.\n this.selectors = {\n CM_NAME: `[data-for='cm_name']`,\n CM_COMPLETION: `[data-for='cm_completion']`,\n };\n \/\/ Default classes to toggle on refresh.\n this.classes = {\n CMHIDDEN: 'dimmed',\n LOCKED: 'editinprogress',\n RESTRICTIONS: 'restrictions',\n PAGEITEM: 'pageitem',\n INDENTED: 'indented',\n };\n \/\/ We need our id to watch specific events.\n this.id = this.element.dataset.id;\n }\n\n \/**\n * Static method to create a component instance form the mustache template.\n *\n * @param {element|string} target the DOM main element or its ID\n * @param {object} selectors optional css selector overrides\n * @return {Component}\n *\/\n static init(target, selectors) {\n return new this({\n element: document.getElementById(target),\n selectors,\n });\n }\n\n \/**\n * Initial state ready method.\n *\n * @param {Object} state the course state.\n *\/\n stateReady(state) {\n this.configDragDrop(this.id);\n const cm = state.cm.get(this.id);\n const course = state.course;\n \/\/ Refresh completion icon.\n this._refreshCompletion({\n state,\n element: cm,\n });\n const url = new URL(window.location.href);\n const anchor = url.hash.replace('#', '');\n \/\/ Check if the current url is the cm url.\n if (window.location.href == cm.url\n || (window.location.href.includes(course.baseurl) && anchor == cm.anchor)\n ) {\n this.element.scrollIntoView({block: \"center\"});\n }\n \/\/ Check if this we are displaying this activity page.\n if (Config.contextid != Config.courseContextId && Config.contextInstanceId == this.id) {\n this.reactive.dispatch('setPageItem', 'cm', this.id, true);\n this.element.scrollIntoView({block: \"center\"});\n }\n \/\/ Add anchor logic if the element is not user visible or the element hasn't URL.\n if (!cm.uservisible || !cm.url) {\n const element = this.getElement(this.selectors.CM_NAME);\n this.addEventListener(\n element,\n 'click',\n this._activityAnchor,\n );\n \/\/ If the element is not user visible we also need to update the anchor link including the section page.\n if (!document.getElementById(cm.anchor)) {\n element.setAttribute('href', this._getActivitySectionURL(cm));\n }\n }\n }\n\n \/**\n * Component watchers.\n *\n * @returns {Array} of watchers\n *\/\n getWatchers() {\n return [\n {watch: `cm[${this.id}]:deleted`, handler: this.remove},\n {watch: `cm[${this.id}]:updated`, handler: this._refreshCm},\n {watch: `cm[${this.id}].completionstate:updated`, handler: this._refreshCompletion},\n {watch: `course.pageItem:updated`, handler: this._refreshPageItem},\n ];\n }\n\n \/**\n * Update a course index cm using the state information.\n *\n * @param {object} param\n * @param {Object} param.element details the update details.\n *\/\n _refreshCm({element}) {\n \/\/ Update classes.\n this.element.classList.toggle(this.classes.CMHIDDEN, !element.visible);\n this.getElement(this.selectors.CM_NAME).innerHTML = element.name;\n this.element.classList.toggle(this.classes.DRAGGING, element.dragging ?? false);\n this.element.classList.toggle(this.classes.LOCKED, element.locked ?? false);\n this.element.classList.toggle(this.classes.RESTRICTIONS, element.hascmrestrictions ?? false);\n this.element.classList.toggle(this.classes.INDENTED, element.indent);\n this.locked = element.locked;\n }\n\n \/**\n * Handle a page item update.\n *\n * @param {Object} details the update details\n * @param {Object} details.element the course state data.\n *\/\n _refreshPageItem({element}) {\n if (!element.pageItem) {\n return;\n }\n const isPageId = (element.pageItem.type == 'cm' && element.pageItem.id == this.id);\n this.element.classList.toggle(this.classes.PAGEITEM, isPageId);\n if (isPageId && !this.reactive.isEditing) {\n this.element.scrollIntoView({block: \"nearest\"});\n }\n }\n\n \/**\n * Update the activity completion icon.\n *\n * @param {Object} details the update details\n * @param {Object} details.state the state data\n * @param {Object} details.element the element data\n *\/\n async _refreshCompletion({state, element}) {\n \/\/ No completion icons are displayed in edit mode.\n if (this.reactive.isEditing || !element.istrackeduser) {\n return;\n }\n \/\/ Check if the completion value has changed.\n const completionElement = this.getElement(this.selectors.CM_COMPLETION);\n if (!completionElement || completionElement.dataset.value == element.completionstate) {\n return;\n }\n\n \/\/ Collect section information from the state.\n const exporter = this.reactive.getExporter();\n const data = exporter.cmCompletion(state, element);\n\n const {html, js} = await Templates.renderForPromise(completionTemplate, data);\n Templates.replaceNode(completionElement, html, js);\n }\n\n \/**\n * The activity anchor event.\n *\n * @param {Event} event\n *\/\n _activityAnchor(event) {\n const cm = this.reactive.get('cm', this.id);\n \/\/ If the user cannot access the element but the element is present in the page\n \/\/ the new url should be an anchor link.\n const element = document.getElementById(cm.anchor);\n if (element) {\n \/\/ Make sure the section is expanded.\n this.reactive.dispatch('sectionContentCollapsed', [cm.sectionid], false);\n \/\/ Marc the element as page item once the event is handled.\n const pendingAnchor = new Pending(`courseformat\/activity:openAnchor`);\n setTimeout(() => {\n this.reactive.dispatch('setPageItem', 'cm', cm.id);\n pendingAnchor.resolve();\n }, 50);\n return;\n }\n \/\/ If the element is not present in the page we need to go to the specific section.\n event.preventDefault();\n window.location = this._getActivitySectionURL(cm);\n }\n\n \/**\n * Get the anchor link in section page for the cm.\n *\n * @param {Object} cm the course module data.\n * @return {String} the anchor link.\n *\/\n _getActivitySectionURL(cm) {\n let section = this.reactive.get('section', cm.sectionid);\n\n \/\/ If the section is delegated get its parent section if it has one.\n if (section.component && section.parentsectionid) {\n section = this.reactive.get('section', section.parentsectionid);\n }\n\n if (!section) {\n return '#';\n }\n\n return `${section.sectionurl}#${cm.anchor}`;\n }\n}\n"],"names":["prefetchTemplate","Component","DndCmItem","create","name","selectors","CM_NAME","CM_COMPLETION","classes","CMHIDDEN","LOCKED","RESTRICTIONS","PAGEITEM","INDENTED","id","this","element","dataset","target","document","getElementById","stateReady","state","configDragDrop","cm","get","course","_refreshCompletion","anchor","URL","window","location","href","hash","replace","url","includes","baseurl","scrollIntoView","block","Config","contextid","courseContextId","contextInstanceId","reactive","dispatch","uservisible","getElement","addEventListener","_activityAnchor","setAttribute","_getActivitySectionURL","getWatchers","watch","handler","remove","_refreshCm","_refreshPageItem","classList","toggle","visible","innerHTML","DRAGGING","dragging","locked","hascmrestrictions","indent","pageItem","isPageId","type","isEditing","istrackeduser","completionElement","value","completionstate","data","getExporter","cmCompletion","html","js","Templates","renderForPromise","replaceNode","event","sectionid","pendingAnchor","Pending","setTimeout","resolve","preventDefault","section","component","parentsectionid","sectionurl"],"mappings":";;;;;;;;;;iUAkCSA,iBADkB,0DAGNC,kBAAkBC,mBAKnCC,cAESC,KAAO,sBAEPC,UAAY,CACbC,+BACAC,iDAGCC,QAAU,CACXC,SAAU,SACVC,OAAQ,iBACRC,aAAc,eACdC,SAAU,WACVC,SAAU,iBAGTC,GAAKC,KAAKC,QAAQC,QAAQH,eAUvBI,OAAQb,kBACT,IAAIU,KAAK,CACZC,QAASG,SAASC,eAAeF,QACjCb,UAAAA,YASRgB,WAAWC,YACFC,eAAeR,KAAKD,UACnBU,GAAKF,MAAME,GAAGC,IAAIV,KAAKD,IACvBY,OAASJ,MAAMI,YAEhBC,mBAAmB,CACpBL,MAAAA,MACAN,QAASQ,WAGPI,OADM,IAAIC,IAAIC,OAAOC,SAASC,MACjBC,KAAKC,QAAQ,IAAK,QAEjCJ,OAAOC,SAASC,MAAQR,GAAGW,KACvBL,OAAOC,SAASC,KAAKI,SAASV,OAAOW,UAAYT,QAAUJ,GAAGI,cAE7DZ,QAAQsB,eAAe,CAACC,MAAO,WAGpCC,gBAAOC,WAAaD,gBAAOE,iBAAmBF,gBAAOG,mBAAqB5B,KAAKD,UAC1E8B,SAASC,SAAS,cAAe,KAAM9B,KAAKD,IAAI,QAChDE,QAAQsB,eAAe,CAACC,MAAO,aAGnCf,GAAGsB,cAAgBtB,GAAGW,IAAK,OACtBnB,QAAUD,KAAKgC,WAAWhC,KAAKV,UAAUC,cAC1C0C,iBACDhC,QACA,QACAD,KAAKkC,iBAGJ9B,SAASC,eAAeI,GAAGI,SAC5BZ,QAAQkC,aAAa,OAAQnC,KAAKoC,uBAAuB3B,MAUrE4B,oBACW,CACH,CAACC,mBAAatC,KAAKD,gBAAewC,QAASvC,KAAKwC,QAChD,CAACF,mBAAatC,KAAKD,gBAAewC,QAASvC,KAAKyC,YAChD,CAACH,mBAAatC,KAAKD,gCAA+BwC,QAASvC,KAAKY,oBAChE,CAAC0B,gCAAkCC,QAASvC,KAAK0C,mBAUzDD,iFAAWxC,QAACA,mBAEHA,QAAQ0C,UAAUC,OAAO5C,KAAKP,QAAQC,UAAWO,QAAQ4C,cACzDb,WAAWhC,KAAKV,UAAUC,SAASuD,UAAY7C,QAAQZ,UACvDY,QAAQ0C,UAAUC,OAAO5C,KAAKP,QAAQsD,mCAAU9C,QAAQ+C,+DACxD\/C,QAAQ0C,UAAUC,OAAO5C,KAAKP,QAAQE,+BAAQM,QAAQgD,yDACtDhD,QAAQ0C,UAAUC,OAAO5C,KAAKP,QAAQG,2CAAcK,QAAQiD,gFAC5DjD,QAAQ0C,UAAUC,OAAO5C,KAAKP,QAAQK,SAAUG,QAAQkD,aACxDF,OAAShD,QAAQgD,OAS1BP,4BAAiBzC,QAACA,mBACTA,QAAQmD,sBAGPC,SAAqC,MAAzBpD,QAAQmD,SAASE,MAAgBrD,QAAQmD,SAASrD,IAAMC,KAAKD,QAC1EE,QAAQ0C,UAAUC,OAAO5C,KAAKP,QAAQI,SAAUwD,UACjDA,WAAarD,KAAK6B,SAAS0B,gBACtBtD,QAAQsB,eAAe,CAACC,MAAO,gDAWnBjB,MAACA,MAADN,QAAQA,kBAEzBD,KAAK6B,SAAS0B,YAActD,QAAQuD,2BAIlCC,kBAAoBzD,KAAKgC,WAAWhC,KAAKV,UAAUE,mBACpDiE,mBAAqBA,kBAAkBvD,QAAQwD,OAASzD,QAAQ0D,6BAM\/DC,KADW5D,KAAK6B,SAASgC,cACTC,aAAavD,MAAON,UAEpC8D,KAACA,KAADC,GAAOA,UAAYC,mBAAUC,iBA1JhB,mDA0JqDN,yBAC9DO,YAAYV,kBAAmBM,KAAMC,IAQnD9B,gBAAgBkC,aACN3D,GAAKT,KAAK6B,SAASnB,IAAI,KAAMV,KAAKD,OAGxBK,SAASC,eAAeI,GAAGI,cAGlCgB,SAASC,SAAS,0BAA2B,CAACrB,GAAG4D,YAAY,SAE5DC,cAAgB,IAAIC,qDAC1BC,YAAW,UACF3C,SAASC,SAAS,cAAe,KAAMrB,GAAGV,IAC\/CuE,cAAcG,YACf,SAIPL,MAAMM,iBACN3D,OAAOC,SAAWhB,KAAKoC,uBAAuB3B,IASlD2B,uBAAuB3B,QACfkE,QAAU3E,KAAK6B,SAASnB,IAAI,UAAWD,GAAG4D,kBAG1CM,QAAQC,WAAaD,QAAQE,kBAC7BF,QAAU3E,KAAK6B,SAASnB,IAAI,UAAWiE,QAAQE,kBAG9CF,kBAIKA,QAAQG,uBAAcrE,GAAGI,QAHxB"}