This repository has been archived on 2025-01-01. You can view files and clone it, but cannot push or open issues or pull requests.
moodle-local_treestudyplan/amd/build/studyplan-editor-components.min.js.map

1 line
No EOL
222 KiB
Text

{"version":3,"file":"studyplan-editor-components.min.js","sources":["../src/studyplan-editor-components.js"],"sourcesContent":["/*eslint no-var: \"error\"*/\n/*eslint no-console: \"off\"*/\n/*eslint no-unused-vars: warn */\n/*eslint max-len: [\"error\", { \"code\": 160 }] */\n/*eslint-disable no-trailing-spaces */\n/*eslint-env es6*/\n// Put this file in path/to/plugin/amd/src\n\nimport {SimpleLine} from \"./simpleline/simpleline\";\nimport {call} from 'core/ajax';\nimport notification from 'core/notification';\nimport {get_strings} from 'core/str';\nimport {load_stringkeys, load_strings, strformat } from './util/string-helper';\nimport {format_date, add_days, datespaninfo } from './util/date-helper';\nimport {objCopy,transportItem} from './studyplan-processor';\nimport Debugger from './util/debugger';\nimport {download,upload} from './downloader';\nimport {ProcessStudyplan, ProcessStudyplanPage} from './studyplan-processor';\nimport {eventTypes as editSwEventTypes} from 'core/edit_switch';\n\nimport TSComponents from './treestudyplan-components';\nimport mFormComponents from \"./util/mform-helper\";\n\n\nconst STUDYPLAN_EDITOR_FIELDS =\n['name','shortname','description','idnumber','context_id', 'aggregation','aggregation_config'];\nconst STUDYPLAN_EDITOR_PAGE_FIELDS = //TODO: Add 'fullname', 'shortname' and 'description' when implementing proper page management\n['context_id', 'periods','startdate','enddate'];\nconst PERIOD_EDITOR_FIELDS =\n['fullname','shortname','startdate','enddate'];\n\nconst LINE_GRAVITY = 1.3;\n\nconst datechanger_globals = {\n default: false,\n defaultchoice: false,\n hidewarn: false,\n};\n\nexport default {\n STUDYPLAN_EDITOR_FIELDS: STUDYPLAN_EDITOR_FIELDS, // make copy available in plugin\n install(Vue/*,options*/){\n Vue.use(TSComponents);\n Vue.use(mFormComponents);\n let debug = new Debugger(\"treestudyplan-editor\");\n /************************************\n * *\n * Treestudyplan Editor components *\n * *\n ************************************/\n\n /**\n * Check if element is visible\n * @param {Object} elem The element to check\n * @returns {boolean} True if visible\n */\n function isVisible(elem){\n return !!( elem.offsetWidth || elem.offsetHeight || elem.getClientRects().length );\n }\n\n // Create new eventbus for interaction between item components\n const ItemEventBus = new Vue();\n\n // Add event listener for the edit mode event so we can react to it, or at the very least ignore it\n document.addEventListener(editSwEventTypes.editModeSet,(e) => {\n e.preventDefault();\n ItemEventBus.$emit('editModeSet', e.detail.editMode);\n });\n\n let string_keys = load_stringkeys({\n conditions: [\n { value: 'ALL', textkey: 'condition_all'},\n { value: 'ANY', textkey: 'condition_any'},\n ],\n });\n\n let strings = load_strings({\n studyplan_text: {\n studyline_editmode: 'studyline_editmode',\n toolbox_toggle: 'toolbox_toggle',\n editmode_modules_hidden:'editmode_modules_hidden',\n studyline_add: 'studyline_add',\n add$core: 'add$core',\n edit$core: 'edit$core',\n studyline_name: 'studyline_name',\n studyline_name_ph: 'studyline_name_ph',\n studyline_shortname: 'studyline_shortname',\n studyline_shortname_ph: 'studyline_shortname_ph',\n studyline_color: 'studyline_color',\n associations: 'associations',\n associated_cohorts: 'associated_cohorts',\n associated_users: 'associated_users',\n studyline_edit: 'studyline_edit',\n studyplan_name: 'studyplan_name',\n studyplan_name_ph: 'studyplan_name_ph',\n studyplan_shortname: 'studyplan_shortname',\n studyplan_shortname_ph: 'studyplan_shortname_ph',\n studyplan_description: 'studyplan_description',\n studyplan_description_ph: 'studyplan_description_ph',\n studyplan_idnumber: 'studyplan_idnumber',\n studyplan_idnumber_ph: 'studyplan_idnumber_ph',\n studyplan_slots: 'studyplan_slots',\n studyplan_startdate: 'studyplan_startdate',\n studyplan_enddate: 'studyplan_enddate',\n },\n studyplan_advanced: {\n advanced_tools: 'advanced_tools',\n confirm_cancel: 'confirm_cancel',\n confirm_ok: 'confirm_ok',\n success$core: 'success$core',\n error$core: 'failed$core',\n advanced_converted: 'advanced_converted',\n advanced_skipped: 'advanced_skipped',\n advanced_failed: 'advanced_failed',\n advanced_locked: 'advanced_locked',\n advanced_multiple: 'advanced_multiple',\n advanced_error: 'advanced_error',\n advanced_tools_heading: 'advanced_tools_heading',\n advanced_warning_title: 'advanced_warning_title',\n advanced_warning: 'advanced_warning',\n advanced_pick_scale: 'advanced_pick_scale',\n advanced_course_manipulation_title: 'advanced_course_manipulation_title',\n advanced_bulk_course_timing: 'advanced_bulk_course_timing', \n advanced_bulk_course_timing_desc: 'advanced_bulk_course_timing_desc', \n advanced_force_scale_title: 'advanced_force_scale_title',\n advanced_force_scale_desc: 'advanced_force_scale_desc',\n advanced_force_scale_button: 'advanced_force_scale_button',\n advanced_confirm_header: 'advanced_confirm_header',\n advanced_force_scale_confirm: 'advanced_force_scale_confirm',\n advanced_backup_restore: 'advanced_backup_restore',\n advanced_restore: 'advanced_restore',\n advanced_backup: 'advanced_backup',\n advanced_restore_pages: 'advanced_restore_pages',\n advanced_restore_lines: 'advanced_restore_lines',\n advanced_backup_plan: 'advanced_backup_plan',\n advanced_backup_page: 'advanced_backup_page',\n advanced_export: 'advanced_export',\n advanced_export_csv_plan: 'advanced_export_csv_plan',\n advanced_export_csv_page: 'advanced_export_csv_page',\n advanced_import_from_file: 'advanced_import_from_file',\n advanced_purge: \"advanced_purge\",\n advanced_purge_plan: \"advanced_purge_plan\",\n advanced_purge_plan_expl: \"advanced_purge_plan_expl\",\n advanced_purge_page: \"advanced_purge_page\",\n advanced_purge_page_expl: \"advanced_purge_page_expl\",\n advanced_cascade_cohortsync_title: \"advanced_cascade_cohortsync_title\",\n advanced_cascade_cohortsync_desc: \"advanced_cascade_cohortsync_desc\",\n advanced_cascade_cohortsync: \"advanced_cascade_cohortsync\",\n currentpage: \"currentpage\",\n },\n studyplan_edit: {\n studyplan_edit: 'studyplan_edit',\n studyplan_add: 'studyplan_add',\n studyplanpage_add: 'studyplanpage_add',\n studyplanpage_edit: 'studyplanpage_edit',\n info_periodsextended: 'studyplanpage_info_periodsextended',\n warning: 'warning@core',\n },\n period_edit: {\n edit: 'period_edit',\n fullname: 'studyplan_name',\n shortname: 'studyplan_shortname',\n startdate: 'studyplan_startdate',\n enddate: 'studyplan_enddate',\n },\n course_timing: {\n title: 'course_timing_title',\n desc: 'course_timing_desc',\n question: 'course_timing_question',\n warning: 'course_timing_warning',\n timing_ok: 'course_timing_ok',\n timing_off: 'course_timing_off',\n course: 'course@core',\n period: 'period',\n yes: 'yes$core',\n no: 'no$core',\n duration: 'duration',\n years: 'years$core',\n year: 'year$core',\n weeks: 'weeks$core',\n week: 'week$core',\n days: 'days$core',\n day: 'day$core',\n rememberchoice: 'course_timing_rememberchoice',\n hidewarning: 'course_timing_hidewarning',\n periodspan: 'course_period_span',\n periods: 'periods',\n periodspan_desc: 'course_period_span_desc',\n },\n studyplan_associate: {\n associations: 'associations',\n associated_cohorts: 'associated_cohorts',\n associated_users: 'associated_users',\n associate_cohorts: 'associate_cohorts',\n associate_users: 'associate_users',\n add_association: 'add_association',\n delete_association: 'delete_association',\n associations_empty: 'associations_empty',\n associations_search: 'associations_search',\n cohorts: 'cohorts',\n users: 'users',\n selected: 'selected',\n name: 'name',\n context: 'context',\n },\n item_text: {\n select_conditions: \"select_conditions\",\n item_configuration: \"item_configuration\",\n ok: \"ok@core\",\n delete: \"delete@core\",\n item_delete_message: \"item_delete_message\",\n type_course: \"course@core\",\n type_junction: \"tool-junction\",\n type_start: \"tool-start\",\n type_finish: \"tool-finish\",\n type_badge: \"tool-badge\",\n type_invalid: \"course-invalid\",\n },\n item_course_text: {\n select_conditions: \"select_conditions\",\n select_grades: \"select_grades\",\n coursetiming_past: \"coursetiming_past\",\n coursetiming_present: \"coursetiming_present\",\n coursetiming_future: \"coursetiming_future\",\n grade_include: \"grade_include\",\n grade_require: \"grade_require\",\n ok: \"ok@core\",\n cancel: \"cancel@core\",\n delete: \"delete@core\",\n },\n invalid: {\n error: 'error',\n },\n completion: {\n completion_completed: \"completion_completed\",\n completion_incomplete: \"completion_incomplete\",\n aggregation_all: \"aggregation_all\",\n aggregation_any: \"aggregation_any\",\n aggregation_overall_all: \"aggregation_overall_all\",\n aggregation_overall_any: \"aggregation_overall_any\",\n completion_not_configured: \"completion_not_configured\",\n configure_completion: \"configure_completion\",\n },\n competency: {\n competency_not_configured: \"competency_not_configured\",\n configure_competency: \"configure_competency\",\n when: \"when\",\n required: \"required\",\n points: \"points@core_grades\",\n heading: \"competency_heading\",\n details: \"competency_details\",\n },\n badge: {\n share_badge: \"share_badge\",\n dateissued: \"dateissued\",\n dateexpire: \"dateexpire\",\n badgeinfo: \"badgeinfo\",\n },\n });\n\n /*\n * T-STUDYPLAN-ADVANCED\n */\n Vue.component('t-studyplan-advanced', {\n props: {\n value: {\n type: Object,\n default(){ return null;},\n },\n selectedpage: {\n type: Object,\n default(){ return null;},\n }\n\n },\n data() {\n return {\n force_scales: {\n selected_scale: null,\n result: [],\n },\n text: strings.studyplan_advanced,\n };\n },\n created() {\n },\n mounted() {\n },\n updated() {\n },\n computed: {\n scales(){\n return [{\n id: null,\n disabled: true,\n name: this.text.advanced_pick_scale,\n }].concat(this.value.advanced.force_scales.scales);\n },\n },\n methods: {\n force_scales_start(){\n // set confirmation box\n const self=this;\n this.$bvModal.msgBoxConfirm(this.text.advanced_force_scale_confirm,{\n title: this.text.advanced_force_scale_confirm,\n okVariant: 'danger',\n okTitle: this.text.confirm_ok,\n cancelTitle: this.text.confirm_cancel,\n }).then( value => {\n if(value == true){\n call([{\n methodname: 'local_treestudyplan_force_studyplan_scale',\n args: {\n studyplan_id: this.value.id,\n scale_id: this.force_scales.selected_scale,\n }\n }])[0].then(function(response){\n self.force_scales.result = response;\n }).catch(notification.exception);\n }\n });\n },\n export_page(format){\n const self = this;\n if(format == undefined || ![\"json\",\"csv\"].includes(format)){\n format = \"json\";\n }\n call([{\n methodname: 'local_treestudyplan_export_page',\n args: {\n page_id: this.selectedpage.id,\n format: format,\n },\n }])[0].then(function(response){\n\n download(self.value.shortname+\".page.\"+format,response.content,response.format);\n }).catch(notification.exception);\n },\n export_plan(){\n const self = this;\n call([{\n methodname: 'local_treestudyplan_export_plan',\n args: {\n studyplan_id: this.value.id,\n format: \"json\",\n },\n }])[0].then(function(response){\n download(self.value.shortname+\".plan.json\",response.content,response.format);\n }).catch(notification.exception);\n },\n bulk_course_timing() {\n const self = this;\n call([{\n methodname: 'local_treestudyplan_bulk_course_timing',\n args: {\n page_id: this.selectedpage.id,\n },\n }])[0].then(function(response){\n if(response.success){\n // Reloading the webpage saves trouble reloading the specific page updated.\n location.reload();\n } else {\n this.$bvModal.msgBoxOk(response.msg, {title: \"Could not set bulk course timing\"} );\n debug.error(\"Could not set bulk course timing: \",response.msg);\n }\n }).catch(notification.exception);\n },\n import_studylines(){\n //const self = this;\n upload((filename,content)=>{\n call([{\n methodname: 'local_treestudyplan_import_studylines',\n args: {\n page_id: this.selectedpage.id,\n content: content,\n format: \"application/json\",\n },\n }])[0].then(function(response){\n if(response.success){\n location.reload();\n } else {\n this.$bvModal.msgBoxOk(response.msg, {title: \"Import failed\"} );\n debug.error(\"Import failed: \",response.msg);\n }\n }).catch(notification.exception);\n }, \"application/json\");\n },\n import_pages(){\n //const self = this;\n upload((filename,content)=>{\n call([{\n methodname: 'local_treestudyplan_import_pages',\n args: {\n studyplan_id: this.value.id,\n content: content,\n format: \"application/json\",\n },\n }])[0].then(function(response){\n if(response.success){\n location.reload();\n } else {\n this.$bvModal.msgBoxOk(response.msg, {title: \"Import failed\"} );\n debug.error(\"Import failed: \",response.msg);\n }\n }).catch(notification.exception);\n }, \"application/json\");\n },\n purge_studyplan(){\n call([{\n methodname: 'local_treestudyplan_delete_studyplan',\n args: {\n id: this.value.id,\n force: true,\n },\n }])[0].then(function(response){\n if(response.success){\n location.reload();\n } else {\n this.$bvModal.msgBoxOk(response.msg, {title: \"Could not delete plan \"} );\n debug.error(\"Could not delete plan: \",response.msg);\n }\n }).catch(notification.exception);\n },\n purge_studyplanpage(){\n if (this.selectedpage) {\n call([{\n methodname: 'local_treestudyplan_delete_studyplanpage',\n args: {\n id: this.selectedpage.id,\n force: true,\n },\n }])[0].then(function(response){\n if(response.success){\n location.reload();\n } else {\n this.$bvModal.msgBoxOk(response.msg, {title: \"Could not delete page\"} );\n debug.error(\"Could not delete page: \",response.msg);\n }\n }).catch(notification.exception);\n }\n },\n cascade_cohortsync(){\n const self = this;\n call([{\n methodname: 'local_treestudyplan_cascade_cohortsync',\n args: {\n studyplan_id: this.value.id,\n },\n }])[0].then(function(response){\n self.$bvModal.msgBoxOk(response.success?self.text.success$core:self.text.error$core,\n { title: self.text.advanced_cascade_cohortsync});\n }).catch(notification.exception);\n },\n modal_close(){\n this.force_scales.result = [];\n }\n },\n template:\n `\n <span>\n <a v-if=\"value.advanced\"\n href='#'\n @click.prevent=''\n class='text-danger'\n v-b-modal=\"'t-studyplan-'+value.id+'-advanced'\"\n ><i class='fa fa-wrench'></i> {{text.advanced_tools}}</a>\n <b-modal v-if=\"value.advanced\"\n :id=\"'t-studyplan-'+value.id+'-advanced'\"\n size=\"lg\"\n :title=\"text.advanced_tools_heading\"\n ok-only\n @hide=\"modal_close\"\n body-class=\"p-0\"\n >\n <b-tabs card>\n <b-tab :title=\"text.advanced_warning_title\" active>\n {{ text.advanced_warning}}\n </b-tab>\n <b-tab :title=\"text.advanced_course_manipulation_title\" >\n <h3>{{ text.advanced_cascade_cohortsync_title}}</h3>\n <p>{{ text.advanced_cascade_cohortsync_desc}}</p>\n <p class=\"mt-2\"><b-button\n variant=\"info\"\n @click=\"cascade_cohortsync\"\n >{{ text.advanced_cascade_cohortsync}}</b-button></p>\n <h3>{{ text.advanced_bulk_course_timing}}</h3>\n <p>{{ text.advanced_bulk_course_timing_desc}}</p>\n <p>{{text.currentpage}} <i>{{selectedpage.fullname}}</i></p>\n <p class=\"mt-2\"><b-button\n variant=\"info\"\n @click=\"bulk_course_timing\"\n >{{ text.advanced_bulk_course_timing}}</b-button></p>\n <template v-if=\"['bistate','tristate'].includes(value.aggregation)\">\n <h3>{{ text.advanced_force_scale_title}}</h3>\n {{ text.advanced_force_scale_desc}}\n <p class=\"mt-2\"><b-form-select v-model=\"force_scales.selected_scale\"\n :options=\"scales\" text-field=\"name\" value-field=\"id\"\n ></b-form-select>\n <b-button\n variant=\"danger\"\n :disabled=\"force_scales.selected_scale == null\"\n @click=\"force_scales_start\"\n >{{ text.advanced_force_scale_button}}</b-button>\n </p>\n <p class=\"mt-2\">\n <ul class='t-advanced-scrollable' v-if=\"force_scales.result.length > 0\">\n <li v-for=\"c in force_scales.result\">\n <span class='t-advanced-coursename'>{{c.course.fullname}}</span>\n <ul v-if=\"c.grades.length > 0\">\n <li v-for='g in c.grades'\n ><span class='t-advanced-gradename'><span v-html=\"g.name\"></span></span>\n <span v-if=\"g.changed == 'converted'\" class='t-advanced-status changed'\n >{{text.advanced_converted}}</span\n ><span v-else-if=\"g.changed == 'skipped'\" class='t-advanced-status skipped'\n >{{text.advanced_skipped}}</span\n ><span v-else class='t-advanced-status skipped'\n >{{text.advanced_error}}</span\n ></li>\n </ul>\n </li>\n </ul>\n </p>\n </template>\n </b-tab>\n <b-tab :title='text.advanced_backup_restore'>\n <h3>{{ text.advanced_backup }}</h3>\n <p><b-button\n variant=\"primary\"\n @click=\"export_page('json')\"\n >{{ text.advanced_backup_page }}</b-button>\n {{text.currentpage}} <i>{{selectedpage.fullname}}</i></p>\n <p><b-button\n variant=\"primary\"\n @click=\"export_plan('json')\"\n >{{ text.advanced_backup_plan }}</b-button></p>\n <h3>{{ text.advanced_restore }}</h3>\n <p><b-button\n variant=\"danger\"\n @click=\"import_studylines\"\n >{{ text.advanced_restore_lines}}</b-button></p>\n <p><b-button\n variant=\"danger\"\n @click=\"import_pages\"\n >{{ text.advanced_restore_pages }}</b-button></p>\n <h3>{{ text.advanced_export }}</h3>\n <p><b-button\n variant=\"primary\"\n @click=\"export_page('csv')\"\n >{{ text.advanced_export_csv_page }}</b-button>\n {{text.currentpage}} <i>{{selectedpage.fullname}}</i></p>\n </b-tab>\n <b-tab :title='text.advanced_purge'>\n <p>{{text.advanced_purge_page_expl}}</p>\n <p>{{text.currentpage}} <i>{{selectedpage.fullname}}</i></p>\n <p><b-button\n variant=\"danger\"\n @click=\"purge_studyplanpage\"\n >{{ text.advanced_purge_page}}</b-button></p>\n <p>{{text.advanced_purge_plan_expl}}</p>\n <p><b-button\n variant=\"danger\"\n @click=\"purge_studyplan\"\n >{{ text.advanced_purge_plan}}</b-button></p>\n </b-tab>\n </b-tabs>\n </b-modal>\n </span>\n `\n });\n\n\n /*\n * T-STUDYPLAN-EDIT\n */\n Vue.component('t-studyplan-edit', {\n props: {\n 'value' :{\n type: Object,\n default(){ return null;},\n },\n 'mode' :{\n type: String,\n default() { return \"edit\";},\n },\n 'type' :{\n type: String,\n default() { return \"link\";},\n },\n 'variant' : {\n type: String,\n default() { return \"\";},\n },\n 'contextid': {\n type: Number,\n default: 1\n },\n },\n data() {\n return {\n text: strings.studyplan_edit,\n };\n },\n computed: {\n },\n methods: {\n planSaved(updatedplan){\n const self = this;\n debug.info(\"Got new plan data\",updatedplan);\n\n if(self.mode == 'create'){\n // Inform parent of the details of the newly created plan\n self.$emit(\"created\",updatedplan);\n }\n else {\n // determine if the plan moved context...\n const moved_from = self.value.context_id;\n const moved_to = updatedplan.context_id;\n const moved = (moved_from != moved_to);\n\n if(updatedplan.pages[0].periods != self.value.pages[0].periods){\n // If the pages changed, just reload the entire model for the plan\n call([{\n methodname: 'local_treestudyplan_get_studyplan_map',\n args: { id: self.value.id}\n }])[0].then(function(response){\n self.value = ProcessStudyplan(response,true);\n debug.info('studyplan processed');\n self.$emit('input',self.value);\n }).catch(function(error){\n notification.exception(error);\n });\n } else {\n // Copy updated fields and trigger update\n objCopy(self.value,updatedplan,STUDYPLAN_EDITOR_FIELDS);\n self.$emit('input',self.value);\n if(moved){\n self.$emit('moved',self.value, moved_from, moved_to);\n }\n }\n }\n },\n }\n ,\n template:\n `\n <span class='s-studyplan-edit'>\n <mform \n name=\"studyplan_editform\" \n :params=\"{studyplan_id: value.id, mode: mode, contextid: contextid }\" \n @saved=\"planSaved\"\n :variant=\"variant\"\n :type=\"type\"\n :title=\"(mode == 'create')?text.studyplan_add:text.studyplan_edit\"\n ><slot><i class='fa fa-gear'></i></slot></mform>\n </span>\n `\n });\n\n /*\n * T-STUDYPLAN-EDIT\n */\n Vue.component('t-studyplan-page-edit', {\n props: {\n 'value' :{\n type: Object,\n default(){ return null;},\n },\n 'mode' :{\n type: String,\n default() { return \"edit\";},\n },\n 'type' :{\n type: String,\n default() { return \"link\";},\n },\n 'variant' : {\n type: String,\n default() { return \"\";},\n },\n 'studyplan': {\n type: Object,\n },\n },\n data() {\n return {\n text: strings.studyplan_edit,\n };\n },\n computed: {\n },\n methods: {\n planSaved(updatedpage){\n const self = this;\n debug.info(\"Got new page data\",updatedpage);\n\n if(self.mode == 'create'){\n // Inform parent of the details of the newly created plan\n self.$emit(\"created\",updatedpage);\n }\n else {\n const page = ProcessStudyplanPage(updatedpage);\n debug.info('studyplan page processed');\n\n if (self.value.periods < page.periods) {\n this.$bvModal.msgBoxOk(this.text.info_periodsextended, {\n title: this.text.warning,\n okVariant: 'success',\n centered: true\n });\n }\n self.$emit('input',page);\n\n }\n },\n }\n ,\n template:\n `\n <span class='s-studyplan-page-edit'>\n <mform \n name=\"studyplanpage_editform\" \n :params=\"{page_id: value.id, studyplan_id: studyplan.id, mode: mode }\" \n @saved=\"planSaved\"\n :variant=\"variant\"\n :type=\"type\"\n :title=\"(mode == 'create')?text.studyplanpage_add:text.studyplanpage_edit\"\n ><slot><i class='fa fa-gear'></i></slot></mform>\n </span>\n `\n });\n\n\n /*\n * T-STUDYPLAN-ASSOCIATE\n */\n Vue.component('t-studyplan-associate', {\n props: ['value',],\n data() {\n return {\n show: false,\n config: {\n userfields: [\n { key: \"selected\",},\n { key: \"firstname\", \"sortable\": true,},\n { key: \"lastname\", \"sortable\": true,},\n ],\n cohortfields:[\n { key: \"selected\",},\n { key: \"name\", \"sortable\": true,},\n { key: \"context\", \"sortable\": true,},\n ]\n },\n association: {\n cohorts: [],\n users: [],\n },\n loading: {\n cohorts: false,\n users: false,\n },\n search: {users: [], cohorts:[]},\n selected: {\n search: {users: [] , cohorts:[]},\n associated: {users: [] , cohorts:[]}\n },\n text: strings.studyplan_associate,\n };\n },\n created() {\n\n },\n mounted() {\n },\n updated() {\n\n },\n methods: {\n showModal(){\n this.show = true;\n this.loadAssociations();\n },\n cohortOptionModel(c){\n return {\n value: c.id,\n text: c.name + ' (' + c.context.path.join(' / ') + ')',\n };\n },\n userOptionModel(u){\n return {\n value: u.id,\n text: u.firstname + ' ' + u.lastname,\n };\n },\n loadAssociations(){\n const self = this;\n self.loading.cohorts = true;\n self.loading.users = true;\n call([{\n methodname: 'local_treestudyplan_associated_users',\n args: { studyplan_id: self.value.id,}\n }])[0].then(function(response){\n self.association.users = response.map(self.userOptionModel);\n self.loading.users = false;\n }).catch(notification.exception);\n\n call([{\n methodname: 'local_treestudyplan_associated_cohorts',\n args: { studyplan_id: self.value.id,}\n }])[0].then(function(response){\n self.association.cohorts = response.map(self.cohortOptionModel);\n self.loading.cohorts = false;\n }).catch(notification.exception);\n },\n searchCohorts(searchtext){\n const self = this;\n\n if(searchtext.length > 0)\n {\n call([{\n methodname: 'local_treestudyplan_list_cohort',\n args: { like: searchtext, exclude_id: self.value.id}\n }])[0].then(function(response){\n self.search.cohorts = response.map(self.cohortOptionModel);\n }).catch(notification.exception);\n }\n else {\n self.search.cohorts = [];\n }\n },\n cohortAssociate(){\n const self = this;\n let requests = [];\n const associated = self.association.cohorts;\n const search = self.search.cohorts;\n const searchselected = self.selected.search.cohorts;\n for(const i in searchselected){\n const r = searchselected[i];\n requests.push({\n methodname: 'local_treestudyplan_connect_cohort',\n args: {studyplan_id: self.value.id, cohort_id: r},\n fail: notification.exception,\n done: function(response){\n if(response.success){\n transportItem(associated,search,r);\n }\n }\n });\n }\n call(requests);\n },\n cohortDisassociate(){\n const self = this;\n let requests = [];\n const associatedselected = self.selected.associated.cohorts;\n const associated = self.association.cohorts;\n const search = self.search.cohorts;\n for(const i in associatedselected){\n const r = associatedselected[i];\n requests.push({\n methodname: 'local_treestudyplan_disconnect_cohort',\n args: {studyplan_id: self.value.id, cohort_id: r},\n fail: notification.exception,\n done: function(response){\n if(response.success){\n transportItem(search,associated,r);\n }\n }\n });\n }\n call(requests);\n },\n searchUsers(searchtext){\n const self = this;\n if(searchtext.length > 0)\n {\n call([{\n methodname: 'local_treestudyplan_find_user',\n args: { like: searchtext, exclude_id: self.value.id}\n }])[0].then(function(response){\n self.search.users = response.map(self.userOptionModel);\n }).catch(notification.exception);\n }\n else {\n self.search.users = [];\n }\n },\n userAssociate(){\n const self = this;\n let requests = [];\n const associated = self.association.users;\n const search = self.search.users;\n const searchselected = self.selected.search.users;\n for(const i in searchselected){\n const r = searchselected[i];\n\n requests.push({\n methodname: 'local_treestudyplan_connect_user',\n args: {studyplan_id: self.value.id, user_id: r},\n fail: notification.exception,\n done: function(response){\n if(response.success){\n transportItem(associated,search,r);\n }\n }\n });\n }\n call(requests);\n },\n userDisassociate(){\n const self = this;\n let requests = [];\n const associated = self.association.users;\n const associatedselected = self.selected.associated.users;\n const search = self.search.users;\n for(const i in associatedselected){\n const r = associatedselected[i];\n\n requests.push({\n methodname: 'local_treestudyplan_disconnect_user',\n args: {studyplan_id: self.value.id, user_id: r},\n fail: notification.exception,\n done: function(response){\n if(response.success){\n transportItem(search,associated,r);\n }\n }\n });\n }\n call(requests);\n },\n }\n ,\n template:\n`\n<span class='s-studyplan-associate'\n ><a href='#' @click.prevent=\"showModal\" ><slot><i class='fa fa-users'></i></slot></a>\n <b-modal\n v-model=\"show\"\n size=\"lg\"\n ok-variant=\"primary\"\n :title=\"text.associations + ' - ' + value.name\"\n ok-only>\n <b-tabs class='s-studyplan-associate-window'>\n <b-tab :title=\"text.cohorts\">\n <b-container>\n <b-row class='mb-2 mt-2'>\n <b-col>{{text.associated_cohorts}}</b-col>\n <b-col>{{text.associate_cohorts}}</b-col>\n </b-row>\n <b-row class='mb-2'>\n <b-col>\n </b-col>\n <b-col>\n <b-form-input\n type=\"text\" @input=\"searchCohorts($event)\"\n :placeholder=\"text.search\"></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col>\n <b-form-select\n multiple\n v-model=\"selected.associated.cohorts\"\n :options=\"association.cohorts\"\n :select-size=\"10\"\n ></b-form-select>\n </b-col>\n <b-col>\n <b-form-select\n multiple\n v-model=\"selected.search.cohorts\"\n :options=\"search.cohorts\"\n :select-size=\"10\"\n ></b-form-select>\n </b-col>\n </b-row>\n <b-row class='mt-2'>\n <b-col>\n <b-button variant='danger' @click=\"cohortDisassociate()\"\n ><i class='fa fa-chain-broken'></i>&nbsp;{{text.delete_association}}</b-button>\n </b-col>\n <b-col>\n <b-button variant='success' @click=\"cohortAssociate()\"\n ><i class='fa fa-link'></i>&nbsp;{{text.add_association}}</b-button>\n </b-col>\n </b-row>\n </b-container>\n </b-tab>\n <b-tab :title=\"text.users\">\n <b-container>\n <b-row class='mb-2 mt-2'>\n <b-col>{{text.associated_users}}</b-col>\n <b-col>{{text.associate_users}}</b-col>\n </b-row>\n <b-row class='mb-2'>\n <b-col>\n </b-col>\n <b-col>\n <b-form-input\n type=\"text\"\n @input=\"searchUsers($event)\"\n placeholder=\"Search users\"></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col>\n <b-form-select\n multiple\n v-model=\"selected.associated.users\"\n :options=\"association.users\"\n :select-size=\"10\"\n ></b-form-select>\n </b-col>\n <b-col>\n <b-form-select\n multiple\n v-model=\"selected.search.users\"\n :options=\"search.users\"\n :select-size=\"10\"\n ></b-form-select>\n </b-col>\n </b-row>\n <b-row class='mt-2'>\n <b-col>\n <b-button variant='danger' @click=\"userDisassociate()\"\n ><i class='fa fa-chain-broken'></i>&nbsp;{{text.delete_association}}</b-button>\n </b-col>\n <b-col>\n <b-button variant='success' @click=\"userAssociate()\"\n ><i class='fa fa-link'></i>&nbsp;{{text.add_association}}</b-button>\n </b-col>\n </b-row>\n </b-container>\n </b-tab>\n </b-tabs>\n </b-modal>\n</span>\n`\n });\n\n /*******************\n *\n * Period editor\n *\n *************/\n\n Vue.component('t-period-edit', {\n props: {\n 'value' :{\n type: Object,\n default(){ return null;},\n },\n 'type' :{\n type: String,\n default() { return \"link\";},\n },\n 'variant' : {\n type: String,\n default() { return \"\";},\n },\n 'minstart' : {\n type: String,\n default() { return null;},\n },\n 'maxend' : {\n type: String,\n default() { return null;},\n }\n },\n data() {\n return {\n show: false,\n editdata: {\n fullname: '',\n shortname: '',\n startdate: (new Date()).getFullYear() + '-08-01',\n enddate: ((new Date()).getFullYear()+1) + '-08-01',\n },\n text: strings.period_edit,\n };\n },\n created() {\n },\n mounted() {\n },\n updated() {\n\n },\n computed: {\n },\n methods: {\n editStart(){\n objCopy(this.editdata,this.value,PERIOD_EDITOR_FIELDS);\n this.show = true;\n },\n editFinish(){\n const self = this;\n let args = { 'id': this.value.id };\n\n objCopy(args,this.editdata,PERIOD_EDITOR_FIELDS);\n\n call([{\n methodname: 'local_treestudyplan_edit_period',\n args: args\n }])[0].then(function(response){\n objCopy(self.value,response,PERIOD_EDITOR_FIELDS);\n self.$emit('input',self.value);\n self.$emit('edited',self.value);\n }).catch(notification.exception);\n },\n refresh(){\n const self = this;\n call([{\n methodname: 'local_treestudyplan_get_period',\n args: { 'id': this.value.id },\n }])[0].then(function(response){\n objCopy(self.value,response,PERIOD_EDITOR_FIELDS);\n self.$emit('input',self.value);\n }).catch(notification.exception);\n },\n add_day(date,days) {\n if( days === undefined ){\n days = 1;\n }\n return add_days(date,days);\n },\n sub_day(date,days) {\n if( days === undefined ){\n days = 1;\n }\n return add_days(date,0 - days);\n },\n }\n ,\n template:\n `\n <span class='t-period-edit'>\n <b-button :variant=\"variant\" v-if='type == \"button\"' @click.prevent='editStart()'\n ><slot><i class='fa fa-gear'></i></slot></b-button>\n <a variant=\"variant\" v-else href='#' @click.prevent='editStart()'\n ><slot><i class='fa fa-gear'></i></slot></a>\n <b-modal\n v-model=\"show\"\n size=\"lg\"\n ok-variant=\"primary\"\n :title=\"text.edit\"\n @ok=\"editFinish()\"\n :ok-disabled=\"Math.min(editdata.fullname.length,editdata.shortname.length) == 0\"\n >\n <b-container>\n <b-row>\n <b-col cols=\"4\">{{ text.fullname}}</b-col>\n <b-col cols=\"8\">\n <b-form-input v-model=\"editdata.fullname\"\n :state='editdata.fullname.length>0'\n ></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"4\">{{ text.shortname}}</b-col>\n <b-col cols=\"8\">\n <b-form-input v-model=\"editdata.shortname\"\n :state='editdata.shortname.length>0'\n ></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"4\">{{ text.studyplan_startdate}}</b-col>\n <b-col cols=\"8\">\n <b-form-datepicker\n start-weekday=\"1\"\n v-model=\"editdata.startdate\"\n :min=\"(minstart ? minstart : '')\"\n :max=\"sub_day(value.enddate)\"\n ></b-form-datepicker>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"4\">{{ text.studyplan_enddate}}</b-col>\n <b-col cols=\"8\">\n <b-form-datepicker\n start-weekday=\"1\"\n v-model=\"editdata.enddate\"\n :min=\"add_day(value.startdate)\"\n :max=\"(maxend ? maxend : '')\"\n ></b-form-datepicker>\n </b-col>\n </b-row>\n </b-container>\n </b-modal>\n </span>\n `\n });\n\n // TAG: Start studyplan component\n /*\n * T-STUDYPLAN\n */\n Vue.component('t-studyplan', {\n props: [ 'value', 'index', 'initeditmode'],\n data() {\n return {\n config: {\n userfields: [\n { key: \"selected\",},\n { key: \"firstname\", \"sortable\": true,},\n { key: \"lastname\", \"sortable\": true,},\n ],\n cohortfields:[\n { key: \"selected\",},\n { key: \"name\", \"sortable\": true,},\n { key: \"context\", \"sortable\": true,},\n ]\n },\n create: {\n studyline: {\n 'name': '',\n 'shortname': '',\n 'color': '#DDDDDD',\n },\n page: {\n 'id': -1,\n 'name' : '',\n 'shortname' : ''\n }\n },\n edit: {\n toolbox_shown: false,\n studyline: {\n editmode: false,\n data: {\n name: '',\n shortname: '',\n color: '#DDDDDD',\n },\n original: {},\n },\n studyplan: {\n data: {\n name: '',\n shortname: '',\n description: '',\n slots : 4,\n startdate: '2020-08-01',\n enddate: '',\n aggregation: '',\n aggregation_config: '',\n aggregation_info: {\n useRequiredGrades: true,\n useItemCondition: false,\n },\n\n },\n original: {},\n }\n },\n text: strings.studyplan_text,\n cache: {\n linelayers: {},\n },\n selectedpageindex: 0,\n emptyline: {\n id: -1,\n name: '<No study lines defined>',\n shortname: '<No study lines>',\n color: '#FF0000',\n filterslots: [{}],\n courseslots: [{}]\n }\n };\n },\n created() {\n const self = this;\n // Listener for the signal that a new connection was made and needs to be drawn\n // Sent by the incoming item - By convention, outgoing items are responsible for drawing the lines\n ItemEventBus.$on('editModeSet', this.onMoodleEditModeChanged);\n ItemEventBus.$on('coursechange', () => {\n self.$emit('pagechanged',this.selectedpage);\n });\n },\n mounted() {\n /*if(this.value.pages[0].studylines.length == 0){\n // start in editmode if studylines on first page are empty\n this.edit.studyline.editmode = true;\n }*/\n this.edit.studyline.editmode = this.initeditmode;\n this.$root.$emit('redrawLines');\n this.$emit('pagechanged',this.selectedpage);\n },\n updated() {\n this.$root.$emit('redrawLines');\n ItemEventBus.$emit('redrawLines');\n },\n computed: {\n selectedpage() {\n return this.value.pages[this.selectedpageindex];\n }\n },\n methods: {\n onMoodleEditModeChanged(mode){\n this.edit.studyline.editmode = mode;\n },\n columns(page) {\n return 1+ (page.periods * 2);\n },\n columns_stylerule(page) {\n // Uses css variables, so width for slots and filters can be configured in css\n let s = \"grid-template-columns: var(--studyplan-filter-width)\"; // use css variable here\n for(let i=0; i<page.periods;i++){\n s+= \" var(--studyplan-course-width) var(--studyplan-filter-width)\";\n }\n return s+\";\";\n },\n trashbin_accepts(type){\n if(type.item){\n return true;\n } else {\n return false;\n }\n },\n countLineLayers(line,page){\n // For some optimization, we cache the value of this calculation for about a second\n // Would be a lot nicer if we could use a computed property for this.....\n if( this.cache.linelayers[line.id]\n && ((new Date()) - this.cache.linelayers[line.id].timestamp < 1000 )\n ){\n return this.cache.linelayers[line.id].value;\n }\n else\n {\n let maxLayer = -1;\n for(let i = 0; i <= page.periods; i++){\n if(line.slots[i]){\n const slot = line.slots[i];\n // Determine the amount of used layers in a studyline slit\n for(const ix in line.slots[i].courses){\n const item = line.slots[i].courses[ix];\n if(item.layer > maxLayer){\n maxLayer = item.layer;\n }\n }\n for(const ix in line.slots[i].filters){\n const item = line.slots[i].filters[ix];\n if(item.layer > maxLayer){\n maxLayer = item.layer;\n }\n }\n\n }\n }\n this.cache.linelayers[line.id] = {\n value: (maxLayer + 1),\n timestamp: (new Date()),\n };\n return maxLayer+1;\n }\n },\n slotsempty(slots) {\n if(Array.isArray(slots)){\n let count = 0;\n for(let i = 0; i < slots.length; i++) {\n if(Array.isArray(slots[i].courses)){\n count += slots[i].courses.length;\n }\n if(Array.isArray(slots[i].filters)){\n count += slots[i].filters.length;\n }\n }\n return (count == 0);\n } else {\n return false;\n }\n },\n movedStudyplan(plan,from,to) {\n this.$emit('moved',plan,from,to); // Throw the event up....\n },\n addStudyLine(page,newlineinfo) {\n call([{\n methodname: 'local_treestudyplan_add_studyline',\n args: {\n 'page_id': page.id,\n 'name': newlineinfo.name,\n 'shortname': newlineinfo.shortname,\n 'color': newlineinfo.color,\n 'sequence': page.studylines.length,\n }\n }])[0].then(function(response){\n page.studylines.push(response);\n newlineinfo.name = '';\n newlineinfo.shortname = '';\n newlineinfo.color = \"#dddddd\";\n }).catch(notification.exception);\n },\n editLineStart(line) {\n Object.assign(this.edit.studyline.data,line);\n this.edit.studyline.original = line;\n this.$bvModal.show('modal-edit-studyline-'+this.value.id);\n },\n editLineFinish() {\n let editedline = this.edit.studyline.data;\n let originalline = this.edit.studyline.original;\n call([{\n methodname: 'local_treestudyplan_edit_studyline',\n args: { 'id': editedline.id,\n 'name': editedline.name,\n 'shortname': editedline.shortname,\n 'color': editedline.color,}\n }])[0].then(function(response){\n originalline['name'] = response['name'];\n originalline['shortname'] = response['shortname'];\n originalline['color'] = response['color'];\n }).catch(notification.exception);\n },\n deleteLine(page,line) {\n const self=this;\n get_strings([\n {key: 'studyline_confirm_remove', param: line.name, component: 'local_treestudyplan' },\n {key: 'delete', component: 'core' },\n ]).then(function(s){\n self.$bvModal.msgBoxConfirm(s[0], {\n okTitle: s[1],\n okVariant: 'danger',\n }).then(function(modalresponse){\n if(modalresponse){\n call([{\n methodname: 'local_treestudyplan_delete_studyline',\n args: { 'id': line.id, }\n }])[0].then(function(response){\n if(response.success == true){\n let index = page.studylines.indexOf(line);\n page.studylines.splice(index, 1);\n }\n }).catch(notification.exception);\n }\n });\n });\n },\n reorderLines(event,lines){\n\n // apply reordering\n event.apply(lines);\n // send the new sequence to the server\n let sequence = [];\n for(let idx in lines)\n {\n sequence.push({'id': lines[idx].id,'sequence': idx});\n }\n call([{\n methodname: 'local_treestudyplan_reorder_studylines',\n args: { 'sequence': sequence }\n }])[0].then(function(response){\n }).catch(notification.exception);\n },\n deletePlan(studyplan){\n const self=this;\n get_strings([\n {key: 'studyplan_confirm_remove', param: studyplan.name, component: 'local_treestudyplan' },\n {key: 'delete', component: 'core' },\n ]).then(function(s){\n self.$bvModal.msgBoxConfirm(s[0], {\n okTitle: s[1],\n okVariant: 'danger',\n }).then(function(modalresponse){\n if(modalresponse){\n call([{\n methodname: 'local_treestudyplan_delete_studyplan',\n args: { 'id': studyplan.id, force: true}\n }])[0].then(function(response){\n if(response.success == true){\n self.$root.$emit(\"studyplanRemoved\",studyplan);\n }\n }).catch(notification.exception);\n }\n });\n });\n },\n deleteStudyItem(event){\n //const self = this;\n let item = event.data;\n call([{\n methodname: 'local_treestudyplan_delete_studyitem',\n args: { 'id': item.id, }\n }])[0].then(function(response){\n if(response.success == true){\n event.source.$emit('cut',event);\n }\n }).catch(notification.exception);\n\n },\n showslot(page,line,index, layeridx, type){\n // check if the slot should be hidden because a previous slot has an item with a span\n // so big that it hides this slot\n const forGradable = (type == 'gradable')?true:false;\n const periods = page.periods;\n let show = true;\n for(let i = 0; i < periods; i++){\n if(line.slots[index-i] && line.slots[index-i].courses){\n const list = line.slots[index-i].courses;\n for(const ix in list){ // Really wish that 'for of' would work with the minifier moodle uses\n const item = list[ix];\n if(item.layer == layeridx){\n if(forGradable){\n if(i > 0 && (item.span - i) > 0){\n show = false;\n }\n } else {\n if((item.span - i) > 1){\n show = false;\n }\n }\n }\n }\n }\n }\n\n return show;\n },\n periodEdited(pi) {\n const prev = this.$refs[\"periodeditor-\" + (pi.period - 1)][0];\n if(prev) {\n prev.refresh();\n }\n const next = this.$refs[\"periodeditor-\" + (pi.period + 1)][0];\n if(next) {\n next.refresh();\n }\n },\n add_day(date,days) {\n if( days === undefined ){\n days = 1;\n }\n return add_days(date,days);\n },\n sub_day(date,days) {\n if( days === undefined ){\n days = 1;\n }\n return add_days(date,0 - days);\n },\n toolbox_switched(event){\n this.$emit('toggletoolbox',event);\n },\n pagecreated(page) {\n this.value.pages.push(page);\n },\n selectedpageChanged(newTabIndex,prevTabIndex) {\n const page = this.value.pages[newTabIndex];\n this.$emit('pagechanged',page);\n }\n }\n ,\n template:\n `\n <div>\n <div class='controlbox t-studyplan-controlbox'>\n <div class=\"controlbox-group\">\n <b-form-checkbox v-model=\"edit.studyline.editmode\" class=\"sw-studyplan-toolbar\" switch\n @change=\"toolbox_switched(edit.toolbox_shown && !edit.studyline.editmode); \"\n >{{ text.studyline_editmode }}</b-form-checkbox>\n <b-form-checkbox v-if=\"!edit.studyline.editmode\" v-model=\"edit.toolbox_shown\" class=\"sw-studyplan-toolbar\" switch\n @change=\"toolbox_switched\"\n >{{ text.toolbox_toggle}}</b-form-checkbox>\n <drop\n mode='copy'\n class='t-item-deletebox text-danger border-danger'\n @drop='deleteStudyItem'\n :accepts-type=\"trashbin_accepts\"\n ><i class='fa fa-trash'></i>\n </drop>\n </div>\n <div class=\"controlbox-group\">\n <span class='control editable'>\n <t-studyplan-advanced v-model=\"value\" :selectedpage=\"selectedpage\"></t-studyplan-advanced>\n </span>\n <span class='control editable'>\n <t-studyplan-associate\n v-model=\"value\"><i class='fa fa-users'></i> {{text.associations}}</t-studyplan-associate>\n </span>\n <span class='control editable'>\n <t-studyplan-edit v-model=\"value\" @moved=\"movedStudyplan\" \n ><i class='fa fa-gear'></i> {{text.edit$core}}</t-studyplan-edit>\n </span>\n <span class='control deletable'>\n <a v-if='value.pages.length == 0' href='#' @click='deletePlan(value)'\n ><i class='text-danger fa fa-trash'></i></a>\n </span>\n </div>\n </div>\n <b-card no-body>\n <b-tabs \n v-model='selectedpageindex' \n @activate-tab='selectedpageChanged'\n content-class=\"mt-1\">\n <!-- New Tab Button (Using tabs-end slot) -->\n <template #tabs-end>\n <t-studyplan-page-edit \n :studyplan=\"value\"\n v-model=\"create.page\"\n type=\"link\"\n mode=\"create\"\n @created=\"pagecreated\"\n ><i class='fa fa-plus'></i></t-studyplan-page-edit>\n </template>\n <b-tab\n v-for=\"(page,pageindex) in value.pages\"\n :key=\"page.id\"\n >\n <template #title>\n {{page.shortname}}\n <t-studyplan-page-edit \n v-if=\"pageindex == selectedpageindex\"\n v-model=\"value.pages[pageindex]\"\n :studyplan=\"value\"\n type=\"link\"\n ></t-studyplan-page-edit>\n </template>\n <div class='t-studyplan-content-edit' \n v-if=\"edit.studyline.editmode\">\n <drop-list\n :items=\"page.studylines\"\n class=\"t-slot-droplist\"\n :accepts-type=\"'studyline-'+page.id\"\n xreorder=\"$event.apply(page.studylines)\"\n @reorder=\"reorderLines($event,page.studylines)\"\n mode=\"copy\"\n row\n >\n <template v-slot:item=\"{item}\">\n <drag\n :key=\"item.id\"\n class='t-studyline-drag'\n :data=\"item\"\n :type=\"'studyline-'+page.id\"\n >\n <template v-slot:drag-image>\n <i class=\"fa fa-arrows text-primary\"></i>\n </template>\n <t-studyline-edit\n v-model=\"item\"\n @edit='editLineStart(item)'\n @delete='deleteLine(page,item)'\n >\n <div v-if=\"!slotsempty(item.slots)\"> {{ text.editmode_modules_hidden}} </div>\n </t-studyline-edit>\n </drag>\n </template>\n </drop-list>\n </div>\n <div class='t-studyplan-content' v-else>\n <!-- Now paint the headings column -->\n <div class='t-studyplan-headings'>\n <s-studyline-header-heading :identifier='Number(page.id)'></s-studyline-header-heading>\n <template v-if=\"page.studylines.length > 0\">\n <t-studyline-heading v-for=\"(line,lineindex) in page.studylines\"\n :key=\"line.id\"\n @resize=\"headingresized(lineindex,$event)\"\n v-model=\"page.studylines[lineindex]\"\n :layers='countLineLayers(line,page)+1'\n :class=\" 't-studyline' + ((lineindex%2==0)?' odd ' :' even ' )\n + ((lineindex==0)?' first ':' ')\n + ((lineindex==page.studylines.length-1)?' last ':' ')\"\n ></t-studyline-heading>\n </template>\n <t-studyline-heading v-else\n @resize=\"headingresized(0,$event)\"\n \n :layers=\"1\"\n :class=\"'odd first last'\"\n ></t-studyline-heading>\n </div>\n <!-- Next, paint all the cells in the scrollable -->\n <div class=\"t-studyplan-scrollable\" >\n <div class=\"t-studyplan-timeline\" :style=\"columns_stylerule(page)\">\n <!-- add period information --> \n <template v-for=\"(n,index) in (page.periods+1)\">\n <s-studyline-header-period\n :identifier='Number(page.id)'\n v-if=\"index > 0\"\n v-model=\"page.perioddesc[index-1]\"\n ><t-period-edit\n :ref=\"'periodeditor-'+index\"\n @edited=\"periodEdited\"\n v-model=\"page.perioddesc[index-1]\"\n :minstart=\"(index > 1) ? add_day(page.perioddesc[index-2].startdate,2) : null\"\n :maxend=\"(index < page.periods) ? sub_day(page.perioddesc[index].enddate,2) : null\"\n ></t-period-edit\n ></s-studyline-header-period>\n <div class=\"s-studyline-header-filter\"></div>\n </template>\n\n <!-- Line by line add the items -->\n <!-- The grid layout handles putting it in rows and columns -->\n <template v-for=\"(line,lineindex) in page.studylines\"\n ><template v-for=\"(layernr,layeridx) in countLineLayers(line,page)+1\"\n ><template v-for=\"(n,index) in (page.periods+1)\"\n >\n <t-studyline-slot\n v-if=\"index > 0 && showslot(page,line, index, layeridx, 'gradable')\"\n type='gradable'\n v-model=\"line.slots[index].courses\"\n :key=\"'c-'+lineindex+'-'+index+'-'+layernr\"\n :slotindex=\"index\"\n :line=\"line\"\n :plan=\"value\"\n :page=\"page\"\n :period=\"page.perioddesc[index-1]\"\n :layer=\"layeridx\"\n :class=\"'t-studyline ' + ((lineindex%2==0)?' odd ':' even ')\n + ((lineindex==0 && layernr==1)?' first ':' ')\n + ((lineindex==page.studylines.length-1)?' last ':' ')\n + ((layernr == countLineLayers(line,page))?' lastlyr ':' ')\n + ((layernr == countLineLayers(line,page)+1)?' newlyr ':' ')\"\n ></t-studyline-slot\n ><t-studyline-slot\n type='filter'\n v-if=\"showslot(page,line, index, layeridx, 'filter')\"\n v-model=\"line.slots[index].filters\"\n :key=\"'f-'+lineindex+'-'+index+'-'+layernr\"\n :slotindex=\"index\"\n :line=\"line\"\n :plan=\"value\"\n :page=\"page\"\n :layer=\"layeridx\"\n :class=\"'t-studyline ' + ((lineindex%2==0)?' odd ':' even ')\n + ((lineindex==0 && layernr==1)?' first ':'')\n + ((lineindex==page.studylines.length-1)?' last ':' ')\n + ((index==page.periods)?' rightmost':'')\n + ((layernr == countLineLayers(line,page))?' lastlyr ':' ')\n + ((layernr == countLineLayers(line,page)+1)?' newlyr ':' ')\"\n ></t-studyline-slot\n ></template\n ></template\n ></template\n ></div>\n </div>\n </div>\n <div v-if=\"edit.studyline.editmode\" class='t-studyline-add ml-2 mt-1'>\n <a href=\"#\" v-b-modal=\"'modal-add-studyline-'+page.id\" @click=\"false;\"\n ><i class='fa fa-plus'></i>{{ text.studyline_add }}</a>\n </div>\n <b-modal\n :id=\"'modal-add-studyline-'+page.id\"\n size=\"lg\"\n :ok-title=\"text.add$core\"\n ok-variant=\"primary\"\n :title=\"text.studyline_add\"\n @ok=\"addStudyLine(page,create.studyline)\"\n :ok-disabled=\"Math.min(create.studyline.name.length,create.studyline.shortname.length) == 0\"\n >\n <b-container>\n <b-row>\n <b-col cols=\"3\">{{text.studyline_name}}</b-col>\n <b-col>\n <b-form-input v-model=\"create.studyline.name\" :placeholder=\"text.studyline_name_ph\"></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"3\">{{text.studyline_shortname}}</b-col>\n <b-col>\n <b-form-input\n v-model=\"create.studyline.shortname\"\n :placeholder=\"text.studyline_shortname_ph\"></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"3\">{{text.studyline_color}}</b-col>\n <b-col>\n <input type=\"color\" v-model=\"create.studyline.color\" />\n <!-- hsluv-picker v-model=\"create.studyline.color\" horizontal displaysize=\"175\" ></hsluv-picker -->\n </b-col>\n </b-row>\n </b-container>\n </b-modal>\n <b-modal\n :id=\"'modal-edit-studyline-'+page.id\"\n size=\"lg\"\n ok-variant=\"primary\"\n :title=\"text.studyline_edit\"\n @ok=\"editLineFinish()\"\n :ok-disabled=\"Math.min(edit.studyline.data.name.length,edit.studyline.data.shortname.length) == 0\"\n >\n <b-container>\n <b-row>\n <b-col cols=\"3\">{{ text.studyline_name}}</b-col>\n <b-col>\n <b-form-input\n v-model=\"edit.studyline.data.name\"\n :placeholder=\"text.studyline_name_ph\"></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"3\">{{ text.studyline_shortname}}</b-col>\n <b-col>\n <b-form-input\n v-model=\"edit.studyline.data.shortname\"\n :placeholder=\"text.studyline_shortname_ph\"></b-form-input>\n </b-col>\n </b-row>\n <b-row>\n <b-col cols=\"3\">{{ text.studyline_color}}</b-col>\n <b-col>\n <input type=\"color\" v-model=\"edit.studyline.data.color\" />\n </b-col>\n </b-row>\n </b-container>\n </b-modal>\n </b-tab>\n </b-tabs>\n </b-card>\n </div>\n `\n });\n\n\n /*\n * T-STUDYLINE-HEADER\n */\n Vue.component('t-studyline-heading', {\n props: {\n value : {\n type: Object, // Studyline\n default: function(){ return {};},\n },\n layers: {\n type: Number,\n default: 1,\n },\n },\n data() {\n return {\n layerHeights: {}\n };\n },\n created() {\n // Listener for the signal that a new connection was made and needs to be drawn\n // Sent by the incoming item - By convention, outgoing items are responsible for drawing the lines\n ItemEventBus.$on('lineHeightChange', this.onLineHeightChange);\n },\n computed: {\n\n },\n methods: {\n onLineHeightChange(lineid,layerid,newheight){\n // All layers for this line have the first slot send an update message on layer height change.\n // When one of those updates is received, record the height and recalculate the total height of the\n // header\n if(this.$refs.main && lineid == this.value.id){\n const items = document.querySelectorAll(\n `.t-studyline-slot-0[data-studyline='${this.value.id}']`);\n\n // determine the height of all the lines and add them up.\n let heightSum = 0;\n items.forEach((el) => {\n // getBoundingClientRect() Gets the actual fractional height instead of rounded to integer pixels\n const r = el.getBoundingClientRect();\n const height = r.height;\n heightSum += height;\n });\n\n const heightStyle=`${heightSum}px`;\n this.$refs.main.style.height = heightStyle;\n }\n }\n },\n template: `\n <div class=\"t-studyline t-studyline-heading \"\n :data-studyline=\"value.id\" ref=\"main\"\n ><div class=\"t-studyline-handle\" :style=\"'background-color: ' + value.color\"></div>\n <div class=\"t-studyline-title\">\n <abbr v-b-tooltip.hover.right :title=\"value.name\">{{ value.shortname }}</abbr>\n </div>\n </div>\n `,\n });\n\n /*\n * T-STUDYLINE (Used only for study line edit mode)\n */\n Vue.component('t-studyline-edit', {\n props: {\n value : {\n type: Object, // Studyline\n default: function(){ return {};},\n }\n },\n data() {\n return {\n };\n },\n computed: {\n deletable() {\n // Check if all the slots are empty\n const slots = this.value.slots;\n if(Array.isArray(slots)){\n let count = 0;\n for(let i = 0; i < slots.length; i++) {\n if(Array.isArray(slots[i].courses)){\n count += slots[i].courses.length;\n }\n if(Array.isArray(slots[i].filters)){\n count += slots[i].filters.length;\n }\n }\n return (count == 0);\n } else {\n return false;\n }\n },\n editable() {\n return true;\n }\n },\n methods: {\n onEdit() {\n this.$emit('edit',this.value);\n },\n onDelete() {\n this.$emit('delete',this.value);\n },\n\n },\n template: `\n <div :class=\"'t-studyline '\" >\n <div class=\"t-studyline-handle\" :style=\"'background-color: ' + value.color\"></div>\n <div class=\"t-studyline-title\">\n <div>\n <i class='fa fa-arrows text-primary'></i>\n <abbr v-b-tooltip.hover :title=\"value.name\">{{ value.shortname }}</abbr>\n </div>\n </div>\n <div class='t-studyline-editmode-content'>\n <slot></slot>\n </div>\n <div class='controlbox'>\n <template v-if='editable || deletable'>\n <span class='control editable' v-if='editable'>\n <a href='#' @click='onEdit'><i class='fa fa-pencil'></i></a>\n </span>\n <span class='control deletable' v-if='deletable'>\n <a v-if='deletable' href='#' @click='onDelete'><i class='text-danger fa fa-trash'></i></a>\n </span>\n </template>\n </div>\n </div>\n `,\n });\n\n /*\n * During a redisign it was decided to have the studyline still get the entire array as a value,\n * even though it only shows one drop slot for the layer it is in. This is to make repainting easier,\n * since we modify the array for the layer we handle. FIXME: Make this less weird\n */\n Vue.component('t-studyline-slot', {\n props: {\n type : {\n type: String,\n default: 'gradable',\n },\n slotindex : {\n type: Number,\n default: '',\n },\n line : {\n type: Object,\n default(){ return null;},\n },\n layer : {\n type: Number,\n },\n value: {\n type: Array, // dict with layer as index\n default(){ return [];},\n },\n plan: {\n type: Object, // Studyplan data\n default(){ return null;},\n },\n page: {\n type: Object, // Studyplan data\n default(){ return null;},\n },\n period: {\n type: Object, // Studyplan data\n default(){ return null;},\n },\n },\n mounted() {\n const self=this;\n if(self.type == \"gradable\" && self.slotindex == 1){\n self.resizeListener = new ResizeObserver(() => {\n if(self.$refs.main){\n const size = self.$refs.main.getBoundingClientRect();\n\n ItemEventBus.$emit('lineHeightChange', self.line.id, self.layer, size.height);\n }\n }).observe(self.$refs.main);\n }\n },\n unmounted() {\n if(this.resizeListener) {\n this.resizeListener.disconnect();\n }\n },\n computed: {\n slotkey(){\n return `${this.type}'-'${this.line.id}-${this.slotindex}-${this.layer}`;\n },\n itemidx(){\n for(const ix in this.value){\n const itm = this.value[ix];\n if(itm.layer == this.layer){\n return ix;\n }\n }\n return null;\n },\n item(){\n for(const ix in this.value){\n const itm = this.value[ix];\n if(itm.layer == this.layer){\n return itm;\n }\n }\n return null;\n },\n listtype() {\n return this.type;\n },\n dragacceptlist(){\n if(this.type == \"gradable\"){\n return [\"course\", \"gradable-item\"];\n } else {\n return [\"filter\", \"filter-item\"];\n }\n },\n courseHoverDummy(){\n return {course: this.hover.component};\n },\n current(){\n if( this.period && this.period.startdate && this.period.enddate){\n const now = new Date();\n const pstart = new Date(this.period.startdate);\n const pend = new Date(this.period.enddate);\n return (now >= pstart && now < pend);\n }\n else {\n return false;\n }\n },\n spanCss(){\n if(this.item && this.item.span > 1){\n const span = (2 * this.item.span) - 1;\n return `width: 100%; grid-column: span ${span};`;\n } else {\n return \"\";\n }\n }\n },\n data() {\n return {\n text: strings.course_timing,\n resizeListener: null,\n hover: {\n component:null,\n type: null,\n },\n datechanger: {\n coursespan: null,\n periodspan: null,\n default: false,\n defaultchoice: false,\n hidewarn: false,\n }\n };\n },\n methods: {\n onDrop(event) {\n this.hover.component = null;\n this.hover.type = null;\n debug.info(event);\n const self = this;\n if(event.type.item) {\n let item = event.data;\n\n // To avoid weird visuals with the lines,\n // we add the item to the proper place in the front-end first\n item.layer = this.layer;\n item.slot = this.slotindex;\n self.value.push(item);\n self.$emit(\"input\",self.value);\n\n // then on the next tick, we inform the back end\n // Since moving things around has never been unsuccessful, unless you have other problems,\n // it's better to have nice visuals.\n self.relocateStudyItem(item).done(()=>{\n if(this.$refs.timingChecker){\n this.$refs.timingChecker.validate_course_period();\n }\n });\n }\n else if(event.type.component){\n debug.info(\"Adding new component\");\n if(event.type.type == \"gradable\"){\n call([{\n methodname: 'local_treestudyplan_add_studyitem',\n args: {\n \"line_id\": self.line.id,\n \"slot\" : self.slotindex,\n \"layer\" : self.layer,\n \"type\": 'course',\n \"details\": {\n \"competency_id\": null,\n 'conditions':'',\n 'course_id':event.data.id,\n 'badge_id':null,\n 'continuation_id':null,\n }\n }\n }])[0].then((response) => {\n let item = response;\n self.relocateStudyItem(item).done(()=>{\n self.value.push(item);\n self.$emit(\"input\",self.value);\n\n // call the validate period function on next tick,\n // since it paints the item in the slot first\n this.$nextTick(() => {\n if(this.$refs.timingChecker){\n this.$refs.timingChecker.validate_course_period();\n }\n });\n ItemEventBus.$emit('coursechange');\n });\n }).catch(notification.exception);\n }\n else if(event.type.type == \"filter\") {\n debug.info(\"Adding new filter compenent\");\n call([{\n methodname: 'local_treestudyplan_add_studyitem',\n args: {\n \"line_id\": self.line.id,\n \"slot\" : self.slotindex,\n \"type\": event.data.type,\n \"details\":{\n \"badge_id\": event.data.badge?event.data.badge.id:undefined,\n }\n }\n }])[0].then((response) => {\n let item = response;\n self.relocateStudyItem(item).done(()=>{\n item.layer = this.layer;\n self.value.push(item);\n self.$emit(\"input\",self.value);\n });\n }).catch(notification.exception);\n }\n }\n },\n onCut(event) {\n const self=this;\n let id = event.data.id;\n for(let i = 0; i < self.value.length; i++){\n if(self.value[i].id == id){\n self.value.splice(i, 1); i--;\n break; // just remove one\n }\n }\n // Do something to signal that this item has been removed\n this.$emit(\"input\",this.value);\n ItemEventBus.$emit('coursechange');\n },\n relocateStudyItem(item){\n const iteminfo = {'id': item.id, 'layer': this.layer, 'slot': this.slotindex, 'line_id': this.line.id};\n return call([{\n methodname: 'local_treestudyplan_reorder_studyitems',\n args: { 'items': [iteminfo] } // function was designed to relocate multiple items at once, hence the array\n }])[0].catch(notification.exception);\n },\n onDragEnter(event){\n this.hover.component = event.data;\n this.hover.type = event.type;\n },\n onDragLeave(){\n this.hover.component = null;\n this.hover.type = null;\n },\n maxSpan(){\n // Determine the maximum span for components in this slot\n return this.page.periods - this.slotindex + 1;\n },\n makeType(item){\n return {\n item: true,\n component: false,\n span: item.span,\n type: this.type,\n };\n },\n checkType(type) {\n if(type.type == this.type){\n if(type == 'filter'){\n return true;\n } else if(type.span <= this.maxSpan()){\n return true;\n } else {\n return false;\n }\n } else {\n return false;\n }\n }\n },\n template: `\n <div :class=\"'t-studyline-slot '+type + ' t-studyline-slot-'+slotindex + ' ' + ((slotindex==0)?' t-studyline-firstcolumn ':' ')\n + (current?'current ':' ')\"\n :data-studyline=\"line.id\" ref=\"main\"\n :style='spanCss'\n ><drag v-if=\"item\"\n :key=\"item.id\"\n class=\"t-slot-item\"\n :data=\"item\"\n :type=\"makeType(item)\"\n @cut=\"onCut\"\n ><t-item\n @deleted=\"onCut\"\n v-model=\"value[itemidx]\"\n :plan=\"plan\"\n :line='line'\n :page='page'\n :period='period'\n :maxspan='maxSpan()'\n ></t-item\n ></drag\n ><drop v-else\n :class=\"'t-slot-drop '+type + (layer > 0?' secondary':' primary')\"\n :accepts-type=\"checkType\"\n @drop=\"onDrop\"\n mode=\"cut\"\n @dragenter=\"onDragEnter\"\n @dragleave=\"onDragLeave\"\n ><template v-if=\"hover.component\">\n <div v-if=\"hover.type.item\"\n class=\"t-slot-item feedback\"\n :key=\"hover.component.id\"\n ><t-item v-model=\"hover.component\" dummy></t-item\n ></div\n ><div v-else-if=\"hover.type.type == 'gradable'\"\n class=\"t-slot-item feedback\"\n :key=\"'course-'+hover.component.id\"\n ><t-item-course v-model=\"courseHoverDummy\"></t-item-course></div\n ><div v-else-if=\"hover.type.type == 'filter'\"\n class=\"t-slot-item feedback\"\n key=\"tooldrop\"\n ><t-item-junction v-if=\"hover.component.type == 'junction'\" ></t-item-junction\n ><t-item-start v-else-if=\"hover.component.type == 'start'\" ></t-item-start\n ><t-item-finish v-else-if=\"hover.component.type == 'finish'\" ></t-item-finish\n ><t-item-badge v-else-if=\"hover.component.type == 'badge'\" ></t-item-badge\n ></div\n ><div v-else\n class=\"t-slot-item feedback\"\n :key=\"hover.type\">--{{ hover.type }}--</div\n ></template\n ></drop>\n <t-item-timing-checker hidden\n v-if=\"value && value[itemidx] && value[itemidx].course\"\n ref=\"timingChecker\"\n :maxspan=\"maxSpan()\"\n :page=\"page\"\n :line=\"line\"\n :period=\"period\"\n v-model=\"value[itemidx]\"\n ></t-item-timing-checker>\n </div>\n `,\n });\n\n\n Vue.component('t-item-timing-checker', {\n props: {\n value: {\n type: Object, // t-item model\n },\n page: {\n type: Object, // Studyplanpage\n },\n line: {\n type: Object, // Studyline\n },\n period: {\n type: Object, // Studyplan data\n },\n maxspan: {\n type: Number,\n },\n hidden: {\n type: Boolean,\n default: false,\n }\n },\n computed: {\n endperiod() {\n const endperiodnr = Math.min(this.page.periods,this.period.period + (this.value.span - 1));\n return this.page.perioddesc[endperiodnr-1];\n },\n course_period_matches() {\n const self=this;\n if(self.value && self.value.type == 'course'){\n self.datechanger.coursespan = datespaninfo(self.value.course.startdate,self.value.course.enddate);\n self.datechanger.periodspan = datespaninfo(self.period.startdate,self.endperiod.enddate);\n if( self.datechanger.coursespan.first.getTime() == self.datechanger.periodspan.first.getTime()\n && self.datechanger.coursespan.last.getTime() == self.datechanger.periodspan.last.getTime() ){\n return true;\n }\n else {\n return false;\n }\n } else {\n debug.warn(\"Timing thing not properly configured\",self.value,self.period,self.maxspan);\n return false;\n }\n },\n },\n data() {\n return {\n // Create random id to avoid opening the wrong modals\n id: Math.floor(Math.random() * Date.now()).toString(16),\n text: strings.course_timing,\n datechanger: {\n coursespan: null,\n periodspan: null,\n globals: datechanger_globals,\n }\n };\n },\n methods: {\n validate_course_period() {\n const self = this;\n\n debug.info(\"Validating course and period\");\n if(!(self.course_period_matches)){\n debug.info(\"Course timing does not match period timing\");\n\n if(self.value.course.canupdatecourse){\n if(!self.hidden || !self.datechanger.globals.default){\n // Periods do not match, pop up the date change request\n this.$bvModal.show('t-course-timing-matching-'+this.id);\n } else if (self.datechanger.globals.defaultvalue){\n // go for it without asking\n self.change_course_period();\n }\n }\n else {\n // user is not able to change course timing - show a warning\n if(!self.hidden || !self.datechanger.globals.hidewarn){\n this.$bvModal.show('t-course-timing-warning-'+this.id);\n }\n }\n }\n else {\n debug.info(\"Course timing matches period\",self.datechanger);\n }\n },\n has_filter() {\n const slots = this.page.slots;\n },\n change_course_period() {\n const self=this;\n // Save the state\n if(self.datechanger.globals.default){\n self.datechanger.globals.defaultvalue = true;\n }\n return call([{\n methodname: 'local_treestudyplan_course_period_timing',\n args: { period_id: self.period.id,\n course_id: this.value.course.id,\n span: this.value.span,\n }\n }])[0].catch(notification.exception).done((response) => {\n self.value.course.startdate = response.startdate;\n self.value.course.enddate = response.enddate;\n self.value.course.timing = response.timing;\n self.$emit(\"input\",self.value);\n });\n },\n checkFilterSlotBusy(slotindex) {\n debug.info(\"checking filter\",this.line.slots, slotindex,this.value.layer);\n if (this.line.slots[slotindex]) {\n const list = this.line.slots[slotindex].filters;\n for(const ix in list) {\n if( list[ix].layer == this.value.layer) {\n debug.info(\"Busy:\",list[ix]);\n return list[ix];\n }\n }\n }\n return null;\n },\n nextFreeFilterLayer(slotindex) {\n const layer = this.value.layer;\n const list = this.line.slots[slotindex].filters;\n const usedLayers = [];\n for(const ix in list) {\n usedLayers.push(list[ix].layer);\n }\n let nextlyr = layer+1;\n while (usedLayers.includes(nextlyr)) {\n nextlyr++;\n }\n return nextlyr;\n },\n checkCourseSlotBusy(slotindex) {\n debug.info(\"checking \",this.line.slots, slotindex,this.value.layer);\n if (this.line.slots[slotindex]) {\n const list = this.line.slots[slotindex].courses;\n for(const ix in list) {\n if( list[ix].layer == this.value.layer) {\n debug.info(\"Busy:\",list[ix]);\n return list[ix];\n }\n }\n }\n return null;\n },\n nextFreeCourseLayer(slotindex) {\n const layer = this.value.layer;\n const list = this.line.slots[slotindex].courses;\n const usedLayers = [];\n for(const ix in list) {\n usedLayers.push(list[ix].layer);\n }\n let nextlyr = layer+1;\n while (usedLayers.includes(nextlyr)) {\n nextlyr++;\n }\n return nextlyr;\n },\n shift_collisions(span) {\n // Check all periods for collision\n const shiftfilters = [];\n const shiftcourses = [];\n const items = [];\n for(let i = this.value.slot; i < this.value.slot + span; i++) {\n const busyFilter = this.checkFilterSlotBusy(i);\n if (busyFilter) {\n const nextlyr = this.nextFreeFilterLayer(i);\n items.push({\n id: busyFilter.id,\n layer: nextlyr,\n line_id: this.line.id,\n slot: busyFilter.slot,\n });\n busyFilter.layer = nextlyr;\n }\n const busyCourse = this.checkCourseSlotBusy(i);\n if (busyCourse && busyCourse.id != this.value.id) {\n const nextlyr = this.nextFreeCourseLayer(i);\n items.push({ \n id: busyCourse.id,\n layer: nextlyr,\n line_id: this.line.id,\n slot: busyCourse.slot,\n });\n busyCourse.layer = nextlyr;\n }\n }\n if (items.length > 0){\n call([{\n methodname: 'local_treestudyplan_reorder_studyitems',\n args: { items: items }\n }])[0].catch(notification.exception).done((response) => {\n \n });\n }\n },\n change_span(span) {\n const self=this;\n this.shift_collisions(span);\n return call([{\n methodname: 'local_treestudyplan_set_studyitem_span',\n args: { id: self.value.id,\n span: span\n }\n }])[0].catch(notification.exception).done((response) => {\n self.value.span = response.span;\n self.$emit('input',self.value);\n self.$nextTick(() => {\n self.validate_course_period();\n });\n } );\n },\n format_duration(dsi){\n let s = \"\";\n if(dsi.years == 1){ s += `1 ${this.text.year}, `;}\n else if(dsi.years > 1){ s += `${dsi.years} ${this.text.years}, `;}\n if(dsi.weeks == 1){ s += `1 ${this.text.week}, `;}\n else if(dsi.weeks > 1){ s += `${dsi.weeks} ${this.text.weeks}, `;}\n if(dsi.days == 1){ s += `1 ${this.text.day}, `;}\n else if(dsi.days > 1){ s += `${dsi.days} ${this.text.days}, `;}\n\n return s.toLocaleLowerCase();\n },\n },\n // To avoid the span creeping in the dom where it shouldn't, set display to none if it is hidden\n // This does not affect the modals, which are rendered outside of this element when needed\n template: `\n <div :class=\"'t-item-timing-checker'\" :style=\"hidden?'display: none ':''\">\n <template v-if=\"!hidden\" >\n <span class=\"mr-1\" v-if=\"course_period_matches\">\n <i class=\"text-success fa fa-calendar-check-o\"\n v-b-tooltip.hover.topright :title=\"text.timing_ok\"\n ></i>\n </span>\n <span class=\"mr-1\" v-else>\n <a href='#' @click=\"validate_course_period()\" class=\"text-warning\"\n v-b-tooltip.hover.bottomleft :title=\"text.timing_off\"\n ><i class=\"fa fa-calendar-times-o\"\n ></i\n ><i class=\"fa fa-question-circle text-black-50\"\n style=\"font-size: 0.8em; top: -0.3em; position: relative;\"\n\n ></i\n ></a>\n </span>\n <span class=\"ml-1\" v-b-tooltip.hover.bottomleft :title=\"text.periodspan_desc\"\n >{{ text.periodspan\n }}&nbsp;<b-form-select v-if=\"maxspan > 1\"\n class=\"\"\n size=\"sm\" @change=\"change_span\" v-model=\"value.span\">\n <b-form-select-option v-for=\"(n,i) in maxspan\" :value='n' :key='i'\n >{{ n }}</b-form-select-option>\n </b-form-select\n ><span v-else>{{value.span}}</span>&nbsp;{{\n (value.span == 1)?text.period.toLowerCase():text.periods.toLowerCase()\n }}<i\n class=\"fa fa-question-circle text-black-50\"\n style=\"font-size: 0.8em; top: -0.3em; position: relative;\"\n ></i>\n </span>\n </template>\n <b-modal\n :id=\"'t-course-timing-matching-'+this.id\"\n size=\"lg\"\n :title=\"text.title\"\n @ok=\"change_course_period\"\n :ok-title=\"text.yes\"\n ok-variant=\"danger\"\n :cancel-title=\"text.no\"\n cancel-variant=\"primary\"\n >\n <b-container v-if=\"datechanger.coursespan && datechanger.periodspan && value && value.course\">\n <b-row><b-col cols=\"12\">{{ text.desc }}</b-col></b-row>\n <b-row><b-col cols=\"12\"><div class=\"generalbox alert alert-warning\">{{ text.question }}</div></b-col></b-row>\n <b-row>\n <b-col cols=\"6\">\n <h3> {{ text.course }} </h3>\n <p class=\"mb-0\"><b>{{ value.course.fullname }}</b></p>\n <p class=\"mb-1\"><b>{{ value.course.shortname }}</b></p>\n <p class=\"mb-1\">{{ datechanger.coursespan.formatted.first}} - {{ datechanger.coursespan.formatted.last}}</p>\n <p class=\"mb-0\"><b>{{ text.duration }}</b><br>\n {{ format_duration(datechanger.coursespan)}}</p>\n </b-col>\n <b-col cols=\"6\">\n <h3> {{ text.period }} </h3>\n <p class=\"mb-0\"><b>{{ period.fullname }}</b><b v-if=\"value.span > 1\"> - {{ endperiod.fullname }}</b></p>\n <p class=\"mb-1\"><b>{{ period.shortname }}</b><b v-if=\"value.span > 1\"> - {{ endperiod.shortname }}</b></p>\n <p class=\"mb-1\">{{ datechanger.periodspan.formatted.first}} - {{ datechanger.periodspan.formatted.last}}</p>\n <p class=\"mb-0\"><b>{{ text.duration }}</b><br>\n {{ format_duration(datechanger.periodspan)}}</p>\n </b-col>\n </b-row>\n <b-row v-if='hidden' class=\"pt-2\"><b-col cols=\"12\">\n <b-form-checkbox type=\"checkbox\" v-model=\"datechanger.globals.default\">{{ text.rememberchoice }}</b-form-checkbox>\n </b-col></b-row>\n </b-container>\n </b-modal>\n <b-modal\n :id=\"'t-course-timing-warning-'+this.id\"\n size=\"lg\"\n ok-variant=\"primary\"\n :title=\"text.title\"\n :ok-title=\"text.yes\"\n ok-only\n >\n <b-container v-if=\"datechanger.coursespan && datechanger.periodspan && value && value.course\">\n <b-row><b-col cols=\"12\">{{ text.desc }}</b-col></b-row>\n <b-row><b-col cols=\"12\"><div class=\"generalbox alert alert-warning\">{{ text.warning }}</div></b-col></b-row>\n <b-row>\n <b-col cols=\"6\">\n <h3> {{ text.course }} </h3>\n <p class=\"mb-0\"><b>{{ value.course.fullname }}</b></p>\n <p class=\"mb-1\"><b>{{ value.course.shortname }}</b></p>\n <p class=\"mb-1\">{{ datechanger.coursespan.formatted.first}} - {{ datechanger.coursespan.formatted.last}}</p>\n <p class=\"mb-0\"><b>{{ text.duration }}</b><br>\n {{ format_duration(datechanger.coursespan)}}</p>\n </b-col>\n <b-col cols=>\"6\">\n <h3> {{ text.period }} </h3>\n <p class=\"mb-0\"><b>{{ period.fullname }}</b><b v-if=\"value.span > 1\"> - {{ endperiod.fullname }}</b></p>\n <p class=\"mb-1\"><b>{{ period.shortname }}</b><b v-if=\"value.span > 1\"> - {{ endperiod.shortname }}</b></p>\n <p class=\"mb-1\">{{ datechanger.periodspan.formatted.first}} - {{ datechanger.periodspan.formatted.last}}</p>\n <p class=\"mb-0\"><b>{{ text.duration }}</b><br>\n {{ format_duration(datechanger.periodspan)}}</p>\n </b-col>\n </b-row>\n <b-row v-if='hidden' class=\"pt-2\"><b-col cols=\"12\">\n <b-form-checkbox type=\"checkbox\" v-model=\"datechanger.globals.hidewarn\">{{ text.hidewarning }}</b-form-checkbox>\n </b-col></b-row>\n </b-container>\n </b-modal>\n </div>\n `,\n });\n\n\n Vue.component('t-item', {\n props: {\n value :{\n type: Object,\n default(){ return null;},\n },\n dummy :{\n type: Boolean,\n default() { return false;},\n },\n plan: {\n type: Object, // Studyplan page\n default() { return null;},\n },\n line: {\n type: Object, // Studyplan page\n default() { return null;},\n },\n page: {\n type: Object, // Studyplan page\n default() { return null;},\n },\n period: {\n type: Object, // Studyplan page\n default() { return null;},\n },\n maxspan: {\n type: Number,\n default() { return 0;},\n },\n },\n data() {\n return {\n dragLine: null,\n dragEventListener: null,\n deleteMode: false,\n condition_options: string_keys.conditions,\n text: strings.item_text,\n showContext: false,\n lines: [],\n };\n },\n methods: {\n dragStart(event){\n // Add line between start point and drag image\n this.deleteMode = false;\n let start = document.getElementById('studyitem-'+this.value.id);\n let dragelement= document.getElementById('t-item-cdrag-'+this.value.id);\n dragelement.style.position = 'fixed';\n dragelement.style.left = event.position.x+'px';\n dragelement.style.top = event.position.y+'px';\n this.dragLine = new SimpleLine(start,dragelement,{\n color: \"#777\",\n gravity: {\n start: LINE_GRAVITY,\n end: LINE_GRAVITY,\n },\n });\n // Add separate event listener to reposition mouse move\n document.addEventListener(\"mousemove\",this.onMouseMove);\n },\n dragEnd(){\n if(this.dragLine !== null) {\n this.dragLine.remove();\n }\n let dragelement = document.getElementById('t-item-cdrag-'+this.value.id);\n dragelement.style.removeProperty('left');\n dragelement.style.removeProperty('top');\n dragelement.style.removeProperty('position');\n document.removeEventListener(\"mousemove\",this.onMouseMove);\n },\n onMouseMove: function(event){\n let dragelement = document.getElementById('t-item-cdrag-'+this.value.id);\n dragelement.style.position = 'fixed';\n dragelement.style.left = event.clientX+'px';\n dragelement.style.top = event.clientY+'px';\n // line will follow automatically\n },\n onDrop(event){\n let from_id = event.data.id;\n let to_id = this.value.id;\n this.redrawLines();\n call([{\n methodname: 'local_treestudyplan_connect_studyitems',\n args: { 'from_id': from_id, 'to_id': to_id }\n }])[0].then((response)=>{\n let conn = {'id': response.id, 'from_id': response.from_id, 'to_id': response.to_id};\n ItemEventBus.$emit(\"createdConnection\",conn);\n this.value.connections.in.push(conn);\n }).catch(notification.exception);\n },\n redrawLine(conn){\n const lineColor = \"var(--success)\";\n const start = document.getElementById(`studyitem-${conn.from_id}`);\n const end = document.getElementById(`studyitem-${conn.to_id}`);\n\n // delete old line\n if(this.lines[conn.to_id]){\n this.lines[conn.to_id].remove();\n delete this.lines[conn.to_id];\n }\n // create a new line if the start and finish items are visible\n if(start !== null && end !== null && isVisible(start) && isVisible(end)){\n this.lines[conn.to_id] = new SimpleLine( start,end,{\n color: lineColor,\n gravity: {\n start: LINE_GRAVITY,\n end: LINE_GRAVITY,\n },\n });\n }\n\n },\n deleteLine(conn){\n const self = this;\n call([{\n methodname: 'local_treestudyplan_disconnect_studyitems',\n args: { 'from_id': conn.from_id, 'to_id': conn.to_id }\n }])[0].then((response)=>{\n if(response.success){\n this.removeLine(conn);\n // send disconnect event on message bus, so the connection on the other end can delete it too\n ItemEventBus.$emit(\"connectionDisconnected\",conn);\n // Remove connection from our outgoing list\n let index = self.value.connections.out.indexOf(conn);\n self.value.connections.out.splice(index, 1);\n }\n }).catch(notification.exception);\n },\n highlight(conn){\n if(this.lines[conn.to_id]){\n this.lines[conn.to_id].setConfig({color:\"var(--danger)\",});\n }\n },\n normalize(conn){\n if(this.lines[conn.to_id]){\n this.lines[conn.to_id].setConfig({color:\"var(--success)\",});\n }\n },\n updateItem() {\n call([{\n methodname: 'local_treestudyplan_edit_studyitem',\n args: { 'id': this.value.id,\n 'conditions': this.value.conditions,\n 'continuation_id': this.value.continuation_id,}\n }])[0].catch(notification.exception);\n },\n doShowContext(event) {\n if(this.hasContext){\n this.showContext=true;\n event.preventDefault();\n }\n },\n redrawLines(){\n if(this.value.connections && this.value.connections.out){\n for(let i in this.value.connections.out){\n let conn = this.value.connections.out[i];\n this.redrawLine(conn);\n }\n }\n },\n\n // EVENT LISTENERS\n onCreatedConnection(conn){\n if(conn.from_id == this.value.id){\n this.value.connections.out.push(conn);\n this.redrawLine(conn);\n }\n },\n // Listener for the signal that a connection was removed by the outgoing item\n onRemovedConnection(conn){\n if(this.value.connections && this.value.connections.out){\n for(let i in this.value.connections.in){\n let c_in = this.value.connections.in[i];\n if(conn.id == c_in.id){\n self.value.connections.out.splice(i, 1);\n }\n }\n }\n },\n // Listener for reposition events\n // When an item in the list is repositioned, all lines need to be redrawn\n onRePositioned(){\n this.redrawLines();\n },\n // When an item is disPositioned - (temporarily) removed from the list,\n // all connections need to be deleted.\n onDisPositioned(re_id){\n if(this.value.connections && this.value.connections.out){\n for(let i in this.value.connections.out){\n let conn = this.value.connections.out[i];\n if(conn.to_id == re_id){\n this.removeLine(conn);\n }\n }\n }\n },\n\n // When an item is deleted\n // all connections to/from that item need to be cleaned up\n onItemDeleted(item_id){\n const self = this;\n if(this.value.connections && this.value.connections.out){\n for(const i in this.value.connections.out){\n let conn = this.value.connections.out[i];\n if(conn.to_id == item_id){\n self.removeLine(conn);\n self.value.connections.out.splice(i, 1);\n }\n }\n }\n if(this.value.connections && this.value.connections.in){\n for(const i in this.value.connections.in){\n let conn = this.value.connections.in[i];\n if(conn.from_id == item_id){\n self.value.connections.out.splice(i, 1);\n }\n }\n }\n },\n\n onRedrawLines(){\n this.redrawLines();\n },\n\n removeLine(conn){\n if(this.lines[conn.to_id]){\n this.lines[conn.to_id].remove();\n delete this.lines[conn.to_id];\n }\n },\n deleteItem(){\n const self = this;\n const msgparams = {\n item: this.text['type_' + this.value.type].toLocaleLowerCase(),\n name: (this.value.type=='course')?this.value.course.displayname:\"\",\n line: (this.line) ? this.line.name : \"\",\n period: (this.period) ? this.period.fullname : this.plan.name,\n };\n\n this.$bvModal.msgBoxConfirm(strformat(this.text.item_delete_message, msgparams), {\n okVariant : 'danger',\n okTitle : this.text.ok,\n cancelTitle: this.text.cancel,\n }).then(value => {\n if ( value ) {\n // Confirmed to delete.\n call([{\n methodname: 'local_treestudyplan_delete_studyitem',\n args: { 'id': self.value.id, }\n }])[0].then(function(response){\n if(response.success == true){\n self.$emit(\"deleted\",{ data: self.value });\n }\n }).catch(notification.exception);\n }\n }).catch(err => {\n debug.console.error(err);\n });\n }\n\n\n },\n computed: {\n hasConnectionsOut() {\n return !([\"finish\",].includes(this.value.type));\n },\n hasConnectionsIn() {\n return !([\"start\",].includes(this.value.type));\n },\n hasContext() {\n return ['start','junction','finish'].includes(this.value.type);\n }\n },\n created(){\n // Add event listeners on the message bus\n // But only if not in \"dummy\" mode - mode which is used for droplist placeholders\n // Since an item is \"fully made\" with all references, not specifying dummy mode really messes things up\n if(!this.dummy){\n\n // Listener for the signal that a new connection was made and needs to be drawn\n // Sent by the incoming item - By convention, outgoing items are responsible for drawing the lines\n ItemEventBus.$on('createdConnection', this.onCreatedConnection);\n // Listener for the signal that a connection was removed by the outgoing item\n ItemEventBus.$on('removedConnection', this.onRemovedConnection);\n // Listener for reposition events\n // When an item in the list is repositioned, all lines need to be redrawn\n ItemEventBus.$on('rePositioned', this.onRePositioned);\n // When an item is disPositioned - (temporarily) removed from the list,\n // all connections need to be deleted.\n ItemEventBus.$on('disPositioned', this.onDisPositioned);\n // When an item is deleted\n // all connections to/from that item need to be cleaned up\n ItemEventBus.$on('itemDeleted', this.onItemDeleted);\n ItemEventBus.$on('redrawLines', this.onRedrawLines);\n\n\n }\n },\n mounted(){\n // Initialize connection lines when mounting\n // But only if not in \"dummy\" mode - mode which is used for droplist placeholders\n // Since an item is \"fully made\" with all references, not specifying dummy mode really messes things up\n\n if(!this.dummy)\n {\n this.redrawLines();\n setTimeout(()=>{\n ItemEventBus.$emit(\"rePositioned\",this.value.id);\n },10);\n }\n },\n beforeDestroy(){\n if(!this.dummy) {\n for(let i in this.value.connections.out){\n let conn = this.value.connections.out[i];\n this.removeLine(conn);\n }\n ItemEventBus.$emit(\"disPositioned\",this.value.id);\n\n // Remove event listeners\n ItemEventBus.$off('createdConnection', this.onCreatedConnection);\n ItemEventBus.$off('removedConnection', this.onRemovedConnection);\n ItemEventBus.$off('rePositioned', this.onRePositioned);\n ItemEventBus.$off('disPositioned', this.onDisPositioned);\n ItemEventBus.$off('itemDeleted', this.onItemDeleted);\n ItemEventBus.$off('redrawLines', this.onRedrawLines);\n }\n },\n beforeUpdate(){\n },\n updated(){\n if(!this.dummy) {\n this.redrawLines();\n }\n },\n template: `\n <div class=\"t-item-base\" :id=\"'studyitem-'+value.id\">\n <t-item-course\n v-if=\"value.type == 'course'\"\n @deleterq=\"deleteItem\"\n v-model=\"value\"\n :plan='plan'\n :line='line'\n :page='page'\n :period='period'\n :maxspan='maxspan'\n ></t-item-course>\n <t-item-junction\n v-if=\"value.type == 'junction'\"\n @deleterq=\"deleteItem\"\n v-model=\"value\"\n ></t-item-junction>\n <t-item-start\n v-if=\"value.type == 'start'\"\n @deleterq=\"deleteItem\"\n v-model=\"value\"\n ></t-item-start>\n <t-item-finish\n v-if=\"value.type == 'finish'\"\n @deleterq=\"deleteItem\"\n v-model=\"value\"\n ></t-item-finish>\n <t-item-badge\n v-if=\"value.type == 'badge'\"\n @deleterq=\"deleteItem\"\n v-model=\"value\"\n ></t-item-badge>\n <t-item-invalid\n v-if=\"value.type == 'invalid'\"\n @deleterq=\"deleteItem\"\n v-model=\"value\"\n ></t-item-invalid>\n <drop v-if='!dummy && hasConnectionsIn' accepts-type=\"linestart\"\n :id=\"'t-item-cend-'+value.id\"\n class=\"t-item-connector-end\"\n mode=\"copy\"\n @drop=\"onDrop\"\n ><svg width='5px' height='10px'><rect ry=\"1px\" rx=\"1px\" y=\"0px\" x=\"0px\" height=\"10px\" width=\"5px\"/></svg></drop>\n <drag v-if='!dummy && hasConnectionsOut' type=\"linestart\"\n :id=\"'t-item-cstart-'+value.id\"\n :class=\"'t-item-connector-start ' + ((deleteMode&&value.connections.out.length)?'deleteMode':'')\"\n :data=\"value\"\n @dragstart=\"dragStart\"\n @dragend=\"dragEnd\"\n @click=\"deleteMode = (value.connections.out.length)?(!deleteMode):false\"\n >\n <svg width='5px' height='10px'><rect ry=\"1px\" rx=\"1px\" y=\"0px\" x=\"0px\" height=\"10px\" width=\"5px\"/></svg>\n <template v-slot:drag-image=\"{data}\"> <i :id=\"'t-item-cdrag-'+value.id\" class=\"fa\"></i>\n </template>\n </drag>\n <div class=\"deletebox\" v-if=\"deleteMode && value.connections.out.length > 0\"\n >\n <a v-for=\"conn in value.connections.out\"\n @click=\"deleteLine(conn)\"\n @mouseenter=\"highlight(conn)\"\n @mouseleave=\"normalize(conn)\"\n class=\"t-connection-delete text-danger\"\n :title=\"conn.id\">\n <i class=\"fa fa-trash\"></i>\n </a>\n </div>\n <a v-if=\"hasContext\" class=\"t-item-config\"\n v-b-modal=\"'t-item-config-'+value.id\" href=\"#\" @click.prevent=\"\"><i class=\"fa fa-gear\"></i></a>\n <b-modal no-body class=\"\"\n :id=\"'t-item-config-'+value.id\"\n :title=\"text['item_configuration']\"\n scrollable\n ok-only\n class=\"t-item-contextview b-modal-justify-footer-between\"\n >\n <b-form-group\n v-if=\"value.type != 'start'\"\n :label=\"text.select_conditions\"\n >\n <b-form-select size=\"sm\"\n @input=\"updateItem\"\n v-model=\"value.conditions\"\n :options=\"condition_options\"\n ></b-form-select>\n </b-form-group>\n\n <template #modal-footer=\"{ ok, cancel, hide }\" >\n <a href='#' @click='deleteItem()' class=\"text-danger\"\n ><i class=\"fa fa-trash\"></i>\n {{ text.delete }}\n </a>\n <b-button size=\"sm\" variant=\"primary\" @click=\"ok()\">\n {{ text.ok }}\n </b-button>\n </template>\n\n </b-modal>\n </div>\n `,\n });\n\n Vue.component('t-item-invalid', {\n props: {\n 'value' :{\n type: Object,\n default: function(){ return null;},\n },\n },\n data() {\n return {\n text: strings.invalid,\n };\n },\n methods: {\n },\n template: `\n <div class=\"t-item-invalid\">\n <b-card no-body >\n <b-row no-gutters>\n <b-col md=\"1\">\n <span class=\"t-timing-indicator timing-invalid\"></span>\n </b-col>\n <b-col md=\"11\">\n <b-card-body class=\"align-items-center\">\n <i class=\"fa fa-exclamation\"></i> {{text.error}}\n <a href='#' @click='$emit(\"deleterq\")' class=\"text-danger\"\n ><i class=\"fa fa-trash\"></i></a>\n </b-card-body>\n </b-col>\n </b-row>\n </b-card>\n </div>\n `,\n });\n\n //TAG: Course item\n Vue.component('t-item-course', {\n props: {\n value:{\n type: Object,\n default(){ return null;},\n },\n plan:{\n type: Object,\n default(){ return null;},\n },\n line:{\n type: Object,\n default(){ return null;},\n },\n page: {\n type: Object, // PAge data\n default() { return null;}\n },\n period: {\n type: Object, // Period data\n default() { return null;}\n },\n maxspan: {\n type: Number,\n default() { return 0;}\n },\n },\n data() {\n return {\n condition_options: string_keys.conditions,\n text: strings.item_course_text,\n };\n },\n computed: {\n useItemConditions() {\n if(this.plan && this.plan.aggregation_info && this.plan.aggregation_info.useItemConditions !== undefined){\n return this.plan.aggregation_info.useItemConditions;\n }\n else {\n return false;\n }\n },\n\n configurationState(){\n if(this.hasGrades() || this.hasCompletions() || this.hasCompetencies()) {\n return \"t-configured-ok\";\n } else {\n return \"t-configured-alert\";\n }\n },\n\n configurationIcon(){\n if(this.hasGrades() || this.hasCompletions() || this.hasCompetencies()) {\n return \"check\";\n } else {\n return \"exclamation-circle\";\n }\n },\n startdate(){\n return format_date(this.value.course.startdate);\n },\n enddate(){\n if(this.value.course.enddate){\n return format_date(this.value.course.enddate);\n }\n else {\n return this.text.noenddate;\n }\n }\n\n },\n methods: {\n hasGrades() {\n if(this.value.course.grades && this.value.course.grades.length > 0){\n for(const g of this.value.course.grades){\n if(g.selected){\n return true;\n }\n }\n }\n return false;\n },\n hasCompletions() {\n if(this.value.course.completion && this.value.course.completion.conditions) {\n for(const cgroup of this.value.course.completion.conditions){\n if(cgroup.items && cgroup.items.length > 0){\n return true;\n }\n }\n }\n return false;\n },\n hasCompetencies() {\n if(this.value.course.competency && this.value.course.competency.competencies) {\n return (this.value.course.competency.competencies.length > 0);\n }\n return false;\n },\n includeChanged(newValue,g){\n call([{\n methodname: 'local_treestudyplan_include_grade',\n args: { 'grade_id': g.id,\n 'item_id': this.value.id,\n 'include': newValue,\n 'required': g.required,\n }\n }])[0].catch(notification.exception);\n },\n requiredChanged(newValue,g){\n call([{\n methodname: 'local_treestudyplan_include_grade',\n args: { 'grade_id': g.id,\n 'item_id': this.value.id,\n 'include': g.selected,\n 'required': newValue,\n }\n }])[0].catch(notification.exception);\n },\n updateConditions() {\n call([{\n methodname: 'local_treestudyplan_edit_studyitem',\n args: { 'id': this.value.id,\n 'conditions': this.value.conditions,\n }\n }])[0].catch(notification.exception);\n },\n },\n created() {\n\n },\n template: `\n <div class=\"t-item-course\">\n <b-card no-body >\n <div class='d-flex flex-wrap mr-0 ml-0'>\n <div>\n <span\n :title=\"text['coursetiming_'+value.course.timing]\"\n v-b-popover.hover.top=\"startdate+' - '+enddate\"\n :class=\"'t-timing-indicator timing-'+value.course.timing\"></span>\n </div>\n <div class=\"flex-fill align-items-center\">\n <b-card-body >\n <a class=\"t-item-course-config\"\n v-b-modal=\"'t-item-course-config-'+value.id\"\n href=\"#\" @click.prevent=\"\"\n ><i :class=\"'fa fa-'+configurationIcon+' ' + configurationState\"></i></a>\n <a v-b-modal=\"'t-item-course-config-'+value.id\"\n :id=\"'t-item-course-details-'+value.id\"\n :href=\"'/course/view.php?id='+value.course.id\"\n @click.prevent.stop=\"\">{{ value.course.displayname }}</a>\n </b-card-body>\n </div>\n </div>\n <b-modal class=\"\"\n :id=\"'t-item-course-config-'+value.id\"\n :title=\"value.course.displayname + ' - ' + value.course.fullname\"\n ok-only\n size=\"lg\"\n scrollable\n class=\"b-modal-justify-footer-between\"\n >\n <template #modal-header>\n <div>\n <h1><a :href=\"'/course/view.php?id='+value.course.id\" target=\"_blank\"\n ><i class=\"fa fa-graduation-cap\"></i> {{ value.course.fullname }}</a>\n <a v-if='!!value.course.completion'\n :href=\"'/course/completion.php?id='+value.course.id\" target=\"_blank\"\n :title=\"text.configure_completion\"><i class=\"fa fa-gear\"></i></a>\n </h1>\n {{ value.course.context.path.join(\" / \")}} / {{value.course.shortname}}\n </div>\n <div class=\"r-course-detail-header-right\">\n <div :class=\"'r-timing-'+value.course.timing\">\n {{text['coursetiming_'+value.course.timing]}}<br>\n {{ startdate }} - {{ enddate }}\n </div>\n <t-item-timing-checker\n class=\"mt-1\"\n :maxspan=\"maxspan\"\n :page=\"page\"\n :line=\"line\"\n :period=\"period\"\n v-model=\"value\"\n ></t-item-timing-checker>\n </div>\n </template>\n\n <s-course-extrafields \n v-if=\"value.course.extrafields\" \n v-model=\"value.course.extrafields\"\n position=\"above\"\n ></s-course-extrafields>\n <t-item-course-grades\n v-if='!!value.course.grades && value.course.grades.length > 0'\n v-model='value' :plan=\"plan\"\n ></t-item-course-grades>\n <t-item-course-completion\n v-if='!!value.course.completion'\n v-model='value.course.completion'\n :course='value.course'\n ></t-item-course-completion>\n <t-item-course-competency\n v-if='!!value.course.competency'\n v-model='value.course.competency'\n :item='value'\n ></t-item-course-competency>\n <s-course-extrafields \n v-if=\"value.course.extrafields\" \n v-model=\"value.course.extrafields\"\n position=\"below\"\n ></s-course-extrafields>\n <template #modal-footer=\"{ ok, cancel, hide }\" >\n <a href='#' @click='$emit(\"deleterq\")' class=\"text-danger\"\n ><i class=\"fa fa-trash\"></i>\n {{ text.delete }}\n </a>\n <b-button size=\"sm\" variant=\"primary\" @click=\"ok()\">\n {{ text.ok }}\n </b-button>\n </template>\n </b-modal>\n </b-card></div>\n `,\n });\n\n\n Vue.component('t-item-course-grades', {\n props: {\n 'value' :{\n type: Object,\n default(){ return null;},\n },\n 'plan' :{\n type: Object,\n default(){ return null;},\n },\n },\n data() {\n return {\n condition_options: string_keys.conditions,\n text: strings.item_course_text,\n };\n },\n computed: {\n useRequiredGrades() {\n if(this.plan && this.plan.aggregation_info && this.plan.aggregation_info.useRequiredGrades !== undefined){\n return this.plan.aggregation_info.useRequiredGrades;\n }\n else {\n return false;\n }\n },\n selectedgrades(){\n let list = [];\n for(let ix in this.value.course.grades){\n let g = this.value.course.grades[ix];\n if(g.selected){\n list.push(g);\n }\n }\n return list;\n },\n },\n methods: {\n includeChanged(newValue,g){\n call([{\n methodname: 'local_treestudyplan_include_grade',\n args: { 'grade_id': g.id,\n 'item_id': this.value.id,\n 'include': newValue,\n 'required': g.required,\n }\n }])[0].catch(notification.exception);\n },\n requiredChanged(newValue,g){\n call([{\n methodname: 'local_treestudyplan_include_grade',\n args: { 'grade_id': g.id,\n 'item_id': this.value.id,\n 'include': g.selected,\n 'required': newValue,\n }\n }])[0].catch(notification.exception);\n },\n },\n created() {\n\n },\n template: `\n <div>\n <b-form-group\n :label=\"text.select_grades\"\n ><ul class=\"t-item-module-children\">\n <li class=\"t-item-course-gradeinfo\">\n <span class='t-item-course-chk-lbl'>{{text.grade_include}}</span\n ><span v-if=\"useRequiredGrades\" class='t-item-course-chk-lbl'>{{text.grade_require}}</span>\n </li>\n <li class=\"t-item-course-gradeinfo\" v-for=\"g in value.course.grades\">\n <b-form-checkbox inline\n @change=\"includeChanged($event,g)\" v-model=\"g.selected\"\n ></b-form-checkbox>\n <b-form-checkbox v-if=\"useRequiredGrades\" inline :disabled=\"!g.selected\"\n @change=\"requiredChanged($event,g)\" v-model=\"g.required\"\n ></b-form-checkbox>\n <span :title=\"g.typename\" v-html=\"g.icon\"></span>\n <s-edit-mod\n :title=\"value.course.fullname\"\n @saved=\"(fd) => g.name = fd.get('name')\"\n v-if=\"g.cmid > 0\"\n :cmid=\"g.cmid\"\n :coursectxid=\"value.course.ctxid\"\n genericonly><span v-html=\"g.name\"></span></s-edit-mod>\n </li>\n </ul>\n </b-form-group>\n </div>\n `,\n });\n\n Vue.component('t-item-course-completion',{\n props: {\n value : {\n type: Object,\n default: function(){ return {};},\n },\n guestmode: {\n type: Boolean,\n default: false,\n },\n course: {\n type: Object,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n text: strings.completion,\n };\n },\n created(){\n const self = this;\n // Get text strings for condition settings\n let stringkeys = [];\n for(const key in this.text){\n stringkeys.push({ key: key, component: 'local_treestudyplan'});\n }\n get_strings(stringkeys).then(function(strings){\n let i = 0;\n for(const key in self.text){\n self.text[key] = strings[i];\n i++;\n }\n });\n },\n computed: {\n hasCompletions() {\n if(this.value.conditions) {\n for(const cgroup of this.value.conditions){\n if(cgroup.items && cgroup.items.length > 0){\n return true;\n }\n }\n }\n return false;\n },\n },\n methods: {\n completion_icon(completion) {\n switch(completion){\n case \"progress\":\n return \"exclamation-circle\";\n case \"complete\":\n return \"check-circle\";\n case \"complete-pass\":\n return \"check-circle\";\n case \"complete-fail\":\n return \"times-circle\";\n default: // case \"incomplete\"\n return \"circle-o\";\n }\n },\n\n completion_tag(cgroup){\n return cgroup.completion?'completed':'incomplete';\n }\n },\n template: `\n <table class=\"r-item-course-grade-details\">\n <tr v-if=\"hasCompletions\">\n <td colspan='2'><span v-if=\"value.aggregation == 'all'\">{{ text.aggregation_overall_all}}</span\n ><span v-else>{{ text.aggregation_overall_any}}</span></td>\n </tr>\n <tr v-else>\n <td colspan='2'>{{text.completion_not_configured}}!\n <br/><a :href=\"'/course/completion.php?id='+course.id\" target='_blank'>{{text.configure_completion}}</a>\n </td>\n </tr>\n <template v-for='cgroup in value.conditions'>\n <tr>\n <th colspan='2'><span v-if=\"cgroup.items.length > 1\"\n ><span v-if=\"cgroup.aggregation == 'all'\">{{ text.aggregation_all}}</span\n ><span v-else>{{ text.aggregation_any}}</span></span>\n {{cgroup.title}}</th>\n </tr>\n <tr v-for='ci in cgroup.items'>\n <td><span v-html='ci.details.criteria'></span>\n </td>\n <td v-if=\"ci.details.requirement\" class=\"font-italic\">\n {{ci.details.requirement}}\n </td>\n </tr>\n </template>\n </table>\n `,\n });\n\n //TAG: Course competency\n Vue.component('t-item-course-competency',{\n props: {\n value : {\n type: Object,\n default: function(){ return {};},\n },\n guestmode: {\n type: Boolean,\n default: false,\n },\n item: {\n type: Object,\n default: function(){ return { id: null};},\n }\n },\n data() {\n return {\n text: strings.competency,\n };\n },\n created(){\n\n },\n computed: {\n hasCompletions() {\n if(this.value.conditions) {\n for(const cgroup of this.value.conditions){\n if(cgroup.items && cgroup.items.length > 0){\n return true;\n }\n }\n }\n return false;\n },\n },\n methods: {\n pathtags(competency) {\n const path = competency.path;\n let s = \"\";\n for (const ix in path) {\n const p = path[ix];\n if ( ix > 0) {\n s += \" / \";\n }\n let url;\n if (p.type =='competency') { \n url = `/admin/tool/lp/competencies.php?competencyid=${p.id}`;\n } else {\n url = `/admin/tool/lp/competencies.php?competencyframeworkid=${p.id}&pagecontextid=${p.contextid}`;\n }\n \n s += `<a href=\"${url}\">${p.title}</a>`;\n }\n return s;\n },\n requiredChanged(newValue,c){\n call([{\n methodname: 'local_treestudyplan_require_competency',\n args: { 'competency_id': c.id,\n 'item_id': this.item.id,\n 'required': newValue,\n }\n }])[0].catch(notification.exception);\n },\n },\n template: `\n <table class=\"t-item-course-competency-list\">\n <tr v-if=\"value.competencies.length == 0\">\n <td colspan='2'>{{text.competency_not_configured}}\n <br><a :href=\"'/admin/tool/lp/coursecompetencies.php?courseid='+item.course.id\" target='_blank'>{{text.configure_competency}}</a>\n </td>\n </tr>\n <template v-else>\n <tr class='t-item-course-competency-headers'>\n <th>{{text.heading}}</th>\n <th></th>\n <th>{{text.required}}</th>\n </tr>\n <tr v-for='c in value.competencies'>\n <td :colspan=\"(c.details)?1:2\"><a href='#' v-b-modal=\"'modal-competency-id-'+c.id\"><span v-html='c.title'></span></a></td>\n <td class='details' v-if=\"c.details\">\n <a href='#' v-b-modal=\"'modal-competency-id-'+c.id\"><span v-html='c.details'></span></a>\n </td>\n <td>\n <b-form-checkbox inline\n @change=\"requiredChanged($event,c)\"\n v-model=\"c.required\"\n >{{ text.required }}</b-form-checkbox>\n </td>\n <b-modal :id=\"'modal-competency-id-'+c.id\"\n size=\"lg\"\n ok-only\n centered\n scrollable\n >\n <template #modal-header>\n <div>\n <h1><i class=\"fa fa-puzzle-piece\"></i>\n <a :href=\"'/admin/tool/lp/competencies.php?competencyid='+c.id\" target=\"_blank\"\n >{{c.title}} {{c.details}} </a\n ></h1>\n <div><span v-html=\"pathtags(c)\"></span></div>\n </div>\n </template>\n <div class=\"mb-2\" v-if=\"c.description\"><span v-html='c.description'></span></div>\n\n <template v-if=\"c.rule && c.children\">\n <div>{{ c.ruleoutcome }} {{ text.when}} <span v-html=\"c.rule.toLocaleLowerCase()\"></span></div>\n <table v-if=\"c.children\" class='t-item-course-competency-list'>\n <tr class='t-item-course-competency-headers'>\n <th>{{text.heading}}</th>\n <th></th>\n <th>{{text.required}}</th>\n </tr>\n <tr v-for=\"cc in c.children\">\n <td :colspan=\"(c.details)?1:2\" ><span v-html='cc.title'></span></td>\n <td class='details' v-if=\"cc.details\"><span v-html='cc.details'></span></td>\n <td><span class=\"text-info\">{{ cc.points }} {{ text.points }}</span></td>\n <td><span class=\"text-danger\" v-if='cc.required'>{{ text.required }}</span></td>\n </tr>\n </table>\n </template>\n </b-modal>\n </tr>\n </template>\n </table>\n `,\n });\n /************************************\n * *\n * Toolbox list components *\n * *\n ************************************/\n Vue.component('t-item-junction',{\n props: {\n value : {\n type: Object,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n condition_options: string_keys.conditions,\n };\n },\n methods: {\n\n },\n template: `\n <div class='t-item-junction t-item-filter'>\n <i class=\"fa fa-check-circle\"></i>\n </div>\n `,\n });\n\n Vue.component('t-item-finish',{\n props: {\n value : {\n type: Object,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n };\n },\n methods: {\n },\n template: `\n <div class='t-item-finish t-item-filter'>\n <i class=\"fa fa-stop-circle\"></i>\n </div>\n `,\n });\n\n Vue.component('t-item-start',{\n props: {\n value : {\n type: Object,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n };\n },\n created(){\n\n },\n methods: {\n },\n template: `\n <div class='t-item-start t-item-filter'>\n <i class=\"fa fa-play-circle\"></i>\n </div>\n `,\n });\n\n Vue.component('t-item-badge',{\n props: {\n value : {\n type: Object,\n default: function(){ return { badge: {}};},\n },\n },\n data() {\n return {\n txt: strings,\n text: strings.item_text,\n };\n },\n methods: {\n },\n template: `\n <div class='t-item-badge t-item-filter' v-b-tooltip.hover :title=\"value.badge.name\">\n <svg class=\"t-badge-backdrop \" width='50px' height='50px' viewBox=\"0 0 100 100\">\n <title>{{value.badge.name}}</title>\n <circle cx=\"50\" cy=\"50\" r=\"46\"\n style=\"stroke: currentcolor; stroke-width: 4; fill: currentcolor; fill-opacity: 0.5;\"/>\n <image class=\"badge-image\" clip-path=\"circle() fill-box\"\n :href=\"value.badge.imageurl\" x=\"12\" y=\"12\" width=\"76\" height=\"76\"/>\n </svg>\n <a class=\"t-item-config badge\"\n v-b-modal=\"'t-item-badge-details-'+value.id\" href=\"#\" @click.prevent=\"\"><i class=\"fa fa-gear\"></i></a>\n <b-modal class=\"\"\n :id=\"'t-item-badge-details-'+value.id\"\n :title=\"value.badge.name\"\n size=\"lg\"\n ok-only\n centered\n scrollable\n class=\"b-modal-justify-footer-between\"\n >\n <template #modal-header>\n <div>\n <h1><i class=\"fa fa-certificate\"></i>\n <a :href=\"value.badge.infolink\" target=\"_blank\"\n >{{ value.badge.name }}</a\n ></h1>\n </div>\n </template>\n <b-container fluid>\n <b-row><b-col cols=\"3\">\n <img :src=\"value.badge.imageurl\"/>\n </b-col><b-col cols=\"9\">\n <p>{{value.badge.description}}</p>\n <ul class=\"list-unstyled w-100 border-grey border-top border-bottom pt-1 pb-1 mb-1\"\n v-if=\"value.badge.criteria\"><li v-for=\"crit in value.badge.criteria\"\n ><span v-html='crit'></span></li></ul>\n <p><strong><i class=\"fa fa-link\"></i>\n <a :href=\"value.badge.infolink\">{{ txt.badge.badgeinfo }}</a></strong></p>\n </b-col></b-row>\n </b-container>\n <template #modal-footer=\"{ ok, cancel, hide }\" >\n <a href='#' @click='$emit(\"deleterq\")' class=\"text-danger\"\n ><i class=\"fa fa-trash\"></i>\n {{ text.delete }}\n </a>\n <b-button size=\"sm\" variant=\"primary\" @click=\"ok()\">\n {{ text.ok }}\n </b-button>\n </template>\n </b-modal>\n\n </div>\n `,\n });\n\n Vue.component('t-coursecat-list',{\n props: {\n value : {\n type: Array,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n };\n },\n methods: {\n },\n template: `\n <ul class=\"t-coursecat-list\">\n <t-coursecat-list-item\n v-for=\"coursecat,idx in value\"\n v-model=\"value[idx]\"\n :key=\"coursecat.id\"></t-coursecat-list-item>\n </ul>\n `,\n });\n\n Vue.component('t-coursecat-list-item',{\n props: {\n value : {\n type: Object,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n loading: false,\n\n };\n },\n computed: {\n showSpinner() {\n return this.canLoadMore();\n },\n hasDetails() {\n return (this.value.haschildren || this.value.hascourses);\n }\n\n },\n methods: {\n canLoadMore() {\n return (this.value.haschildren && (!this.value.children || this.value.children.length == 0)) ||\n (this.value.hascourses && (!this.value.courses || this.value.courses.length == 0));\n },\n onShowDetails(){\n const self = this;\n if(this.canLoadMore()) {\n call([{\n methodname: 'local_treestudyplan_get_category',\n args: { \"id\": this.value.id}\n }])[0].then(function(response){\n self.$emit('input', response);\n }).catch(notification.exception);\n }\n }\n },\n template: `\n <li class=\"t-coursecat-list-item\">\n <span v-if=\"hasDetails\" v-b-toggle=\"'coursecat-'+value.id\">\n <i class=\"when-closed fa fa-caret-right\"></i>\n <i class=\"when-open fa fa-caret-down\"></i>\n <span class=\"t-coursecat-heading\">\n <i class=\"t-coursecat-list-item fa fa-tasks\"></i>\n {{ value.category.name }}\n </span>\n </span>\n <span v-else>\n <span class=\"t-coursecat-heading\">\n <i class=\"t-coursecat-list-item fa fa-tasks\"></i>\n {{ value.category.name }}\n </span>\n </span>\n <b-collapse v-if=\"hasDetails\" :id=\"'coursecat-'+value.id\"\n @show=\"onShowDetails\" :visible=\"!!(value.children) || !!(value.courses)\">\n <b-spinner class=\"ml-4\" v-if=\"showSpinner\" small variant=\"primary\"></b-spinner>\n <t-coursecat-list v-if=\"value.children\" v-model=\"value.children\"></t-coursecat-list>\n <t-course-list v-if=\"value.courses\" v-model=\"value.courses\"></t-course-list>\n </b-collapse>\n </li>\n `,\n });\n\n Vue.component('t-course-list',{\n props: {\n value : {\n type: Array,\n default: function(){ return {};},\n },\n },\n data() {\n return {\n };\n },\n methods: {\n makeType(){\n return {\n item: false,\n component: true,\n span: 1, //TODO: Detect longer courses and give them an appropriate span\n type: 'gradable',\n };\n },\n },\n template: `\n <ul class=\"t-course-list\">\n <li class=\"t-course-list-item\" v-for=\"course in value\" :key=\"course.id\">\n <drag\n class=\"draggable-course\"\n :data=\"course\"\n :type=\"makeType()\"\n @cut=\"\"\n >\n <i class=\"t-course-list-item fa fa-book\"></i> {{ course.shortname }} - {{ course.fullname }}\n </drag>\n </li>\n </ul>\n `,\n });\n\n },\n};\n"],"names":["STUDYPLAN_EDITOR_FIELDS","PERIOD_EDITOR_FIELDS","datechanger_globals","default","defaultchoice","hidewarn","install","Vue","use","TSComponents","mFormComponents","debug","Debugger","isVisible","elem","offsetWidth","offsetHeight","getClientRects","length","ItemEventBus","document","addEventListener","editSwEventTypes","editModeSet","e","preventDefault","$emit","detail","editMode","string_keys","conditions","value","textkey","strings","studyplan_text","studyline_editmode","toolbox_toggle","editmode_modules_hidden","studyline_add","add$core","edit$core","studyline_name","studyline_name_ph","studyline_shortname","studyline_shortname_ph","studyline_color","associations","associated_cohorts","associated_users","studyline_edit","studyplan_name","studyplan_name_ph","studyplan_shortname","studyplan_shortname_ph","studyplan_description","studyplan_description_ph","studyplan_idnumber","studyplan_idnumber_ph","studyplan_slots","studyplan_startdate","studyplan_enddate","studyplan_advanced","advanced_tools","confirm_cancel","confirm_ok","success$core","error$core","advanced_converted","advanced_skipped","advanced_failed","advanced_locked","advanced_multiple","advanced_error","advanced_tools_heading","advanced_warning_title","advanced_warning","advanced_pick_scale","advanced_course_manipulation_title","advanced_bulk_course_timing","advanced_bulk_course_timing_desc","advanced_force_scale_title","advanced_force_scale_desc","advanced_force_scale_button","advanced_confirm_header","advanced_force_scale_confirm","advanced_backup_restore","advanced_restore","advanced_backup","advanced_restore_pages","advanced_restore_lines","advanced_backup_plan","advanced_backup_page","advanced_export","advanced_export_csv_plan","advanced_export_csv_page","advanced_import_from_file","advanced_purge","advanced_purge_plan","advanced_purge_plan_expl","advanced_purge_page","advanced_purge_page_expl","advanced_cascade_cohortsync_title","advanced_cascade_cohortsync_desc","advanced_cascade_cohortsync","currentpage","studyplan_edit","studyplan_add","studyplanpage_add","studyplanpage_edit","info_periodsextended","warning","period_edit","edit","fullname","shortname","startdate","enddate","course_timing","title","desc","question","timing_ok","timing_off","course","period","yes","no","duration","years","year","weeks","week","days","day","rememberchoice","hidewarning","periodspan","periods","periodspan_desc","studyplan_associate","associate_cohorts","associate_users","add_association","delete_association","associations_empty","associations_search","cohorts","users","selected","name","context","item_text","select_conditions","item_configuration","ok","delete","item_delete_message","type_course","type_junction","type_start","type_finish","type_badge","type_invalid","item_course_text","select_grades","coursetiming_past","coursetiming_present","coursetiming_future","grade_include","grade_require","cancel","invalid","error","completion","completion_completed","completion_incomplete","aggregation_all","aggregation_any","aggregation_overall_all","aggregation_overall_any","completion_not_configured","configure_completion","competency","competency_not_configured","configure_competency","when","required","points","heading","details","badge","share_badge","dateissued","dateexpire","badgeinfo","component","props","type","Object","selectedpage","data","force_scales","selected_scale","result","text","created","mounted","updated","computed","scales","id","disabled","this","concat","advanced","methods","force_scales_start","self","$bvModal","msgBoxConfirm","okVariant","okTitle","cancelTitle","then","methodname","args","studyplan_id","scale_id","response","catch","notification","exception","export_page","format","undefined","includes","page_id","content","export_plan","bulk_course_timing","success","location","reload","msgBoxOk","msg","import_studylines","filename","import_pages","purge_studyplan","force","purge_studyplanpage","cascade_cohortsync","modal_close","template","String","Number","planSaved","updatedplan","info","mode","moved_from","context_id","moved_to","moved","pages","updatedpage","page","centered","show","config","userfields","key","cohortfields","association","loading","search","associated","showModal","loadAssociations","cohortOptionModel","c","path","join","userOptionModel","u","firstname","lastname","map","searchCohorts","searchtext","like","exclude_id","cohortAssociate","requests","searchselected","i","r","push","cohort_id","fail","done","cohortDisassociate","associatedselected","searchUsers","userAssociate","user_id","userDisassociate","editdata","Date","getFullYear","editStart","editFinish","refresh","add_day","date","sub_day","create","studyline","toolbox_shown","editmode","color","original","studyplan","description","slots","aggregation","aggregation_config","aggregation_info","useRequiredGrades","useItemCondition","cache","linelayers","selectedpageindex","emptyline","filterslots","courseslots","$on","onMoodleEditModeChanged","initeditmode","$root","columns","columns_stylerule","s","trashbin_accepts","item","countLineLayers","line","timestamp","maxLayer","ix","courses","layer","filters","slotsempty","Array","isArray","count","movedStudyplan","plan","from","to","addStudyLine","newlineinfo","studylines","editLineStart","assign","editLineFinish","editedline","originalline","deleteLine","param","modalresponse","index","indexOf","splice","reorderLines","event","lines","apply","sequence","idx","deletePlan","deleteStudyItem","source","showslot","layeridx","forGradable","list","span","periodEdited","pi","prev","$refs","next","toolbox_switched","pagecreated","selectedpageChanged","newTabIndex","prevTabIndex","layers","layerHeights","onLineHeightChange","lineid","layerid","newheight","main","items","querySelectorAll","heightSum","forEach","el","height","getBoundingClientRect","heightStyle","style","deletable","editable","onEdit","onDelete","slotindex","resizeListener","ResizeObserver","size","observe","unmounted","disconnect","slotkey","itemidx","itm","listtype","dragacceptlist","courseHoverDummy","hover","current","now","pstart","pend","spanCss","datechanger","coursespan","onDrop","slot","relocateStudyItem","timingChecker","validate_course_period","$nextTick","onCut","iteminfo","onDragEnter","onDragLeave","maxSpan","makeType","checkType","maxspan","hidden","Boolean","endperiod","endperiodnr","Math","min","perioddesc","course_period_matches","first","getTime","last","warn","floor","random","toString","globals","canupdatecourse","defaultvalue","change_course_period","has_filter","period_id","course_id","timing","checkFilterSlotBusy","nextFreeFilterLayer","usedLayers","nextlyr","checkCourseSlotBusy","nextFreeCourseLayer","shift_collisions","busyFilter","line_id","busyCourse","change_span","format_duration","dsi","toLocaleLowerCase","dummy","dragLine","dragEventListener","deleteMode","condition_options","showContext","dragStart","start","getElementById","dragelement","position","left","x","top","y","SimpleLine","gravity","end","onMouseMove","dragEnd","remove","removeProperty","removeEventListener","clientX","clientY","from_id","to_id","redrawLines","conn","connections","in","redrawLine","removeLine","out","highlight","setConfig","normalize","updateItem","continuation_id","doShowContext","hasContext","onCreatedConnection","onRemovedConnection","c_in","onRePositioned","onDisPositioned","re_id","onItemDeleted","item_id","onRedrawLines","deleteItem","msgparams","displayname","err","console","hasConnectionsOut","hasConnectionsIn","setTimeout","beforeDestroy","$off","beforeUpdate","useItemConditions","configurationState","hasGrades","hasCompletions","hasCompetencies","configurationIcon","noenddate","grades","g","cgroup","competencies","includeChanged","newValue","requiredChanged","updateConditions","selectedgrades","guestmode","stringkeys","completion_icon","completion_tag","pathtags","p","url","contextid","txt","showSpinner","canLoadMore","hasDetails","haschildren","hascourses","children","onShowDetails"],"mappings":"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"}