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
2024-06-05 22:22:19 +02:00

1 line
No EOL
264 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 max-depth: [\"error\", 6] */\n\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 {loadStringKeys, loadStrings, strformat} from './util/string-helper';\nimport {formatDate, addDays, datespaninfo} from './util/date-helper';\nimport {objCopy, transportItem} from './studyplan-processor';\nimport Debugger from './util/debugger';\nimport Config from 'core/config';\nimport {download, upload} from './downloader';\nimport {processStudyplan, processStudyplanPage} from './studyplan-processor';\nimport {premiumenabled} from \"./util/premium\";\nimport FitTextVue from './util/fittext-vue';\nimport {settings} from \"./util/settings\";\nimport TSComponents from './treestudyplan-components';\nimport mFormComponents from \"./util/mform-helper\";\nimport pSideBarComponents from \"./util/psidebar-vue\";\n\nimport {Drag, Drop, DropList} from './vue-easy-dnd/vue-easy-dnd.esm';\n\nconst STUDYPLAN_EDITOR_FIELDS =\n['name', 'shortname', 'description', 'idnumber', 'context_id', 'aggregation', 'aggregation_config'];\nconst PERIOD_EDITOR_FIELDS =\n['fullname', 'shortname', 'startdate', 'enddate'];\n\nconst LINE_GRAVITY = 1.3;\n\nexport default {\n install(Vue/* ,options */) {\n Vue.component('drag', Drag);\n Vue.component('drop', Drop);\n Vue.component('drop-list', DropList);\n Vue.use(TSComponents);\n Vue.use(mFormComponents);\n Vue.use(pSideBarComponents);\n Vue.use(FitTextVue);\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 /*\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\n let stringKeys = loadStringKeys({\n conditions: [\n {value: 'ALL', textkey: 'condition_all'},\n {value: 'ANY', textkey: 'condition_any'},\n ],\n });\n\n let strings = loadStrings({\n studyplanText: {\n 'studyline_editmode': 'studyline_editmode',\n 'toolbox_toggle': 'toolbox_toggle',\n 'editmode_modules_hidden': 'editmode_modules_hidden',\n 'studyline_add': 'studyline_add',\n add: 'add@core',\n edit: 'edit@core',\n 'delete': \"delete@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_enrollable': 'studyline_enrollable',\n 'studyline_enrolroles': 'studyline_enrolroles',\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 'line_enrollable_0': 'line_enrollable:0',\n 'line_enrollable_1': 'line_enrollable:1',\n 'line_enrollable_2': 'line_enrollable:2',\n 'line_enrollable_3': 'line_enrollable:3',\n drophere: 'drophere',\n studylineConfirmRemove: 'studyline_confirm_remove',\n studyplanConfirmRemove: 'studyplan_confirm_remove',\n\n },\n studyplanAdvanced: {\n 'advanced_tools': 'advanced_tools',\n 'confirm_cancel': 'confirm_cancel',\n 'confirm_ok': 'confirm_ok',\n success: 'success@core',\n error: 'failed@completion',\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 studyplanEdit: {\n studyplanEdit: 'studyplanEdit',\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 periodEdit: {\n edit: 'period_edit',\n fullname: 'studyplan_name',\n shortname: 'studyplan_shortname',\n startdate: 'studyplan_startdate',\n enddate: 'studyplan_enddate',\n },\n courseTiming: {\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 studyplanAssociate: {\n 'associations': 'associations',\n 'associated_cohorts': 'associated_cohorts',\n 'associated_users': 'associated_users',\n 'associated_coaches': 'associated_coaches',\n 'associate_cohorts': 'associate_cohorts',\n 'associate_users': 'associate_users',\n 'associate_coached': 'associate_coaches',\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 coaches: 'coaches',\n selected: 'selected',\n name: 'name',\n context: 'context',\n search: 'search',\n },\n itemText: {\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 itemCourseText: {\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 noenddate: \"noenddate\",\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 toolbox: {\n toolbox: 'toolbox',\n toolbarRight: 'toolbar-right',\n courses: 'courses',\n flow: 'flow',\n toolJunction: 'tool-junction',\n toolFinish: 'tool-finish',\n toolStart: 'tool-start',\n badges: 'badges',\n relatedbages: 'relatedbages@badges',\n filter: 'filter@core',\n sitebadges: 'sitebadges@badges',\n }\n });\n\n /*\n * T-STUDYPLAN-ADVANCED\n */\n Vue.component('t-studyplan-advanced', {\n props: {\n value: {\n type: Object,\n default() {\n return null;\n },\n },\n selectedpage: {\n type: Object,\n default() {\n return null;\n },\n }\n\n },\n data() {\n return {\n forceScales: {\n selectedScale: null,\n result: [],\n },\n text: strings.studyplanAdvanced,\n };\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 forceScalesStart() {\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.forceScales.selectedScale,\n }\n }])[0].then((response) => {\n self.forceScales.result = response;\n return;\n }).catch(notification.exception);\n }\n return;\n }).catch(notification.exception);\n },\n exportPage(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((response) => {\n download(self.value.shortname + \".page.\" + format, response.content, response.format);\n return;\n }).catch(notification.exception);\n },\n exportPlan() {\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((response) => {\n download(self.value.shortname + \".plan.json\", response.content, response.format);\n return;\n }).catch(notification.exception);\n },\n bulkCourseTiming() {\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((response) => {\n if (response.success) {\n // Reloading the webpage saves trouble reloading the specific page updated.\n location.reload();\n } else {\n self.$bvModal.msgBoxOk(response.msg, {title: \"Could not set bulk course timing\"});\n debug.error(\"Could not set bulk course timing: \", response.msg);\n }\n return;\n }).catch(notification.exception);\n },\n importStudylines() {\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((response) => {\n if (response.success) {\n location.reload();\n } else {\n self.$bvModal.msgBoxOk(response.msg, {title: \"Import failed\"});\n debug.error(\"Import failed: \", response.msg);\n }\n return;\n }).catch(notification.exception);\n }, \"application/json\");\n },\n importPages() {\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((response) => {\n if (response.success) {\n location.reload();\n } else {\n self.$bvModal.msgBoxOk(response.msg, {title: \"Import failed\"});\n debug.error(\"Import failed: \", response.msg);\n }\n return;\n }).catch(notification.exception);\n }, \"application/json\");\n },\n purgeStudyplan() {\n const self = this;\n call([{\n methodname: 'local_treestudyplan_delete_studyplan',\n args: {\n id: this.value.id,\n force: true,\n },\n }])[0].then((response) => {\n if (response.success) {\n location.reload();\n } else {\n self.$bvModal.msgBoxOk(response.msg, {title: \"Could not delete plan \"});\n debug.error(\"Could not delete plan: \", response.msg);\n }\n return;\n }).catch(notification.exception);\n },\n purgeStudyplanpage() {\n const self = this;\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((response) => {\n if (response.success) {\n location.reload();\n } else {\n self.$bvModal.msgBoxOk(response.msg, {title: \"Could not delete page\"});\n debug.error(\"Could not delete page: \", response.msg);\n }\n return;\n }).catch(notification.exception);\n }\n },\n cascadeCohortsync() {\n const self = this;\n call([{\n methodname: 'local_treestudyplan_cascade_cohortsync',\n args: {\n 'studyplan_id': this.value.id,\n },\n }])[0].then((response) => {\n self.$bvModal.msgBoxOk(response.success ? self.text.success : self.text.error,\n {title: self.text.advanced_cascade_cohortsync});\n return;\n }).catch(notification.exception);\n },\n modalClose() {\n this.forceScales.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=\"modalClose\"\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.prevent=\"cascadeCohortsync\"\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.prevent=\"bulkCourseTiming\"\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=\"forceScales.selectedScale\"\n :options=\"scales\" text-field=\"name\" value-field=\"id\"\n ></b-form-select>\n <b-button\n variant=\"danger\"\n :disabled=\"forceScales.selectedScale == null\"\n @click.prevent=\"forceScalesStart\"\n >{{ text.advanced_force_scale_button}}</b-button>\n </p>\n <p class=\"mt-2\">\n <ul class='t-advanced-scrollable' v-if=\"forceScales.result.length > 0\">\n <li v-for=\"c in forceScales.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.prevent=\"exportPage('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.prevent=\"exportPlan('json')\"\n >{{ text.advanced_backup_plan }}</b-button></p>\n <h3>{{ text.advanced_restore }}</h3>\n <p><b-button\n variant=\"danger\"\n @click.prevent=\"importStudylines\"\n >{{ text.advanced_restore_lines}}</b-button></p>\n <p><b-button\n variant=\"danger\"\n @click.prevent=\"importPages\"\n >{{ text.advanced_restore_pages }}</b-button></p>\n <h3>{{ text.advanced_export }}</h3>\n <p><b-button\n variant=\"primary\"\n @click.prevent=\"exportPage('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.prevent=\"purgeStudyplanpage\"\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.prevent=\"purgeStudyplan\"\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() {\n return null;\n },\n },\n 'mode': {\n type: String,\n default() {\n return \"edit\";\n },\n },\n 'type': {\n type: String,\n default() {\n return \"link\";\n },\n },\n 'variant': {\n type: String,\n default() {\n return \"\";\n },\n },\n 'contextid': {\n type: Number,\n 'default': 1\n },\n },\n data() {\n return {\n text: strings.studyplanEdit,\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 } else {\n // Determine if the plan moved context...\n const movedFrom = self.value.context_id;\n const movedTo = updatedplan.context_id;\n const moved = (movedFrom != movedTo);\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((response) => {\n self.value = processStudyplan(response, true);\n debug.info('studyplan processed');\n self.$emit('input', self.value);\n return;\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, movedFrom, movedTo);\n }\n }\n }\n },\n },\n template:\n `\n <span class='s-studyplan-edit'>\n <mform \n name=\"studyplanEditform\" \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.studyplanEdit\"\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() {\n return null;\n },\n },\n 'mode': {\n type: String,\n default() {\n return \"edit\";\n },\n },\n 'type': {\n type: String,\n default() {\n return \"link\";\n },\n },\n 'variant': {\n type: String,\n default() {\n return \"\";\n },\n },\n 'studyplan': {\n type: Object,\n },\n },\n data() {\n return {\n text: strings.studyplanEdit,\n };\n },\n computed: {\n },\n methods: {\n planSaved(updatedpage) {\n const self = this;\n\n if (self.mode == 'create') {\n // Inform parent of the details of the newly created plan\n self.$emit(\"created\", updatedpage);\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 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 coaches: []\n },\n loading: {\n cohorts: false,\n users: false,\n coaches: false,\n },\n search: {users: [], cohorts: [], coaches: []},\n selected: {\n search: {users: [], cohorts: [], coaches: []},\n associated: {users: [], cohorts: [], coaches: []}\n },\n text: strings.studyplanAssociate,\n };\n },\n methods: {\n premiumenabled,\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: {\n 'studyplan_id': self.value.id,\n }\n }])[0].then((response) => {\n self.association.users = response.map(self.userOptionModel);\n self.loading.users = false;\n return;\n }).catch(notification.exception);\n\n call([{\n methodname: 'local_treestudyplan_associated_cohorts',\n args: {\n 'studyplan_id': self.value.id,\n }\n }])[0].then((response) => {\n self.association.cohorts = response.map(self.cohortOptionModel);\n self.loading.cohorts = false;\n return;\n }).catch(notification.exception);\n\n if (premiumenabled()) {\n self.loading.coaches = true;\n call([{\n methodname: 'local_treestudyplan_associated_coaches',\n args: {\n 'studyplan_id': self.value.id,\n }\n }])[0].then((response) => {\n self.association.coaches = response.map(self.userOptionModel);\n self.loading.coaches = false;\n return;\n }).catch(notification.exception);\n }\n },\n searchCohorts(searchtext) {\n const self = this;\n\n if (searchtext.length > 0) {\n call([{\n methodname: 'local_treestudyplan_list_cohort',\n args: {\n like: searchtext,\n 'studyplan_id': self.value.id\n }\n }])[0].then((response) => {\n self.search.cohorts = response.map(self.cohortOptionModel);\n return;\n }).catch(notification.exception);\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 call([{\n methodname: 'local_treestudyplan_connect_cohort',\n args: {\n 'studyplan_id': self.value.id,\n 'cohort_id': r,\n },\n }]).then((response) => {\n if (response.success) {\n transportItem(associated, search, r);\n }\n return;\n }).catch(notification.exception);\n }\n call(requests);\n },\n cohortDisassociate() {\n const self = this;\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 call([{\n methodname: 'local_treestudyplan_disconnect_cohort',\n args: {\n 'studyplan_id': self.value.id,\n 'cohort_id': r,\n }\n }])[0].then((response) => {\n if (response.success) {\n transportItem(associated, search, r);\n }\n return;\n }).catch(notification.exception);\n }\n },\n searchUsers(searchtext) {\n const self = this;\n if (searchtext.length > 0) {\n call([{\n methodname: 'local_treestudyplan_find_user',\n args: {\n like: searchtext,\n 'studyplan_id': self.value.id\n }\n }])[0].then((response) => {\n self.search.users = response.map(self.userOptionModel);\n return;\n }).catch(notification.exception);\n } else {\n self.search.users = [];\n }\n },\n userAssociate() {\n const self = this;\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 call([{\n methodname: 'local_treestudyplan_connect_user',\n args: {\n 'studyplan_id': self.value.id,\n 'user_id': r,\n },\n }]).then((response) => {\n if (response.success) {\n transportItem(associated, search, r);\n }\n return;\n }).catch(notification.exception);\n }\n },\n userDisassociate() {\n const self = this;\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 call([{\n methodname: 'local_treestudyplan_disconnect_user',\n args: {\n 'studyplan_id': self.value.id,\n 'user_id': r,\n }\n }])[0].then((response) => {\n if (response.success) {\n transportItem(associated, search, r);\n }\n return;\n }).catch(notification.exception);\n }\n },\n searchCoaches(searchtext) {\n if (premiumenabled()) {\n const self = this;\n if (searchtext.length > 0) {\n call([{\n methodname: 'local_treestudyplan_find_coach',\n args: {\n like: searchtext,\n 'studyplan_id': self.value.id,\n }\n }])[0].then((response) => {\n self.search.coaches = response.map(self.userOptionModel);\n return;\n }).catch(notification.exception);\n } else {\n self.search.coaches = [];\n }\n }\n },\n coachAssociate() {\n if (premiumenabled()) {\n const self = this;\n const associated = self.association.coaches;\n const search = self.search.coaches;\n const searchselected = self.selected.search.coaches;\n for (const i in searchselected) {\n const r = searchselected[i];\n\n call([{\n methodname: 'local_treestudyplan_connect_coach',\n args: {\n 'studyplan_id': self.value.id,\n 'user_id': r,\n },\n }]).then((response) => {\n if (response.success) {\n transportItem(associated, search, r);\n }\n return;\n }).catch(notification.exception);\n }\n }\n },\n coachDisassociate() {\n if (premiumenabled()) {\n const self = this;\n const associated = self.association.coaches;\n const associatedselected = self.selected.associated.coaches;\n const search = self.search.coaches;\n for (const i in associatedselected) {\n const r = associatedselected[i];\n\n call([{\n methodname: 'local_treestudyplan_disconnect_coach',\n args: {\n 'studyplan_id': self.value.id,\n 'user_id': r,\n }\n }])[0].then((response) => {\n if (response.success) {\n transportItem(associated, search, r);\n }\n return;\n }).catch(notification.exception);\n }\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.prevent=\"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.prevent=\"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=\"text.search + ' ' + text.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.prevent=\"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.prevent=\"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-tab :title=\"text.coaches\" v-if=\"premiumenabled()\">\n <b-container>\n <b-row class='mb-2 mt-2'>\n <b-col>{{text.associated_coaches}}</b-col>\n <b-col>{{text.associate_coaches}}</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=\"searchCoaches($event)\"\n :placeholder=\"text.search + ' ' + text.coaches\"></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.coaches\"\n :options=\"association.coaches\"\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.coaches\"\n :options=\"search.coaches\"\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.prevent=\"coachDisassociate()\"\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.prevent=\"coachAssociate()\"\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() {\n return null;\n },\n },\n 'type': {\n type: String,\n default() {\n return \"link\";\n },\n },\n 'variant': {\n type: String,\n default() {\n return \"\";\n },\n },\n 'minstart': {\n type: String,\n default() {\n return null;\n },\n },\n 'maxend': {\n type: String,\n default() {\n return null;\n },\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.periodEdit,\n };\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_treestudyplanEdit_period',\n args: args\n }])[0].then((response) => {\n objCopy(self.value, response, PERIOD_EDITOR_FIELDS);\n self.$emit('input', self.value);\n self.$emit('edited', self.value);\n return;\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((response) => {\n objCopy(self.value, response, PERIOD_EDITOR_FIELDS);\n self.$emit('input', self.value);\n return;\n }).catch(notification.exception);\n },\n addDay(date, days) {\n if (days === undefined) {\n days = 1;\n }\n return addDays(date, days);\n },\n subDay(date, days) {\n if (days === undefined) {\n days = 1;\n }\n return addDays(date, 0 - days);\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 v-if=\"!value.timeless\">\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=\"subDay(value.enddate)\"\n ></b-form-datepicker>\n </b-col>\n </b-row>\n <b-row v-if=\"!value.timeless\">\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=\"addDay(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: { \n 'value': {\n type: Object,\n }, \n 'coaching': {\n type: Boolean,\n 'default': false,\n },\n },\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 enrol: {\n enrollable: 0,\n enrolroles: [],\n }\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 enrol: {\n enrollable: 0,\n enrolroles: [],\n }\n },\n original: {},\n availableroles: [],\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.studyplanText,\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 availableroles: [],\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('coursechange', () => {\n self.$emit('pagechanged', this.selectedpage);\n });\n },\n mounted() {\n const self = this;\n if (this.value.pages[0].studylines.length == 0 && !this.coaching) {\n // Start in editmode if studylines on first page are empty\n this.edit.studyline.editmode = true;\n }\n\n if (!self.coaching) {\n // Retrieve available roles (only needed as manager)\n call([{\n methodname: 'local_treestudyplan_list_roles',\n args: {\n 'studyplan_id': this.value.id,\n }\n }])[0].then((response) => {\n self.availableroles = response;\n return;\n }).catch(notification.exception);\n }\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 hivizdrop() {\n return settings(\"hivizdropslots\");\n },\n },\n methods: {\n premiumenabled,\n columns(page) {\n return 1 + (page.periods * 2);\n },\n columnsStylerule(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 trashbinAccepts(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 } else {\n let maxLayer = -1;\n for (let i = 0; i <= page.periods; i++) {\n if (line.slots[i]) {\n // Determine the amount of used layers in a studyline slot\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 'enrollable': newlineinfo.enrol.enrollable,\n 'enrolroles': newlineinfo.enrol.enrolroles\n }\n }])[0].then((response) => {\n page.studylines.push(response);\n newlineinfo.name = '';\n newlineinfo.shortname = '';\n newlineinfo.color = \"#dddddd\";\n newlineinfo.enrol.enrollable = 0;\n newlineinfo.enrol.enrolroles = [];\n return;\n }).catch(notification.exception);\n },\n editLineStart(line) {\n const page = this.value.pages[this.selectedpageindex];\n debug.info(\"Starting line edit\", line);\n Object.assign(this.edit.studyline.data, line);\n this.edit.studyline.original = line;\n this.$bvModal.show('modal-edit-studyline-' + page.id);\n },\n editLineFinish() {\n let editedline = this.edit.studyline.data;\n let originalline = this.edit.studyline.original;\n call([{\n methodname: 'local_treestudyplanEdit_studyline',\n args: {'id': editedline.id,\n 'name': editedline.name,\n 'shortname': editedline.shortname,\n 'color': editedline.color,\n 'enrollable': editedline.enrol.enrollable,\n 'enrolroles': editedline.enrol.enrolroles\n }\n }])[0].then((response) => {\n originalline.name = response.name;\n originalline.shortname = response.shortname;\n originalline.color = response.color;\n originalline.enrol.enrollable = response.enrol.enrollable;\n originalline.enrol.enrolroles = response.enrol.enrolroles;\n return;\n }).catch(notification.exception);\n },\n deleteLine(page, line) {\n const self = this;\n\n self.$bvModal.msgBoxConfirm(this.text.studylineConfirmRemove.replace('{$a}', line.name), {\n okTitle: this.text.delete,\n okVariant: 'danger',\n }).then((modalresponse) => {\n if (modalresponse) {\n call([{\n methodname: 'local_treestudyplan_delete_studyline',\n args: {'id': line.id}\n }])[0].then((response) => {\n if (response.success == true) {\n let index = page.studylines.indexOf(line);\n page.studylines.splice(index, 1);\n }\n return;\n }).catch(notification.exception);\n }\n return;\n }).catch(notification.exception);\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 sequence.push({'id': lines[idx].id, 'sequence': idx});\n }\n call([{\n methodname: 'local_treestudyplan_reorder_studylines',\n args: {'sequence': sequence}\n }])[0].then(() => {\n return;\n }).catch(notification.exception);\n },\n deletePlan(studyplan) {\n const self = this;\n self.$bvModal.msgBoxConfirm(this.text.studyplabConfirmRemove.replace('{$a}', studyplan.name), {\n okTitle: this.text.delete,\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((response) => {\n if (response.success == true) {\n self.$root.$emit(\"studyplanRemoved\", studyplan);\n }\n return;\n }).catch(notification.exception);\n }\n return;\n }).catch(notification.exception);\n },\n deleteStudyItem(event) {\n let item = event.data;\n call([{\n methodname: 'local_treestudyplan_delete_studyitem',\n args: {'id': item.id}\n }])[0].then((response) => {\n if (response.success == true) {\n event.source.$emit('cut', event);\n }\n return;\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) { \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)];\n if (prev && prev[0]) {\n prev[0].refresh();\n }\n const next = this.$refs[\"periodeditor-\" + (pi.period + 1)];\n if (next && next[0]) {\n next[0].refresh();\n }\n },\n addDay(date, days) {\n if (days === undefined) {\n days = 1;\n }\n return addDays(date, days);\n },\n subDay(date, days) {\n if (days === undefined) {\n days = 1;\n }\n return addDays(date, 0 - days);\n },\n toolboxSwitched(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 sumLineLayers(idx, page) {\n if (idx < 0 || page.studylines.count == 0) {\n return 0;\n } else {\n let sum = 0;\n for (let i = 0; i < idx; i++) {\n sum += this.countLineLayers(page.studylines[i], page) + 1;\n }\n return sum;\n }\n },\n span(line, slot, layer) {\n let span = 1;\n for (const course of line.slots[slot].courses) {\n if (course.slot == slot && course.layer == layer) {\n span = course.span;\n }\n }\n return span;\n },\n onDrop(event, line, slot) {\n debug.info(\"dropping\", event, line, slot);\n const self = this;\n if (event.type.component) { // Double check in case filter fails\n debug.info(\"Adding new component\");\n if (event.type.type == \"gradable\") {\n // Determine first available layer;\n const lineslot = line.slots[slot].courses;\n let nextlayer = 0;\n for (const itm of lineslot) {\n if (itm.layer >= nextlayer) {\n nextlayer = itm.layer + 1;\n }\n }\n\n call([{\n methodname: 'local_treestudyplan_add_studyitem',\n args: {\n \"line_id\": line.id,\n \"slot\": slot,\n \"layer\": nextlayer,\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 lineslot.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.validateCoursePeriod();\n }\n });\n ItemEventBus.$emit('coursechange');\n return;\n }).catch(notification.exception);\n } else if (event.type.type == \"filter\") {\n debug.info(\"Adding new filter compenent\");\n // Determine first available layer;\n const lineslot = line.slots[slot].filters;\n let nextlayer = 0;\n for (const itm of lineslot) {\n if (itm.layer >= nextlayer) {\n nextlayer = itm.layer + 1;\n }\n }\n call([{\n methodname: 'local_treestudyplan_add_studyitem',\n args: {\n \"line_id\": line.id,\n \"slot\": slot,\n \"type\": event.data.type,\n \"layer\": nextlayer,\n \"details\": {\n \"badge_id\": event.data.badge ? event.data.badge.id : undefined,\n }\n }\n }])[0].then((response) => {\n let item = response;\n lineslot.push(item);\n self.$emit(\"input\", self.value);\n return;\n }).catch(notification.exception);\n }\n }\n },\n checkTypeCourse(type) {\n if (type.type == \"gradable\") {\n if (settings(\"hivizdropslots\") && !type.item) {\n return true;\n } else {\n return false;\n }\n } else {\n return false;\n }\n },\n checkTypeFilter(type) {\n if (type.type == \"filter\") {\n if (settings(\"hivizdropslots\") && !type.item) {\n return true;\n } else {\n return false;\n }\n } else {\n return false;\n }\n }\n },\n template:\n `\n <div>\n <t-toolbox v-model=\"edit.toolbox_shown\"\n :activepage=\"selectedpage\"\n :coaching=\"coaching\"\n :studyplanid=\"value.id\"></t-toolbox>\n <div class='controlbox t-studyplan-controlbox'>\n <div class=\"controlbox-group\">\n <b-form-checkbox v-if=\"!coaching\"\n v-model=\"edit.studyline.editmode\" class=\"sw-studyplan-toolbar\" switch\n @change=\"toolboxSwitched(edit.toolbox_shown && !edit.studyline.editmode); \"\n >{{ text.studyline_editmode }}</b-form-checkbox>\n <b-form-checkbox \n v-if=\"!edit.studyline.editmode\" v-model=\"edit.toolbox_shown\" class=\"sw-studyplan-toolbar\" switch\n @change=\"toolboxSwitched\"\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=\"trashbinAccepts\"\n ><i class='fa fa-trash'></i>\n </drop>\n </div>\n <div class=\"controlbox-group\" v-if=\"!coaching\">\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}}</t-studyplan-edit>\n </span>\n <span class='control deletable'>\n <a v-if='value.pages.length == 0' href='#' @click.prevent='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 v-if=\"!coaching\"\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=\"!coaching && (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 @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-if=\"!coaching\"\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=\"columnsStylerule(page)\">\n <!-- add period information --> \n <template v-for=\"(n,index) in (page.periods+1)\">\n <s-studyline-header-period\n mode=\"edit\"\n :x-index=\"index\"\n :style=\"'grid-area: 1 / '+ ((2*index)) +';'\" \n :identifier='Number(page.id)'\n v-if=\"index > 0\"\n v-model=\"page.perioddesc[index-1]\"\n ><t-period-edit\n v-if=\"!coaching\"\n :ref=\"'periodeditor-'+index\"\n @edited=\"periodEdited\"\n v-model=\"page.perioddesc[index-1]\"\n :minstart=\"(index > 1) ? addDay(page.perioddesc[index-2].startdate,2) : null\"\n :maxend=\"(index < page.periods) ? subDay(page.perioddesc[index].enddate,2) : null\"\n ></t-period-edit\n ></s-studyline-header-period>\n <div class=\"s-studyline-header-filter\" \n :x-index=\"index\"\n :style=\"'grid-area: 1 / '+ ((2*index)+1) +';'\" \n ></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 ><t-studyline-slot\n v-if=\"index > 0 && showslot(page,line, index, layeridx, 'gradable')\"\n :style=\"'grid-area: '+ (1+sumLineLayers(lineindex,page)+layernr) \n + ' / ' + (2 * index) \n + ' / ' + (1 + sumLineLayers(lineindex,page)+layernr) \n + ' / ' + ( (2 * index) + (2*span(line,index,layeridx) - 1)) + ';'\" \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 :style=\"'grid-area: '+ (1+sumLineLayers(lineindex,page)+layernr) + ' / ' + (2*index+1) +';'\" \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 ><template v-if=\"hivizdrop\"\n ><template v-for=\"(line,lineindex) in page.studylines\"\n ><template v-for=\"(n,index) in (page.periods+1)\"\n ><drop v-if=\"index > 0\"\n :style=\"'grid-area: '+ (2 + sumLineLayers(lineindex,page)) \n + ' / ' + (2 * index) \n + ' / ' + (1 + sumLineLayers(lineindex + 1,page)) \n + ' / ' + (2 * index) + '; overflow: hidden;'\" \n :class=\"'t-slot-drop t-slot-linedrop course hiviz'\"\n :accepts-type=\"checkTypeCourse\"\n @drop=\"onDrop($event,line,index)\"\n mode=\"cut\"\n ><span>{{text.drophere}}</span></drop\n ><drop \n :style=\"'grid-area: '+ (2 + sumLineLayers(lineindex,page)) \n + ' / ' + ((2 * index) + 1)\n + ' / ' + (1 + sumLineLayers(lineindex + 1,page)) \n + ' / ' + ( (2 * index) + 1 ) + '; overflow: hidden;'\" \n :class=\"'t-slot-drop t-slot-linedrop filter hiviz'\"\n :accepts-type=\"checkTypeFilter\"\n @drop=\"onDrop($event,line,index)\"\n mode=\"cut\"\n ><span>{{text.drophere}}</span></drop\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.prevent=\"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\"\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 <template v-if=\"premiumenabled()\">\n <b-row>\n <b-col cols=\"3\">{{ text.studyline_enrollable}}</b-col>\n <b-col>\n <b-form-select v-model=\"create.studyline.enrol.enrollable\">\n <b-form-select-option \n v-for=\"(nr,n) in 4\"\n :value=\"n\"\n :key=\"n\"\n >{{text['line_enrollable_'+n]}}</b-form-select-option>\n </b-form-select>\n </b-col>\n </b-row>\n <b-row v-if='[2,3].includes(create.studyline.enrol.enrollable)'>\n <b-col cols=\"3\">{{ text.studyline_enrolroles}}</b-col>\n <b-col>\n <b-form-select \n v-model=\"create.studyline.enrol.enrolroles\"\n :options=\"availableroles\"\n multiple\n value-field=\"id\"\n text-field=\"name\"\n :select-size=\"6\"\n ></b-form-select>\n </b-col>\n </b-row>\n </template> \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 <template v-if=\"premiumenabled()\">\n <b-row>\n <b-col cols=\"3\">{{ text.studyline_enrollable}}</b-col>\n <b-col>\n <b-form-select v-model=\"edit.studyline.data.enrol.enrollable\">\n <b-form-select-option \n v-for=\"(nr,n) in 4\"\n :value=\"n\"\n >{{text['line_enrollable_'+n]}}</b-form-select-option>\n </b-form-select>\n </b-col>\n </b-row>\n <b-row v-if='[2,3].includes(edit.studyline.data.enrol.enrollable)'>\n <b-col cols=\"3\">{{ text.studyline_enrolroles}}</b-col>\n <b-col>\n <b-form-select \n v-model=\"edit.studyline.data.enrol.enrolroles\"\n :options=\"availableroles\"\n multiple\n value-field=\"id\"\n text-field=\"name\"\n :select-size=\"6\"\n ></b-form-select>\n </b-col>\n </b-row>\n </template>\n </b-container>\n </b-modal>\n </b-tab>\n </b-tabs>\n </b-card>\n </div>\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() {\n return {};\n },\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) { // Is called with parameters (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 // Func 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() {\n return {};\n },\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.prevent='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.prevent='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() {\n return null;\n },\n },\n layer: {\n type: Number,\n },\n value: {\n type: Array, // Dict with layer as index\n default() {\n return [];\n },\n },\n plan: {\n type: Object, // Studyplan data\n default() {\n return null;\n },\n },\n page: {\n type: Object, // Studyplan data\n default() {\n return null;\n },\n },\n period: {\n type: Object, // Studyplan data\n default() {\n return null;\n },\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 courseHoverDummy() {\n return {course: this.hover.component};\n },\n spanCss() {\n if (this.item && this.item.span > 1) {\n // Calculate span like this:\n // const span = (2 * this.item.span) - 1;\n return `width: 100%; `;\n } else {\n return \"\";\n }\n }\n },\n data() {\n return {\n text: strings.courseTiming,\n plantext: strings.studyplanText,\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 hivizdrop() {\n return settings(\"hivizdropslots\");\n },\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).then(() => {\n if (this.$refs.timingChecker) {\n this.$refs.timingChecker.validateCoursePeriod();\n }\n return;\n }).catch(notification.exception);\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).then(()=>{\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.validateCoursePeriod();\n }\n });\n ItemEventBus.$emit('coursechange');\n return;\n }).catch(notification.exception);\n return;\n }).catch(notification.exception);\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).then(() => {\n item.layer = this.layer;\n self.value.push(item);\n self.$emit(\"input\", self.value);\n return;\n }).catch(notification.exception);\n return;\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 (settings(\"hivizdropslots\") && !type.item) {\n return false;\n } else {\n if (type == 'filter') {\n return true;\n } else if (type.span <= this.maxSpan()) {\n return true;\n } else {\n return false;\n }\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 :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') + (hivizdrop()?' hiviz':'')\"\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 ><span v-else-if=\"hivizdrop()\">{{plantext.drophere}}</span></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 coursePeriodMatches() {\n const self = this;\n if (self.page.timeless) {\n // Always return true in timeless mode.\n return true;\n }\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 } 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.courseTiming,\n datechanger: {\n coursespan: null,\n periodspan: null,\n globals: {\n 'default': false,\n defaultchoice: false,\n hidewarn: false,\n },\n }\n };\n },\n methods: {\n validateCoursePeriod() {\n const self = this;\n\n if (!self.page.timeless) {\n debug.info(\"Validating course and period\");\n if (!(self.coursePeriodMatches)) {\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 // G for it without asking\n self.changeCoursePeriod();\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 } else {\n debug.info(\"Course timing matches period\", self.datechanger);\n }\n } else {\n debug.info(\"Skipping course timing check because of timeless mode\", self.datechanger);\n }\n },\n changeCoursePeriod() {\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: {\n 'period_id': self.period.id,\n 'course_id': this.value.course.id,\n span: this.value.span,\n }\n }])[0].catch(notification.exception).then((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 return;\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 shiftCollisions(span) {\n // Check all periods for collision\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);\n }\n },\n changeSpan(span) {\n const self = this;\n this.shiftCollisions(span);\n return call([{\n methodname: 'local_treestudyplan_set_studyitem_span',\n args: {\n id: self.value.id,\n span: span\n }\n }])[0].catch(notification.exception).then((response) => {\n self.value.span = response.span;\n self.$emit('input', self.value);\n self.$nextTick(() => {\n self.validateCoursePeriod();\n });\n return;\n });\n },\n formatDuration(dsi) {\n let s = \"\";\n if (dsi.years == 1) {\n s += `1 ${this.text.year}, `;\n } else if (dsi.years > 1) {\n s += `${dsi.years} ${this.text.years}, `;\n }\n if (dsi.weeks == 1) {\n s += `1 ${this.text.week}, `;\n } else if (dsi.weeks > 1) {\n s += `${dsi.weeks} ${this.text.weeks}, `;\n }\n if (dsi.days == 1) {\n s += `1 ${this.text.day}, `;\n } else if (dsi.days > 1) {\n s += `${dsi.days} ${this.text.days}, `;\n }\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=\"coursePeriodMatches\">\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.prevent=\"validateCoursePeriod()\" 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=\"changeSpan\" 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=\"changeCoursePeriod\"\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 {{ formatDuration(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 {{ formatDuration(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 {{ formatDuration(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 {{ formatDuration(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() {\n return null;\n },\n },\n dummy: {\n type: Boolean,\n default() {\n return false;\n },\n },\n plan: {\n type: Object, // Studyplan page\n default() {\n return null;\n },\n },\n line: {\n type: Object, // Studyplan page\n default() {\n return null;\n },\n },\n page: {\n type: Object, // Studyplan page\n default() {\n return null;\n },\n },\n period: {\n type: Object, // Studyplan page\n default() {\n return null;\n },\n },\n maxspan: {\n type: Number,\n default() {\n return 0;\n },\n },\n },\n data() {\n return {\n dragLine: null,\n dragEventListener: null,\n deleteMode: false,\n conditionOptions: stringKeys.conditions,\n text: strings.itemText,\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 fromid = event.data.id;\n let toid = this.value.id;\n this.redrawLines();\n call([{\n methodname: 'local_treestudyplan_connect_studyitems',\n args: {'from_id': fromid, 'to_id': toid}\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 return;\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 return;\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_treestudyplanEdit_studyitem',\n args: {\n 'id': this.value.id,\n 'conditions': this.value.conditions,\n 'continuation_id': this.value.continuation_id,\n }\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 cin = this.value.connections.in[i];\n if (conn.id == cin.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(reid) {\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 == reid) {\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(itemid) {\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 == itemid) {\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 == itemid) {\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: {\n 'id': self.value.id,\n }\n }])[0].then((response) => {\n if (response.success == true) {\n self.$emit(\"deleted\", {data: self.value});\n }\n return;\n }).catch(notification.exception);\n }\n return;\n }).catch(err => {\n debug.console.error(err);\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 // 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 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 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 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.prevent=\"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.prevent=\"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=\"conditionOptions\"\n ></b-form-select>\n </b-form-group>\n\n <template #modal-footer=\"{ ok, cancel, hide }\" >\n <a href='#' @click.prevent='deleteItem()' class=\"text-danger\"\n ><i class=\"fa fa-trash\"></i>\n {{ text.delete }}\n </a>\n <b-button size=\"sm\" variant=\"primary\" @click.prevent=\"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() {\n return null;\n },\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.prevent='$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() {\n return null;\n },\n },\n plan: {\n type: Object,\n default() {\n return null;\n },\n },\n line: {\n type: Object,\n default() {\n return null;\n },\n },\n page: {\n type: Object, // PAge data\n default() {\n return null;\n }\n },\n period: {\n type: Object, // Period data\n default() {\n return null;\n }\n },\n maxspan: {\n type: Number,\n default() {\n return 0;\n }\n },\n },\n data() {\n return {\n conditionOptions: stringKeys.conditions,\n text: strings.itemCourseText,\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 } 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 formatDate(this.value.course.startdate);\n },\n enddate() {\n if (this.value.course.enddate > 0) {\n return formatDate(this.value.course.enddate);\n } else {\n return this.text.noenddate;\n }\n },\n wwwroot() {\n return Config.wwwroot;\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: {\n '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: {\n '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_treestudyplanEdit_studyitem',\n args: {\n 'id': this.value.id,\n 'conditions': this.value.conditions,\n }\n }])[0].catch(notification.exception);\n },\n },\n template: `\n <div class=\"t-item-course card\">\n <div class='t-item-course-cardwrapper mr-0 ml-0 h-100 '>\n <div\n :title=\"text['coursetiming_'+value.course.timing]\"\n v-b-popover.hover.top=\"startdate+' - '+enddate\"\n :class=\"'h-100 t-timing-indicator timing-'+value.course.timing\"\n ></div>\n <div class=\"t-item-course-title card-body h-100\">\n <fittext maxsize=\"12pt\" minsize=\"9pt\">\n <a v-b-modal=\"'t-item-course-config-'+value.id\"\n :id=\"'t-item-course-details-'+value.id\"\n :href=\"wwwroot+'/course/view.php?id='+value.course.id\"\n @click.prevent.stop=\"\">{{ value.course.displayname }}</a>\n </fittext>\n </div>\n <div class=\"h-100 t-item-course-indicator \">\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 </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=\"wwwroot+'/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=\"wwwroot+'/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.prevent='$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.prevent=\"ok()\">\n {{ text.ok }}\n </b-button>\n </template>\n </b-modal>\n </div>\n `,\n });\n\n\n Vue.component('t-item-course-grades', {\n props: {\n 'value': {\n type: Object,\n default() {\n return null;\n },\n },\n 'plan': {\n type: Object,\n default() {\n return null;\n },\n },\n },\n data() {\n return {\n conditionOptions: stringKeys.conditions,\n text: strings.itemCourseText,\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 } 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: {\n '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: {\n '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 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() {\n return {};\n },\n },\n guestmode: {\n type: Boolean,\n 'default': false,\n },\n course: {\n type: Object,\n default() {\n return {};\n },\n },\n },\n data() {\n return {\n text: strings.completion,\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 wwwroot() {\n return Config.wwwroot;\n }\n },\n methods: {\n completionIcon(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 completionTag(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=\"wwwroot+'/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() {\n return {};\n },\n },\n guestmode: {\n type: Boolean,\n 'default': false,\n },\n item: {\n type: Object,\n default() {\n return {id: null};\n },\n }\n },\n data() {\n return {\n text: strings.competency,\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 wwwroot() {\n return Config.wwwroot;\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: {\n '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=\"wwwroot+'/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=\"wwwroot+'/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() {\n return {};\n },\n },\n },\n data() {\n return {\n conditionOptions: stringKeys.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() {\n return {};\n },\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() {\n return {};\n },\n },\n },\n data() {\n return {};\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() {\n return {\n badge: {}\n };\n },\n },\n },\n data() {\n return {\n txt: strings,\n text: strings.itemText,\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.prevent='$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.prevent=\"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() {\n return {};\n },\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() {\n return {};\n },\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)) ||\n (this.value.hascourses && (!this.value.courses));\n },\n onShowDetails() {\n const self = this;\n if (this.canLoadMore()) {\n call([{\n methodname: 'local_treestudyplan_get_category',\n args: {\n \"id\": this.value.id,\n }\n }])[0].then((response) => {\n self.$emit('input', response);\n return;\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 t-caret\"></i>\n <i class=\"when-open fa fa-caret-down t-caret\"></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 <i class=\"when-closed fa t-caret\" style=\"visibility: hidden\"></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 <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() {\n return {};\n },\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 <span class='t-course-heading'>\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 </span>\n </li>\n </ul>\n `,\n });\n\n Vue.component('t-toolbox', {\n props: {\n value: {\n type: Boolean,\n 'default': true,\n },\n activepage: {\n type: Object,\n default() {\n return null;\n }\n },\n coaching: {\n type: Boolean,\n 'default': false,\n },\n studyplanid: {\n type: Number,\n 'default': 0,\n }\n },\n data() {\n return {\n toolboxright: !(settings(\"toolboxleft\")), \n text: strings.toolbox,\n relatedbadges: [],\n systembadges: [],\n courses: [],\n filters: {\n courses: \"\",\n systembadges: \"\",\n relatedbadges: \"\",\n },\n loadingcourses: false,\n loadingcategories: [],\n badgelistshown: {\n relatedbadges: true,\n systembadges: false,\n }\n };\n },\n watch: {\n // Whenever activepage changes, this function will run\n activepage(/* Params newVal, oldVal */) {\n this.filterRelatedbadges();\n }\n },\n mounted() {\n const self = this;\n this.initialize();\n\n this.$root.$on('bv::collapse::state', (collapseId, isJustShown) => {\n self.badgelistshown[collapseId] = !!isJustShown;\n });\n },\n computed: {\n filterComponentType() {\n return {\n item: false,\n component: true,\n span: 1,\n type: 'filter',\n };\n },\n filteredCourses() {\n const self = this;\n if (self.filters.courses) {\n return self.filterCategories(self.courses);\n } else {\n return self.courses;\n }\n }\n },\n methods: {\n hivizdrop() {\n return settings(\"hivizdropslots\");\n },\n filterCategories(catlist) {\n const self = this;\n const list = [];\n const search = new RegExp(`.*?${self.filters.courses}.*?`, \"ig\");\n for (const cat of catlist) {\n const clone = Object.assign({}, cat);\n clone.courses = [];\n\n if (cat.courses) { \n for (const course of cat.courses) {\n if (search.test(course.shortname) || search.test(course.fullname)) {\n clone.courses.push(course);\n }\n }\n } else if (cat.hascourses && !(self.loadingcategories.includes(cat.id))) {\n self.loadingcategories.push(cat.id);\n debug.info(`Loading from category ${cat.category.name}`, cat);\n call([{\n methodname: 'local_treestudyplan_get_category',\n args: {\n \"id\": cat.id\n }\n }])[0].then((response) => {\n // Add reactive array 'children' to cat\n self.$set(cat, \"children\", response.children);\n self.$set(cat, \"courses\", response.courses);\n self.loadingcategories.splice(self.loadingcategories.indexOf(cat.id), 1);\n return;\n }).catch(notification.exception);\n }\n \n if (cat.children) {\n clone.children = self.filterCategories(cat.children);\n } else if (cat.haschildren && !(self.loadingcategories.includes(cat.id))) {\n self.loadingcategories.push(cat.id);\n debug.info(`Loading from category ${cat.category.name}`, cat);\n call([{\n methodname: 'local_treestudyplan_get_category',\n args: {\n \"id\": cat.id,\n }\n }])[0].then((response) => {\n // Add reactive array 'children' to cat\n self.$set(cat, \"children\", response.children);\n self.loadingcategories.splice(self.loadingcategories.indexOf(cat.id), 1);\n return;\n }).catch(notification.exception);\n }\n \n if ((clone.children && clone.children.length) || clone.courses.length) {\n list.push(clone);\n }\n }\n\n return list;\n },\n\n initialize() {\n const self = this;\n self.loadingcourses = true;\n call([{\n methodname: 'local_treestudyplan_map_categories',\n args: {\n 'studyplan_id': self.studyplanid,\n }\n }])[0].then((response) => {\n self.courses = response;\n self.loadingcourses = false;\n return;\n }).catch(notification.exception);\n this.filterSystembadges();\n this.filterRelatedbadges();\n },\n filterSystembadges() {\n const self = this;\n call([{\n methodname: 'local_treestudyplan_search_badges',\n args: { \n search: this.filters.systembadges || \"\",\n }\n }])[0].then((response) => {\n self.systembadges = response;\n return;\n }).catch(notification.exception);\n },\n filterRelatedbadges() {\n const self = this;\n if (this.activepage) {\n call([{\n methodname: 'local_treestudyplan_search_related_badges',\n args: { \n 'page_id': this.activepage.id,\n search: this.filters.relatedbadges || \"\"\n }\n }])[0].then((response) => {\n self.relatedbadges = response;\n return;\n }).catch(notification.exception);\n }\n },\n resetSystembadges() {\n this.filters.systembadges = \"\";\n this.filterSystembadges();\n },\n resetRelatedbadges() {\n this.filters.relatedbadges = \"\";\n this.filterRelatedbadges();\n },\n },\n template: `\n <div class=\"t-toolbox\">\n <p-sidebar\n class=\"t-toolbox-sidebar\"\n :right='toolboxright'\n shadow\n v-model=\"value\"\n offsetRef=\"#page\"\n >\n <div class=\"m-3 border-bottom-1 border-primary\"><h3>{{text.toolbox}}</h3></div>\n <div class='t-toolbox-preface'>\n <b-form-checkbox v-model=\"toolboxright\" switch>{{text.toolbarRight}}</b-form-checkbox>\n </div>\n <b-tabs content-class='mt-3' class=\"t-toolbox-tabs\">\n <b-tab :title=\"text.courses\" class=\"t-toolbox-tab\">\n <div v-if=\"loadingcourses\"\n ><div class=\"spinner-border text-primary\" role=\"status\"></div\n ></div>\n <div v-else class=\"ml-2 t-toolbox-courses\">\n <div class=\"flex-grow-0\">\n <input v-model=\"filters.courses\" :placeholder=\"text.filter\"></input>\n &nbsp; <a @click=\"filters.courses = ''\" v-if=\"filters.courses\" href='#'\n ><i class='fa fa-times'></i></a\n ><b-spinner small v-if=\"loadingcategories.length > 0\" variant=\"primary\"></b-spinner>\n </div>\n <div class=\"t-toolbox-courselist\">\n <t-coursecat-list v-model=\"filteredCourses\"></t-coursecat-list>\n </div>\n </div>\n </b-tab>\n <b-tab :title=\"text.flow\" class=\"t-toolbox-tab\">\n <ul class=\"t-flow\">\n <li><drag\n :type=\"filterComponentType\"\n :data=\"{type: 'junction'}\"\n @cut=\"\"\n ><t-item-junction></t-item-junction>{{ text.toolJunction }}\n <template v-slot:drag-image=\"{data}\"><t-item-junction></t-item-junction></template>\n </drag></li>\n <li><drag\n :type=\"filterComponentType\"\n :data=\"{type: 'finish'}\"\n @cut=\"\"\n ><t-item-finish></t-item-finish>{{ text.toolFinish }}\n <template v-slot:drag-image=\"{data}\"><t-item-finish></t-item-finish></template>\n </drag></li>\n <li><drag\n :type=\"filterComponentType\"\n :data=\"{type: 'start'}\"\n @cut=\"\"\n ><t-item-start></t-item-start>{{ text.toolStart }}\n <template v-slot:drag-image=\"{data}\"><t-item-start></t-item-start></template>\n </drag></li>\n </ul>\n </b-tab>\n <b-tab :title=\"text.badges\" class=\"t-toolbox-tab\">\n \n <b-tabs content-class='mt-2 ml-2' class=\"t-toolbox-badges\">\n <b-tab :title=\"text.relatedbages\">\n <div class=\"t-toolbox-badges-filter\">\n <input v-model=\"filters.relatedbadges\" @input=\"filterRelatedbadges\" :placeholder=\"text.filter\"></input>\n &nbsp;<a @click=\"resetRelatedbadges\" v-if=\"filters.relatedbadges\" href='#'\n ><i class='fa fa-times'></i></a>\n </div>\n <div class=\"t-toolbox-badges-list\">\n <ul class=\"t-badges\">\n <li v-for=\"b in relatedbadges\"><drag\n class=\"t-badge-drag\"\n :type=\"filterComponentType\"\n :data=\"{type: 'badge', badge: b}\"\n @cut=\"\"\n ><img :class=\"(!b.active)?'disabled':''\" :src=\"b.imageurl\" :alt=\"b.name\"> \n <span :class=\"(!b.active)?'disabled':''\">{{b.name}}</span>\n <template v-slot:drag-image=\"{data}\"\n ><img :class=\"(!b.active)?'disabled':''\" :src=\"b.imageurl\" :alt=\"b.name\"\n ></template>\n </drag></li>\n </ul>\n </div>\n </b-tab>\n <b-tab :title=\"text.sitebadges\" v-if=\"!coaching\">\n <div class=\"t-toolbox-badges-filter\">\n <input v-model=\"filters.systembadges\" @input=\"filterSystembadges\" :placeholder=\"text.filter\"></input>\n &nbsp; <a @click=\"resetSystembadges\" v-if=\"filters.systembadges\" href='#'\n ><i class='fa fa-times'></i></a>\n </div>\n <div class=\"t-toolbox-badges-list\">\n <ul class=\"t-badges\">\n <li v-for=\"b in systembadges\"><drag\n class=\"t-badge-drag\"\n :type=\"filterComponentType\"\n :data=\"{type: 'badge', badge: b}\"\n @cut=\"\"\n ><img :class=\"(!b.active)?'disabled':''\" :src=\"b.imageurl\" :alt=\"b.name\"> \n <span :class=\"(!b.active)?'disabled':''\">{{b.name}}</span>\n <template v-slot:drag-image=\"{data}\"\n ><img :class=\"(!b.active)?'disabled':''\" :src=\"b.imageurl\" :alt=\"b.name\"\n ></template>\n </drag></li>\n </ul>\n </div>\n </b-tab>\n </b-tabs>\n </b-tab>\n </b-tabs>\n </p-sidebar>\n </div>\n `,\n });\n\n },\n \n};\n"],"names":["STUDYPLAN_EDITOR_FIELDS","PERIOD_EDITOR_FIELDS","install","Vue","component","Drag","Drop","DropList","use","TSComponents","mFormComponents","pSideBarComponents","FitTextVue","debug","Debugger","isVisible","elem","offsetWidth","offsetHeight","getClientRects","length","ItemEventBus","stringKeys","conditions","value","textkey","strings","studyplanText","add","edit","associations","drophere","studylineConfirmRemove","studyplanConfirmRemove","studyplanAdvanced","success","error","currentpage","studyplanEdit","warning","periodEdit","fullname","shortname","startdate","enddate","courseTiming","title","desc","question","course","period","yes","no","duration","years","year","weeks","week","days","day","rememberchoice","hidewarning","periodspan","periods","studyplanAssociate","cohorts","users","coaches","selected","name","context","search","itemText","ok","itemCourseText","cancel","noenddate","invalid","completion","competency","when","required","points","heading","details","badge","dateissued","dateexpire","badgeinfo","toolbox","toolbarRight","courses","flow","toolJunction","toolFinish","toolStart","badges","relatedbages","filter","sitebadges","props","type","Object","default","selectedpage","data","forceScales","selectedScale","result","text","computed","scales","id","disabled","this","advanced_pick_scale","concat","advanced","force_scales","methods","forceScalesStart","self","$bvModal","msgBoxConfirm","advanced_force_scale_confirm","okVariant","okTitle","confirm_ok","cancelTitle","confirm_cancel","then","methodname","args","response","catch","notification","exception","exportPage","format","undefined","includes","content","exportPlan","bulkCourseTiming","location","reload","msgBoxOk","msg","importStudylines","filename","importPages","purgeStudyplan","force","purgeStudyplanpage","cascadeCohortsync","advanced_cascade_cohortsync","modalClose","template","String","Number","planSaved","updatedplan","info","mode","$emit","movedFrom","context_id","movedTo","moved","pages","updatedpage","page","info_periodsextended","centered","show","config","userfields","key","cohortfields","association","loading","associated","premiumenabled","showModal","loadAssociations","cohortOptionModel","c","path","join","userOptionModel","u","firstname","lastname","map","searchCohorts","searchtext","like","cohortAssociate","searchselected","i","r","cohortDisassociate","associatedselected","searchUsers","userAssociate","userDisassociate","searchCoaches","coachAssociate","coachDisassociate","editdata","Date","getFullYear","editStart","editFinish","refresh","addDay","date","subDay","Boolean","create","studyline","color","enrol","enrollable","enrolroles","editmode","original","availableroles","studyplan","description","slots","aggregation","useRequiredGrades","useItemCondition","cache","linelayers","selectedpageindex","emptyline","filterslots","courseslots","created","$on","mounted","studylines","coaching","$root","updated","hivizdrop","columns","columnsStylerule","s","trashbinAccepts","item","countLineLayers","line","timestamp","maxLayer","ix","layer","filters","slotsempty","Array","isArray","count","movedStudyplan","plan","from","to","addStudyLine","newlineinfo","push","editLineStart","assign","editLineFinish","editedline","originalline","deleteLine","replace","delete","modalresponse","index","indexOf","splice","reorderLines","event","lines","apply","sequence","idx","deletePlan","studyplabConfirmRemove","deleteStudyItem","source","showslot","layeridx","forGradable","list","span","periodEdited","pi","prev","$refs","next","toolboxSwitched","pagecreated","selectedpageChanged","newTabIndex","sumLineLayers","sum","slot","onDrop","lineslot","nextlayer","itm","$nextTick","timingChecker","validateCoursePeriod","checkTypeCourse","checkTypeFilter","layers","layerHeights","onLineHeightChange","lineid","main","items","document","querySelectorAll","heightSum","forEach","el","height","getBoundingClientRect","heightStyle","style","deletable","editable","onEdit","onDelete","slotindex","resizeListener","ResizeObserver","size","observe","unmounted","disconnect","slotkey","itemidx","listtype","courseHoverDummy","hover","spanCss","plantext","datechanger","coursespan","defaultchoice","hidewarn","relocateStudyItem","onCut","iteminfo","onDragEnter","onDragLeave","maxSpan","makeType","checkType","maxspan","hidden","endperiod","endperiodnr","Math","min","perioddesc","coursePeriodMatches","timeless","first","getTime","last","warn","floor","random","now","toString","globals","canupdatecourse","defaultvalue","changeCoursePeriod","timing","checkFilterSlotBusy","nextFreeFilterLayer","usedLayers","nextlyr","checkCourseSlotBusy","nextFreeCourseLayer","shiftCollisions","busyFilter","busyCourse","changeSpan","formatDuration","dsi","toLocaleLowerCase","dummy","dragLine","dragEventListener","deleteMode","conditionOptions","showContext","dragStart","start","getElementById","dragelement","position","left","x","top","y","SimpleLine","gravity","end","addEventListener","onMouseMove","dragEnd","remove","removeProperty","removeEventListener","clientX","clientY","fromid","toid","redrawLines","conn","from_id","to_id","connections","in","redrawLine","removeLine","out","highlight","setConfig","normalize","updateItem","continuation_id","doShowContext","hasContext","preventDefault","onCreatedConnection","onRemovedConnection","cin","onRePositioned","onDisPositioned","reid","onItemDeleted","itemid","onRedrawLines","deleteItem","msgparams","displayname","item_delete_message","err","console","hasConnectionsOut","hasConnectionsIn","setTimeout","beforeDestroy","$off","useItemConditions","aggregation_info","configurationState","hasGrades","hasCompletions","hasCompetencies","configurationIcon","wwwroot","Config","grades","g","cgroup","competencies","includeChanged","newValue","requiredChanged","updateConditions","selectedgrades","guestmode","completionIcon","completionTag","pathtags","p","url","contextid","txt","showSpinner","canLoadMore","hasDetails","haschildren","hascourses","children","onShowDetails","activepage","studyplanid","toolboxright","relatedbadges","systembadges","loadingcourses","loadingcategories","badgelistshown","watch","filterRelatedbadges","initialize","collapseId","isJustShown","filterComponentType","filteredCourses","filterCategories","catlist","RegExp","cat","clone","test","category","$set","filterSystembadges","resetSystembadges","resetRelatedbadges"],"mappings":"ioCA6BMA,wBACN,CAAC,OAAQ,YAAa,cAAe,WAAY,aAAc,cAAe,sBACxEC,qBACN,CAAC,WAAY,YAAa,YAAa,wBAIxB,CACXC,QAAQC,KACJA,IAAIC,UAAU,OAAQC,kBACtBF,IAAIC,UAAU,OAAQE,kBACtBH,IAAIC,UAAU,YAAaG,sBAC3BJ,IAAIK,IAAIC,kCACRN,IAAIK,IAAIE,sBACRP,IAAIK,IAAIG,sBACRR,IAAIK,IAAII,yBACJC,MAAQ,IAAIC,kBAAS,iCAYhBC,UAAUC,eACLA,KAAKC,aAAeD,KAAKE,cAAgBF,KAAKG,iBAAiBC,cAIvEC,aAAe,IAAIlB,QAUrBmB,YAAa,gCAAe,CAC5BC,WAAY,CACR,CAACC,MAAO,MAAOC,QAAS,iBACxB,CAACD,MAAO,MAAOC,QAAS,oBAI5BC,SAAU,6BAAY,CACtBC,cAAe,oBACW,oCACJ,yCACS,wCACV,gBACjBC,IAAK,WACLC,KAAM,mBACI,6BACQ,mCACG,wCACE,6CACG,8CACF,4CACA,uCACL,kBACnBC,aAAc,kCACQ,sCACF,kCACF,gCACA,mCACG,wCACE,6CACG,+CACD,iDACG,8CACN,2CACG,wCACN,sCACI,wCACF,sCACA,sCACA,sCACA,sCACA,oBACrBC,SAAU,WACVC,uBAAwB,2BACxBC,uBAAwB,4BAG5BC,kBAAmB,gBACG,gCACA,4BACJ,aACdC,QAAS,eACTC,MAAO,uCACe,sCACF,mCACD,kCACA,oCACE,mCACH,wCACQ,gDACA,0CACN,uCACG,yDACe,iEACP,+DACK,8DACN,uDACD,wDACE,sDACJ,uDACK,uDACL,2CACP,mCACD,yCACO,gDACA,8CACF,4CACA,uCACL,2CACS,oDACA,qDACC,2CACX,qCACK,+CACK,+CACL,+CACK,6DACS,qEACD,+DACL,8BAC/BC,YAAa,eAEjBC,cAAe,CACXA,cAAe,8BACE,kCACI,uCACC,0CACE,qCACxBC,QAAS,gBAEbC,WAAY,CACRX,KAAM,cACNY,SAAU,iBACVC,UAAW,sBACXC,UAAW,sBACXC,QAAS,qBAEbC,aAAc,CACVC,MAAO,sBACPC,KAAM,qBACNC,SAAU,yBACVT,QAAS,kCACI,8BACC,oBACdU,OAAQ,cACRC,OAAQ,SACRC,IAAK,WACLC,GAAI,UACJC,SAAU,WACVC,MAAO,aACPC,KAAM,YACNC,MAAO,aACPC,KAAM,YACNC,KAAM,YACNC,IAAK,WACLC,eAAgB,+BAChBC,YAAa,4BACbC,WAAY,qBACZC,QAAS,0BACU,2BAEvBC,mBAAoB,cACA,kCACM,sCACF,sCACE,uCACD,oCACF,oCACE,oCACF,qCACG,wCACA,yCACC,sBACvBC,QAAS,UACTC,MAAO,QACPC,QAAS,UACTC,SAAU,WACVC,KAAM,OACNC,QAAS,UACTC,OAAQ,UAEZC,SAAU,mBACe,uCACC,qBACtBC,GAAI,iBACM,kCACa,kCACR,4BACE,2BACH,yBACC,yBACD,0BACE,kBAEpBC,eAAgB,mBACS,kCACJ,kCACI,yCACG,2CACD,oCACN,8BACA,gBACjBD,GAAI,UACJE,OAAQ,qBACE,cACVC,UAAW,aAEfC,QAAS,CACLzC,MAAO,SAEX0C,WAAY,sBACgB,6CACC,wCACN,kCACA,0CACQ,kDACA,oDACE,iDACL,wBAE5BC,WAAY,2BACqB,iDACL,uBACxBC,KAAM,OACNC,SAAU,WACVC,OAAQ,qBACRC,QAAS,qBACTC,QAAS,sBAEbC,MAAO,aACY,cACfC,WAAY,aACZC,WAAY,aACZC,UAAW,aAEfC,QAAS,CACLA,QAAS,UACTC,aAAc,gBACdC,QAAS,UACTC,KAAM,OACNC,aAAc,gBACdC,WAAY,cACZC,UAAW,aACXC,OAAQ,SACRC,aAAc,sBACdC,OAAQ,cACRC,WAAY,uBAOpBhG,IAAIC,UAAU,uBAAwB,CAClCgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,IACI,MAGfC,aAAc,CACVH,KAAMC,OACNC,QAAO,IACI,OAKnBE,KAAI,KACO,CACHC,YAAa,CACTC,cAAe,KACfC,OAAQ,IAEZC,KAAMnF,QAAQQ,oBAGtB4E,SAAU,CACNC,eACW,CAAC,CACJC,GAAI,KACJC,UAAU,EACV5C,KAAM6C,KAAKL,KAAKM,sBACjBC,OAAOF,KAAK1F,MAAM6F,SAASC,aAAaP,UAGnDQ,QAAS,CACLC,yBAEUC,KAAOP,UACRQ,SAASC,cAAcT,KAAKL,KAAKe,6BAA8B,CAChE9E,MAAOoE,KAAKL,KAAKe,6BACjBC,UAAW,SACXC,QAASZ,KAAKL,KAAKkB,WACnBC,YAAad,KAAKL,KAAKoB,iBACxBC,MAAK1G,QACS,GAATA,sBACK,CAAC,CACF2G,WAAY,4CACZC,KAAM,cACclB,KAAK1F,MAAMwF,YACfE,KAAKR,YAAYC,kBAEjC,GAAGuB,MAAMG,WACTZ,KAAKf,YAAYE,OAASyB,YAE3BC,MAAMC,sBAAaC,cAG3BF,MAAMC,sBAAaC,YAE1BC,WAAWC,cACDjB,KAAOP,KACCyB,MAAVD,QAAwB,CAAC,OAAQ,OAAOE,SAASF,UACjDA,OAAS,uBAER,CAAC,CACEP,WAAY,kCACZC,KAAM,SACSlB,KAAKV,aAAaQ,GAC7B0B,OAAQA,WAEZ,GAAGR,MAAMG,oCACAZ,KAAKjG,MAAMkB,UAAY,SAAWgG,OAAQL,SAASQ,QAASR,SAASK,WAE/EJ,MAAMC,sBAAaC,YAE9BM,mBACUrB,KAAOP,oBACR,CAAC,CACEiB,WAAY,kCACZC,KAAM,cACclB,KAAK1F,MAAMwF,GAC3B0B,OAAQ,WAEZ,GAAGR,MAAMG,oCACAZ,KAAKjG,MAAMkB,UAAY,aAAc2F,SAASQ,QAASR,SAASK,WAE1EJ,MAAMC,sBAAaC,YAE9BO,yBACUtB,KAAOP,oBACR,CAAC,CACEiB,WAAY,yCACZC,KAAM,SACSlB,KAAKV,aAAaQ,OAEjC,GAAGkB,MAAMG,WACLA,SAASlG,QAET6G,SAASC,UAETxB,KAAKC,SAASwB,SAASb,SAASc,IAAK,CAACrG,MAAO,qCAC7CjC,MAAMuB,MAAM,qCAAsCiG,SAASc,SAGhEb,MAAMC,sBAAaC,YAE9BY,yBACU3B,KAAOP,6BACN,CAACmC,SAAUR,0BACT,CAAC,CACFV,WAAY,wCACZC,KAAM,SACUlB,KAAKV,aAAaQ,GAC9B6B,QAASA,QACTH,OAAQ,uBAEZ,GAAGR,MAAMG,WACLA,SAASlG,QACT6G,SAASC,UAETxB,KAAKC,SAASwB,SAASb,SAASc,IAAK,CAACrG,MAAO,kBAC7CjC,MAAMuB,MAAM,kBAAmBiG,SAASc,SAG7Cb,MAAMC,sBAAaC,aACvB,qBAEPc,oBACU7B,KAAOP,6BACN,CAACmC,SAAUR,0BACT,CAAC,CACFV,WAAY,mCACZC,KAAM,cACelB,KAAK1F,MAAMwF,GAC5B6B,QAASA,QACTH,OAAQ,uBAEZ,GAAGR,MAAMG,WACLA,SAASlG,QACT6G,SAASC,UAETxB,KAAKC,SAASwB,SAASb,SAASc,IAAK,CAACrG,MAAO,kBAC7CjC,MAAMuB,MAAM,kBAAmBiG,SAASc,SAG7Cb,MAAMC,sBAAaC,aACvB,qBAEPe,uBACU9B,KAAOP,oBACR,CAAC,CACFiB,WAAY,uCACZC,KAAM,CACFpB,GAAKE,KAAK1F,MAAMwF,GAChBwC,OAAO,MAEX,GAAGtB,MAAMG,WACLA,SAASlG,QACT6G,SAASC,UAETxB,KAAKC,SAASwB,SAASb,SAASc,IAAK,CAACrG,MAAO,2BAC7CjC,MAAMuB,MAAM,0BAA2BiG,SAASc,SAGrDb,MAAMC,sBAAaC,YAE1BiB,2BACUhC,KAAOP,KACTA,KAAKV,6BACA,CAAC,CACF2B,WAAY,2CACZC,KAAM,CACFpB,GAAKE,KAAKV,aAAaQ,GACvBwC,OAAO,MAEX,GAAGtB,MAAMG,WACLA,SAASlG,QACT6G,SAASC,UAETxB,KAAKC,SAASwB,SAASb,SAASc,IAAK,CAACrG,MAAO,0BAC7CjC,MAAMuB,MAAM,0BAA2BiG,SAASc,SAGrDb,MAAMC,sBAAaC,YAG9BkB,0BACUjC,KAAOP,oBACR,CAAC,CACFiB,WAAY,yCACZC,KAAM,cACelB,KAAK1F,MAAMwF,OAEhC,GAAGkB,MAAMG,WACTZ,KAAKC,SAASwB,SAASb,SAASlG,QAAUsF,KAAKZ,KAAK1E,QAAUsF,KAAKZ,KAAKzE,MACpE,CAACU,MAAO2E,KAAKZ,KAAK8C,iCAEvBrB,MAAMC,sBAAaC,YAE1BoB,kBACSlD,YAAYE,OAAS,KAGlCiD,SACC,kqNAoHL1J,IAAIC,UAAU,mBAAoB,CAC9BgG,MAAO,OACM,CACLC,KAAMC,OACNC,QAAO,IACI,WAGP,CACJF,KAAMyD,OACNvD,QAAO,IACI,aAGP,CACJF,KAAMyD,OACNvD,QAAO,IACI,gBAGJ,CACPF,KAAMyD,OACNvD,QAAO,IACI,cAGF,CACTF,KAAM0D,eACK,IAGnBtD,KAAI,KACO,CACHI,KAAMnF,QAAQY,gBAGtBwE,SAAU,GAEVS,QAAS,CACLyC,UAAUC,mBACAxC,KAAOP,QACbrG,MAAMqJ,KAAK,oBAAqBD,aAEf,UAAbxC,KAAK0C,KAEL1C,KAAK2C,MAAM,UAAWH,iBACnB,OAEGI,UAAY5C,KAAKjG,MAAM8I,WACvBC,QAAUN,YAAYK,WACtBE,MAASH,WAAaE,QAExBN,YAAYQ,MAAM,GAAG1G,SAAW0D,KAAKjG,MAAMiJ,MAAM,GAAG1G,uBAE/C,CAAC,CACFoE,WAAY,wCACZC,KAAM,CAACpB,GAAIS,KAAKjG,MAAMwF,OACtB,GAAGkB,MAAMG,WACTZ,KAAKjG,OAAQ,wCAAiB6G,UAAU,GACxCxH,MAAMqJ,KAAK,uBACXzC,KAAK2C,MAAM,QAAS3C,KAAKjG,UAE1B8G,OAAM,SAASlG,6BACDoG,UAAUpG,2CAInBqF,KAAKjG,MAAOyI,YAAajK,yBACjCyH,KAAK2C,MAAM,QAAS3C,KAAKjG,OACrBgJ,OACA/C,KAAK2C,MAAM,QAAS3C,KAAKjG,MAAO6I,UAAWE,aAM/DV,SACC,8gBAiBL1J,IAAIC,UAAU,wBAAyB,CACnCgG,MAAO,OACM,CACLC,KAAMC,OACNC,QAAO,IACI,WAGP,CACJF,KAAMyD,OACNvD,QAAO,IACI,aAGP,CACJF,KAAMyD,OACNvD,QAAO,IACI,gBAGH,CACRF,KAAMyD,OACNvD,QAAO,IACI,cAGF,CACTF,KAAMC,SAGdG,KAAI,KACO,CACHI,KAAMnF,QAAQY,gBAGtBwE,SAAU,GAEVS,QAAS,CACLyC,UAAUU,mBACAjD,KAAOP,QAEI,UAAbO,KAAK0C,KAEL1C,KAAK2C,MAAM,UAAWM,iBACnB,OACGC,MAAO,4CAAqBD,aAClC7J,MAAMqJ,KAAK,4BAEPzC,KAAKjG,MAAMuC,QAAU4G,KAAK5G,cACrB2D,SAASwB,SAAShC,KAAKL,KAAK+D,qBAAsB,CACnD9H,MAAOoE,KAAKL,KAAKtE,QACjBsF,UAAW,UACXgD,UAAU,IAGlBpD,KAAK2C,MAAM,QAASO,SAKhCd,SACC,kiBAkBL1J,IAAIC,UAAU,wBAAyB,CACnCgG,MAAO,CAAC,SACRK,KAAI,KACO,CACHqE,MAAM,EACNC,OAAQ,CACJC,WAAY,CACR,CAACC,IAAK,YACN,CAACA,IAAK,sBAAyB,GAC/B,CAACA,IAAK,qBAAwB,IAElCC,aAAc,CACV,CAACD,IAAK,YACN,CAACA,IAAK,iBAAoB,GAC1B,CAACA,IAAK,oBAAuB,KAGrCE,YAAa,CACTlH,QAAS,GACTC,MAAO,GACPC,QAAS,IAEbiH,QAAS,CACLnH,SAAS,EACTC,OAAO,EACPC,SAAS,GAEbI,OAAQ,CAACL,MAAO,GAAID,QAAS,GAAIE,QAAS,IAC1CC,SAAU,CACLG,OAAQ,CAACL,MAAO,GAAID,QAAS,GAAIE,QAAS,IAC1CkH,WAAY,CAACnH,MAAO,GAAID,QAAS,GAAIE,QAAS,KAEnD0C,KAAMnF,QAAQsC,qBAGtBuD,QAAS,CACL+D,eAAAA,wBACAC,iBACST,MAAO,OACPU,oBAETC,kBAAkBC,IACP,CACHlK,MAAOkK,EAAE1E,GACTH,KAAM6E,EAAErH,KAAO,KAAOqH,EAAEpH,QAAQqH,KAAKC,KAAK,OAAS,MAG3DC,gBAAgBC,IACL,CACHtK,MAAOsK,EAAE9E,GACTH,KAAMiF,EAAEC,UAAY,IAAMD,EAAEE,WAGpCR,yBACU/D,KAAOP,KACbO,KAAK2D,QAAQnH,SAAU,EACvBwD,KAAK2D,QAAQlH,OAAQ,iBAChB,CAAC,CACFiE,WAAY,uCACZC,KAAM,cACcX,KAAKjG,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAK0D,YAAYjH,MAAQmE,SAAS4D,IAAIxE,KAAKoE,iBAC3CpE,KAAK2D,QAAQlH,OAAQ,KAEtBoE,MAAMC,sBAAaC,0BAEjB,CAAC,CACFL,WAAY,yCACZC,KAAM,cACcX,KAAKjG,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAK0D,YAAYlH,QAAUoE,SAAS4D,IAAIxE,KAAKgE,mBAC7ChE,KAAK2D,QAAQnH,SAAU,KAExBqE,MAAMC,sBAAaC,YAElB,+BACAf,KAAK2D,QAAQjH,SAAU,iBAClB,CAAC,CACFgE,WAAY,yCACZC,KAAM,cACcX,KAAKjG,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAK0D,YAAYhH,QAAUkE,SAAS4D,IAAIxE,KAAKoE,iBAC7CpE,KAAK2D,QAAQjH,SAAU,KAExBmE,MAAMC,sBAAaC,aAG9B0D,cAAcC,kBACJ1E,KAAOP,KAETiF,WAAW/K,OAAS,iBACf,CAAC,CACF+G,WAAY,kCACZC,KAAM,CACFgE,KAAMD,wBACU1E,KAAKjG,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAKlD,OAAON,QAAUoE,SAAS4D,IAAIxE,KAAKgE,sBAEzCnD,MAAMC,sBAAaC,WAEtBf,KAAKlD,OAAON,QAAU,IAG9BoI,wBACU5E,KAAOP,WAEPmE,WAAa5D,KAAK0D,YAAYlH,QAC9BM,OAASkD,KAAKlD,OAAON,QACrBqI,eAAiB7E,KAAKrD,SAASG,OAAON,YACvC,MAAMsI,KAAKD,eAAgB,OACtBE,EAAIF,eAAeC,kBACpB,CAAC,CACFpE,WAAY,qCACZC,KAAM,cACcX,KAAKjG,MAAMwF,aACdwF,MAEjBtE,MAAMG,WACFA,SAASlG,+CACKkJ,WAAY9G,OAAQiI,MAGvClE,MAAMC,sBAAaC,0BAjBX,KAqBnBiE,2BACUhF,KAAOP,KACPwF,mBAAqBjF,KAAKrD,SAASiH,WAAWpH,QAC9CoH,WAAa5D,KAAK0D,YAAYlH,QAC9BM,OAASkD,KAAKlD,OAAON,YACtB,MAAMsI,KAAKG,mBAAoB,OAC1BF,EAAIE,mBAAmBH,kBACxB,CAAC,CACFpE,WAAY,wCACZC,KAAM,cACcX,KAAKjG,MAAMwF,aACdwF,MAEjB,GAAGtE,MAAMG,WACLA,SAASlG,+CACKkJ,WAAY9G,OAAQiI,MAGvClE,MAAMC,sBAAaC,aAG9BmE,YAAYR,kBACF1E,KAAOP,KACTiF,WAAW/K,OAAS,iBACf,CAAC,CACF+G,WAAY,gCACZC,KAAM,CACFgE,KAAMD,wBACU1E,KAAKjG,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAKlD,OAAOL,MAAQmE,SAAS4D,IAAIxE,KAAKoE,oBAEvCvD,MAAMC,sBAAaC,WAEtBf,KAAKlD,OAAOL,MAAQ,IAG5B0I,sBACUnF,KAAOP,KACPmE,WAAa5D,KAAK0D,YAAYjH,MAC9BK,OAASkD,KAAKlD,OAAOL,MACrBoI,eAAiB7E,KAAKrD,SAASG,OAAOL,UACvC,MAAMqI,KAAKD,eAAgB,OACtBE,EAAIF,eAAeC,kBACpB,CAAC,CACFpE,WAAY,mCACZC,KAAM,cACcX,KAAKjG,MAAMwF,WAChBwF,MAEftE,MAAMG,WACFA,SAASlG,+CACKkJ,WAAY9G,OAAQiI,MAGvClE,MAAMC,sBAAaC,aAG9BqE,yBACUpF,KAAOP,KACPmE,WAAa5D,KAAK0D,YAAYjH,MAC9BwI,mBAAqBjF,KAAKrD,SAASiH,WAAWnH,MAC9CK,OAASkD,KAAKlD,OAAOL,UACtB,MAAMqI,KAAKG,mBAAoB,OAC1BF,EAAIE,mBAAmBH,kBACxB,CAAC,CACFpE,WAAY,sCACZC,KAAM,cACcX,KAAKjG,MAAMwF,WAChBwF,MAEf,GAAGtE,MAAMG,WACLA,SAASlG,+CACKkJ,WAAY9G,OAAQiI,MAGvClE,MAAMC,sBAAaC,aAG9BsE,cAAcX,gBACN,6BAAkB,OACZ1E,KAAOP,KACTiF,WAAW/K,OAAS,iBACf,CAAC,CACF+G,WAAY,iCACZC,KAAM,CACFgE,KAAMD,wBACU1E,KAAKjG,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAKlD,OAAOJ,QAAUkE,SAAS4D,IAAIxE,KAAKoE,oBAEzCvD,MAAMC,sBAAaC,WAEtBf,KAAKlD,OAAOJ,QAAU,KAIlC4I,qBACQ,6BAAkB,OACZtF,KAAOP,KACPmE,WAAa5D,KAAK0D,YAAYhH,QAC9BI,OAASkD,KAAKlD,OAAOJ,QACrBmI,eAAiB7E,KAAKrD,SAASG,OAAOJ,YACvC,MAAMoI,KAAKD,eAAgB,OACtBE,EAAIF,eAAeC,kBAEpB,CAAC,CACFpE,WAAY,oCACZC,KAAM,cACcX,KAAKjG,MAAMwF,WAChBwF,MAEftE,MAAMG,WACFA,SAASlG,+CACKkJ,WAAY9G,OAAQiI,MAGvClE,MAAMC,sBAAaC,cAIlCwE,wBACQ,6BAAkB,OACZvF,KAAOP,KACPmE,WAAa5D,KAAK0D,YAAYhH,QAC9BuI,mBAAqBjF,KAAKrD,SAASiH,WAAWlH,QAC9CI,OAASkD,KAAKlD,OAAOJ,YACtB,MAAMoI,KAAKG,mBAAoB,OAC1BF,EAAIE,mBAAmBH,kBAExB,CAAC,CACFpE,WAAY,uCACZC,KAAM,cACcX,KAAKjG,MAAMwF,WAChBwF,MAEf,GAAGtE,MAAMG,WACLA,SAASlG,+CACKkJ,WAAY9G,OAAQiI,MAGvClE,MAAMC,sBAAaC,eAKtCqB,SACX,07NA+JO1J,IAAIC,UAAU,gBAAiB,CAC3BgG,MAAO,OACM,CACLC,KAAMC,OACNC,QAAO,IACI,WAGP,CACJF,KAAMyD,OACNvD,QAAO,IACI,gBAGJ,CACPF,KAAMyD,OACNvD,QAAO,IACI,aAGH,CACRF,KAAMyD,OACNvD,QAAO,IACI,aAGL,CACNF,KAAMyD,OACNvD,QAAO,IACI,OAInBE,KAAI,KACO,CACHqE,MAAM,EACNmC,SAAU,CACNxK,SAAU,GACVC,UAAW,GACXC,WAAY,IAAIuK,MAAQC,cAAgB,SACxCvK,SAAY,IAAIsK,MAAQC,cAAgB,EAAK,UAEjDtG,KAAMnF,QAAQc,aAGtB+E,QAAS,CACL6F,4CACYlG,KAAK+F,SAAU/F,KAAK1F,MAAOvB,2BAC9B6K,MAAO,GAEhBuC,mBACU5F,KAAOP,SACTkB,KAAO,IAAOlB,KAAK1F,MAAMwF,oCAErBoB,KAAMlB,KAAK+F,SAAUhN,qCAExB,CAAC,CACFkI,WAAY,iCACZC,KAAMA,QACN,GAAGF,MAAMG,2CACDZ,KAAKjG,MAAO6G,SAAUpI,sBAC9BwH,KAAK2C,MAAM,QAAS3C,KAAKjG,OACzBiG,KAAK2C,MAAM,SAAU3C,KAAKjG,UAE3B8G,MAAMC,sBAAaC,YAE1B8E,gBACU7F,KAAOP,oBACR,CAAC,CACFiB,WAAY,iCACZC,KAAM,IAAOlB,KAAK1F,MAAMwF,OACxB,GAAGkB,MAAMG,2CACDZ,KAAKjG,MAAO6G,SAAUpI,sBAC9BwH,KAAK2C,MAAM,QAAS3C,KAAKjG,UAE1B8G,MAAMC,sBAAaC,YAE1B+E,OAAM,CAACC,KAAM9J,aACIiF,IAATjF,OACAA,KAAO,IAEJ,uBAAQ8J,KAAM9J,OAEzB+J,OAAM,CAACD,KAAM9J,aACIiF,IAATjF,OACAA,KAAO,IAEJ,uBAAQ8J,KAAM,EAAI9J,QAGjCmG,SACC,23FA+DL1J,IAAIC,UAAU,cAAe,CACzBgG,MAAO,OACM,CACLC,KAAMC,iBAEE,CACRD,KAAMqH,iBACK,IAGnBjH,KAAI,KACO,CACHsE,OAAQ,CACJC,WAAY,CACR,CAACC,IAAK,YACN,CAACA,IAAK,sBAAyB,GAC/B,CAACA,IAAK,qBAAwB,IAElCC,aAAc,CACV,CAACD,IAAK,YACN,CAACA,IAAK,iBAAoB,GAC1B,CAACA,IAAK,oBAAuB,KAGrC0C,OAAQ,CACJC,UAAW,CACPvJ,KAAM,GACN3B,UAAW,GACXmL,MAAO,UACPC,MAAO,CACHC,WAAY,EACZC,WAAY,KAGpBrD,KAAM,CACF3D,IAAK,EACL3C,KAAM,GACN3B,UAAW,KAGnBb,KAAM,gBACe,EACjB+L,UAAW,CACPK,UAAU,EACVxH,KAAM,CACFpC,KAAM,GACN3B,UAAW,GACXmL,MAAO,UACPC,MAAO,CACHC,WAAY,EACZC,WAAY,KAGpBE,SAAU,GACVC,eAAgB,IAEpBC,UAAW,CACP3H,KAAM,CACFpC,KAAM,GACN3B,UAAW,GACX2L,YAAa,GACbC,MAAO,EACP3L,UAAW,aACXC,QAAS,GACT2L,YAAa,sBACS,oBACF,CAChBC,mBAAmB,EACnBC,kBAAkB,IAI1BP,SAAU,KAGlBrH,KAAMnF,QAAQC,cACd+M,MAAO,CACHC,WAAY,IAEhBC,kBAAmB,EACnBC,UAAW,CACP7H,IAAK,EACL3C,KAAM,2BACN3B,UAAW,mBACXmL,MAAO,UACPiB,YAAa,CAAC,IACdC,YAAa,CAAC,KAElBZ,eAAgB,KAGxBa,gBACcvH,KAAOP,KAGb7F,aAAa4N,IAAI,gBAAgB,KAC7BxH,KAAK2C,MAAM,cAAelD,KAAKV,kBAG3C0I,gBACUzH,KAAOP,KACgC,GAAzCA,KAAK1F,MAAMiJ,MAAM,GAAG0E,WAAW/N,QAAgB8F,KAAKkI,gBAE/CvN,KAAK+L,UAAUK,UAAW,GAG9BxG,KAAK2H,yBAED,CAAC,CACFjH,WAAY,iCACZC,KAAM,cACclB,KAAK1F,MAAMwF,OAE/B,GAAGkB,MAAMG,WACTZ,KAAK0G,eAAiB9F,YAEvBC,MAAMC,sBAAaC,gBAErB6G,MAAMjF,MAAM,oBACZA,MAAM,cAAelD,KAAKV,eAEnC8I,eACSD,MAAMjF,MAAM,eACjB/I,aAAa+I,MAAM,gBAEvBtD,SAAU,CACNN,sBACWU,KAAK1F,MAAMiJ,MAAMvD,KAAK0H,oBAEjCW,UAAS,KACE,sBAAS,mBAGxBhI,QAAS,CACL+D,eAAAA,wBACAkE,QAAQ7E,MACG,EAAoB,EAAfA,KAAK5G,QAErB0L,iBAAiB9E,UAET+E,EAAI,2DACH,IAAInD,EAAI,EAAGA,EAAI5B,KAAK5G,QAASwI,IAC9BmD,GAAK,sEAEFA,EAAI,KAEfC,gBAAgBtJ,QACRA,KAAKuJ,KAMbC,gBAAgBC,KAAMnF,SAGdzD,KAAKwH,MAAMC,WAAWmB,KAAK9I,KACtB,IAAIkG,KAAUhG,KAAKwH,MAAMC,WAAWmB,KAAK9I,IAAI+I,UAAY,WAEvD7I,KAAKwH,MAAMC,WAAWmB,KAAK9I,IAAIxF,MACnC,KACCwO,UAAY,MACX,IAAIzD,EAAI,EAAGA,GAAK5B,KAAK5G,QAASwI,OAC3BuD,KAAKxB,MAAM/B,GAAI,KAEV,MAAM0D,MAAMH,KAAKxB,MAAM/B,GAAG5G,QAAS,OAC9BiK,KAAOE,KAAKxB,MAAM/B,GAAG5G,QAAQsK,IAC/BL,KAAKM,MAAQF,WACbA,SAAWJ,KAAKM,WAGnB,MAAMD,MAAMH,KAAKxB,MAAM/B,GAAG4D,QAAS,OAC9BP,KAAOE,KAAKxB,MAAM/B,GAAG4D,QAAQF,IAC/BL,KAAKM,MAAQF,WACbA,SAAWJ,KAAKM,oBAM3BxB,MAAMC,WAAWmB,KAAK9I,IAAM,CAC7BxF,MAAQwO,SAAW,EACnBD,UAAY,IAAI7C,MAEb8C,SAAW,IAG1BI,WAAW9B,UACH+B,MAAMC,QAAQhC,OAAQ,KAClBiC,MAAQ,MACP,IAAIhE,EAAI,EAAGA,EAAI+B,MAAMlN,OAAQmL,IAC1B8D,MAAMC,QAAQhC,MAAM/B,GAAG5G,WACvB4K,OAASjC,MAAM/B,GAAG5G,QAAQvE,QAE1BiP,MAAMC,QAAQhC,MAAM/B,GAAG4D,WACvBI,OAASjC,MAAM/B,GAAG4D,QAAQ/O,eAGjB,GAATmP,aAED,GAGfC,eAAeC,KAAMC,KAAMC,SAClBvG,MAAM,QAASqG,KAAMC,KAAMC,KAEpCC,aAAajG,KAAMkG,4BACV,CAAC,CACF1I,WAAY,oCACZC,KAAM,SACSuC,KAAK3D,QACR6J,YAAYxM,eACPwM,YAAYnO,gBAChBmO,YAAYhD,eACTlD,KAAKwE,WAAW/N,kBACdyP,YAAY/C,MAAMC,sBAClB8C,YAAY/C,MAAME,eAEpC,GAAG9F,MAAMG,WACTsC,KAAKwE,WAAW2B,KAAKzI,UACrBwI,YAAYxM,KAAO,GACnBwM,YAAYnO,UAAY,GACxBmO,YAAYhD,MAAQ,UACpBgD,YAAY/C,MAAMC,WAAa,EAC/B8C,YAAY/C,MAAME,WAAa,MAEhC1F,MAAMC,sBAAaC,YAE1BuI,cAAcjB,YACJnF,KAAOzD,KAAK1F,MAAMiJ,MAAMvD,KAAK0H,mBACnC/N,MAAMqJ,KAAK,qBAAsB4F,MACjCxJ,OAAO0K,OAAO9J,KAAKrF,KAAK+L,UAAUnH,KAAMqJ,WACnCjO,KAAK+L,UAAUM,SAAW4B,UAC1BpI,SAASoD,KAAK,wBAA0BH,KAAK3D,KAEtDiK,qBACQC,WAAahK,KAAKrF,KAAK+L,UAAUnH,KACjC0K,aAAejK,KAAKrF,KAAK+L,UAAUM,wBAClC,CAAC,CACF/F,WAAY,oCACZC,KAAM,IAAO8I,WAAWlK,QACRkK,WAAW7M,eACN6M,WAAWxO,gBACfwO,WAAWrD,iBACNqD,WAAWpD,MAAMC,sBACjBmD,WAAWpD,MAAME,eAEvC,GAAG9F,MAAMG,WACT8I,aAAa9M,KAAOgE,SAAShE,KAC7B8M,aAAazO,UAAY2F,SAAS3F,UAClCyO,aAAatD,MAAQxF,SAASwF,MAC9BsD,aAAarD,MAAMC,WAAa1F,SAASyF,MAAMC,WAC/CoD,aAAarD,MAAME,WAAa3F,SAASyF,MAAME,cAEhD1F,MAAMC,sBAAaC,YAE1B4I,WAAWzG,KAAMmF,MACA5I,KAERQ,SAASC,cAAcT,KAAKL,KAAK7E,uBAAuBqP,QAAQ,OAAQvB,KAAKzL,MAAO,CACrFyD,QAASZ,KAAKL,KAAKyK,OACnBzJ,UAAW,WACZK,MAAMqJ,gBACDA,8BACK,CAAC,CACFpJ,WAAY,uCACZC,KAAM,IAAO0H,KAAK9I,OAClB,GAAGkB,MAAMG,cACe,GAApBA,SAASlG,QAAiB,KACtBqP,MAAQ7G,KAAKwE,WAAWsC,QAAQ3B,MACpCnF,KAAKwE,WAAWuC,OAAOF,MAAO,OAGnClJ,MAAMC,sBAAaC,cAG3BF,MAAMC,sBAAaC,YAE1BmJ,aAAaC,MAAOC,OAGhBD,MAAME,MAAMD,WAERE,SAAW,OACV,IAAIC,OAAOH,MACZE,SAASjB,KAAK,IAAOe,MAAMG,KAAKhL,YAAgBgL,qBAE/C,CAAC,CACF7J,WAAY,yCACZC,KAAM,UAAa2J,aACnB,GAAG7J,MAAK,SAETI,MAAMC,sBAAaC,YAE1ByJ,WAAW7D,iBACD3G,KAAOP,KACbO,KAAKC,SAASC,cAAcT,KAAKL,KAAKqL,uBAAuBb,QAAQ,OAAQjD,UAAU/J,MAAO,CAC1FyD,QAASZ,KAAKL,KAAKyK,OACnBzJ,UAAW,WACZK,MAAK,SAASqJ,eACTA,8BACK,CAAC,CACFpJ,WAAY,uCACZC,KAAM,IAAOgG,UAAUpH,GAAIwC,OAAO,MAClC,GAAGtB,MAAMG,WACe,GAApBA,SAASlG,SACTsF,KAAK4H,MAAMjF,MAAM,mBAAoBgE,cAG1C9F,MAAMC,sBAAaC,cAG3BF,MAAMC,sBAAaC,YAE1B2J,gBAAgBP,WACRhC,KAAOgC,MAAMnL,oBACZ,CAAC,CACF0B,WAAY,uCACZC,KAAM,IAAOwH,KAAK5I,OAClB,GAAGkB,MAAMG,WACe,GAApBA,SAASlG,SACTyP,MAAMQ,OAAOhI,MAAM,MAAOwH,UAG/BtJ,MAAMC,sBAAaC,YAG1B6J,SAAS1H,KAAMmF,KAAM0B,MAAOc,SAAUjM,YAG5BkM,YAAuB,YAARlM,KACftC,QAAU4G,KAAK5G,YACjB+G,MAAO,MACN,IAAIyB,EAAI,EAAGA,EAAIxI,QAASwI,OACrBuD,KAAKxB,MAAMkD,MAAQjF,IAAMuD,KAAKxB,MAAMkD,MAAQjF,GAAG5G,QAAS,OAClD6M,KAAO1C,KAAKxB,MAAMkD,MAAQjF,GAAG5G,YAC9B,MAAMsK,MAAMuC,KAAM,OACb5C,KAAO4C,KAAKvC,IACdL,KAAKM,OAASoC,WACVC,YACIhG,EAAI,GAAMqD,KAAK6C,KAAOlG,EAAK,IAC3BzB,MAAO,GAGN8E,KAAK6C,KAAOlG,EAAK,IAClBzB,MAAO,YAQxBA,MAEX4H,aAAaC,UACHC,KAAO1L,KAAK2L,MAAM,iBAAmBF,GAAGzP,OAAS,IACnD0P,MAAQA,KAAK,IACbA,KAAK,GAAGtF,gBAENwF,KAAO5L,KAAK2L,MAAM,iBAAmBF,GAAGzP,OAAS,IACnD4P,MAAQA,KAAK,IACbA,KAAK,GAAGxF,WAGhBC,OAAM,CAACC,KAAM9J,aACIiF,IAATjF,OACAA,KAAO,IAEJ,uBAAQ8J,KAAM9J,OAEzB+J,OAAM,CAACD,KAAM9J,aACIiF,IAATjF,OACAA,KAAO,IAEJ,uBAAQ8J,KAAM,EAAI9J,OAE7BqP,gBAAgBnB,YACPxH,MAAM,gBAAiBwH,QAEhCoB,YAAYrI,WACHnJ,MAAMiJ,MAAMqG,KAAKnG,OAE1BsI,oBAAoBC,mBACVvI,KAAOzD,KAAK1F,MAAMiJ,MAAMyI,kBACzB9I,MAAM,cAAeO,OAE9BwI,cAAcnB,IAAKrH,SACXqH,IAAM,GAA8B,GAAzBrH,KAAKwE,WAAWoB,aACpB,EACJ,KACC6C,IAAM,MACL,IAAI7G,EAAI,EAAGA,EAAIyF,IAAKzF,IACrB6G,KAAOlM,KAAK2I,gBAAgBlF,KAAKwE,WAAW5C,GAAI5B,MAAQ,SAErDyI,MAGfX,KAAK3C,KAAMuD,KAAMnD,WACTuC,KAAO,MACN,MAAMxP,UAAU6M,KAAKxB,MAAM+E,MAAM1N,QAC9B1C,OAAOoQ,MAAQA,MAAQpQ,OAAOiN,OAASA,QACvCuC,KAAOxP,OAAOwP,aAGfA,MAEXa,OAAO1B,MAAO9B,KAAMuD,MAChBxS,MAAMqJ,KAAK,WAAY0H,MAAO9B,KAAMuD,YAC9B5L,KAAOP,QACT0K,MAAMvL,KAAKjG,aACXS,MAAMqJ,KAAK,wBACY,YAAnB0H,MAAMvL,KAAKA,KAAoB,OAEzBkN,SAAWzD,KAAKxB,MAAM+E,MAAM1N,YAC9B6N,UAAY,MACX,MAAMC,OAAOF,SACVE,IAAIvD,OAASsD,YACbA,UAAYC,IAAIvD,MAAQ,kBAI3B,CAAC,CACF/H,WAAY,oCACZC,KAAM,SACS0H,KAAK9I,QACRqM,WACCG,eACD,iBACG,eACU,gBACH,aACD5B,MAAMnL,KAAKO,YACZ,qBACO,UAG3B,GAAGkB,MAAMG,eACLuH,KAAOvH,SACXkL,SAASzC,KAAKlB,MACdnI,KAAK2C,MAAM,QAAS3C,KAAKjG,YAIpBkS,WAAU,KACPxM,KAAK2L,MAAMc,oBACNd,MAAMc,cAAcC,0BAGjCvS,aAAa+I,MAAM,mBAEpB9B,MAAMC,sBAAaC,gBACnB,GAAuB,UAAnBoJ,MAAMvL,KAAKA,KAAkB,CACpCxF,MAAMqJ,KAAK,qCAELqJ,SAAWzD,KAAKxB,MAAM+E,MAAMlD,YAC9BqD,UAAY,MACX,MAAMC,OAAOF,SACVE,IAAIvD,OAASsD,YACbA,UAAYC,IAAIvD,MAAQ,kBAG3B,CAAC,CACF/H,WAAY,oCACZC,KAAM,SACS0H,KAAK9I,QACRqM,UACAzB,MAAMnL,KAAKJ,WACVmN,kBACE,UACK5B,MAAMnL,KAAKpB,MAAQuM,MAAMnL,KAAKpB,MAAM2B,QAAK2B,OAG7D,GAAGT,MAAMG,eACLuH,KAAOvH,SACXkL,SAASzC,KAAKlB,MACdnI,KAAK2C,MAAM,QAAS3C,KAAKjG,UAE1B8G,MAAMC,sBAAaC,aAIlCqL,gBAAgBxN,MACK,YAAbA,KAAKA,UACD,sBAAS,mBAAsBA,KAAKuJ,MAShDkE,gBAAgBzN,MACK,UAAbA,KAAKA,UACD,sBAAS,mBAAsBA,KAAKuJ,OAUpD/F,SACC,8wwBAyWN1J,IAAIC,UAAU,sBAAuB,CAChCgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,KAGfwN,OAAQ,CACJ1N,KAAM0D,eACK,IAGnBtD,KAAI,KACO,CACHuN,aAAc,KAGtBhF,UAGI3N,aAAa4N,IAAI,mBAAoB/H,KAAK+M,qBAE9CnN,SAAU,GAGVS,QAAS,CACL0M,mBAAmBC,WAIXhN,KAAK2L,MAAMsB,MAAQD,QAAUhN,KAAK1F,MAAMwF,GAAI,OACtCoN,MAAQC,SAASC,iBAClB,uCAAsCpN,KAAK1F,MAAMwF,YAGlDuN,UAAY,EAChBH,MAAMI,SAASC,WAGLC,OADID,GAAGE,wBACID,OACjBH,WAAaG,MAAb,UAGEE,YAAe,GAAEL,mBAClB1B,MAAMsB,KAAKU,MAAMH,OAASE,eAI3C/K,SAAW,6aAcf1J,IAAIC,UAAU,mBAAoB,CAC9BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,MAInBE,KAAI,KACO,IAGXK,SAAU,CACNgO,kBAEUxG,MAAQpH,KAAK1F,MAAM8M,SACrB+B,MAAMC,QAAQhC,OAAQ,KAClBiC,MAAQ,MACP,IAAIhE,EAAI,EAAGA,EAAI+B,MAAMlN,OAAQmL,IAC1B8D,MAAMC,QAAQhC,MAAM/B,GAAG5G,WACvB4K,OAASjC,MAAM/B,GAAG5G,QAAQvE,QAE1BiP,MAAMC,QAAQhC,MAAM/B,GAAG4D,WACvBI,OAASjC,MAAM/B,GAAG4D,QAAQ/O,eAGjB,GAATmP,aAED,GAGfwE,SAAQ,KACG,GAGfxN,QAAS,CACLyN,cACS5K,MAAM,OAAQlD,KAAK1F,QAE5ByT,gBACS7K,MAAM,SAAUlD,KAAK1F,SAIlCqI,SAAW,6pCA+Bf1J,IAAIC,UAAU,mBAAoB,CAC9BgG,MAAO,CACHC,KAAM,CACFA,KAAMyD,eACK,YAEfoL,UAAW,CACP7O,KAAM0D,eACK,IAEf+F,KAAM,CACFzJ,KAAMC,OACNC,QAAO,IACI,MAGf2J,MAAO,CACH7J,KAAM0D,QAEVvI,MAAO,CACH6E,KAAMgK,MACN9J,QAAO,IACI,IAGfkK,KAAM,CACFpK,KAAMC,OACNC,QAAO,IACI,MAGfoE,KAAM,CACFtE,KAAMC,OACNC,QAAO,IACI,MAGfrD,OAAQ,CACJmD,KAAMC,OACNC,QAAO,IACI,OAInB2I,gBACUzH,KAAOP,KACI,YAAbO,KAAKpB,MAAwC,GAAlBoB,KAAKyN,YAChCzN,KAAK0N,eAAiB,IAAIC,gBAAe,QACjC3N,KAAKoL,MAAMsB,KAAM,OACXkB,KAAO5N,KAAKoL,MAAMsB,KAAKQ,wBAE7BtT,aAAa+I,MAAM,mBAAoB3C,KAAKqI,KAAK9I,GAAIS,KAAKyI,MAAOmF,KAAKX,YAE3EY,QAAQ7N,KAAKoL,MAAMsB,QAG9BoB,YACQrO,KAAKiO,qBACAA,eAAeK,cAG5B1O,SAAU,CACN2O,gBACY,GAAEvO,KAAKb,UAAUa,KAAK4I,KAAK9I,MAAME,KAAKgO,aAAahO,KAAKgJ,SAEpEwF,cACS,MAAMzF,MAAM/I,KAAK1F,MAAO,IACb0F,KAAK1F,MAAMyO,IACfC,OAAShJ,KAAKgJ,aACXD,UAGR,MAEXL,WACS,MAAMK,MAAM/I,KAAK1F,MAAO,OACnBiS,IAAMvM,KAAK1F,MAAMyO,OACnBwD,IAAIvD,OAAShJ,KAAKgJ,aACXuD,WAGR,MAEXkC,kBACWzO,KAAKb,MAEhBuP,yBACW,CAAC3S,OAAQiE,KAAK2O,MAAMzV,YAE/B0V,iBACQ5O,KAAK0I,MAAQ1I,KAAK0I,KAAK6C,KAAO,EAGtB,gBAED,KAInBhM,KAAI,KACO,CACHI,KAAMnF,QAAQmB,aACdkT,SAAUrU,QAAQC,cAClBwT,eAAgB,KAChBU,MAAO,CACHzV,UAAW,KACXiG,KAAM,MAEV2P,YAAa,CACTC,WAAY,KACZnS,WAAY,cACD,EACXoS,eAAe,EACfC,UAAU,KAItB5O,QAAS,CACLgI,UAAS,KACE,sBAAS,kBAEpB+D,OAAO1B,YACEiE,MAAMzV,UAAY,UAClByV,MAAMxP,KAAO,KAClBxF,MAAMqJ,KAAK0H,aACLnK,KAAOP,QACT0K,MAAMvL,KAAKuJ,KAAM,KACbA,KAAOgC,MAAMnL,KAIjBmJ,KAAKM,MAAQhJ,KAAKgJ,MAClBN,KAAKyD,KAAOnM,KAAKgO,UACjBzN,KAAKjG,MAAMsP,KAAKlB,MAChBnI,KAAK2C,MAAM,QAAS3C,KAAKjG,OAKzBiG,KAAK2O,kBAAkBxG,MAAM1H,MAAK,KAC1BhB,KAAK2L,MAAMc,oBACNd,MAAMc,cAAcC,0BAG9BtL,MAAMC,sBAAaC,gBACfoJ,MAAMvL,KAAKjG,YAClBS,MAAMqJ,KAAK,wBACY,YAAnB0H,MAAMvL,KAAKA,oBACN,CAAC,CACF8B,WAAY,oCACZC,KAAM,SACSX,KAAKqI,KAAK9I,QACbS,KAAKyN,gBACJzN,KAAKyI,WACN,iBACG,eACU,gBACH,aACD0B,MAAMnL,KAAKO,YACZ,qBACO,UAG3B,GAAGkB,MAAMG,eACLuH,KAAOvH,SACXZ,KAAK2O,kBAAkBxG,MAAM1H,MAAK,KAC9BT,KAAKjG,MAAMsP,KAAKlB,MAChBnI,KAAK2C,MAAM,QAAS3C,KAAKjG,YAIpBkS,WAAU,KACPxM,KAAK2L,MAAMc,oBACNd,MAAMc,cAAcC,0BAGjCvS,aAAa+I,MAAM,mBAEpB9B,MAAMC,sBAAaC,cAEvBF,MAAMC,sBAAaC,WACI,UAAnBoJ,MAAMvL,KAAKA,OAClBxF,MAAMqJ,KAAK,8CACN,CAAC,CACF/B,WAAY,oCACZC,KAAM,SACSX,KAAKqI,KAAK9I,QACbS,KAAKyN,eACLtD,MAAMnL,KAAKJ,aACR,UACKuL,MAAMnL,KAAKpB,MAAQuM,MAAMnL,KAAKpB,MAAM2B,QAAK2B,OAG7D,GAAGT,MAAMG,eACLuH,KAAOvH,SACXZ,KAAK2O,kBAAkBxG,MAAM1H,MAAK,KAC9B0H,KAAKM,MAAQhJ,KAAKgJ,MAClBzI,KAAKjG,MAAMsP,KAAKlB,MAChBnI,KAAK2C,MAAM,QAAS3C,KAAKjG,UAE1B8G,MAAMC,sBAAaC,cAEvBF,MAAMC,sBAAaC,cAIlC6N,MAAMzE,aACInK,KAAOP,SACTF,GAAK4K,MAAMnL,KAAKO,OACf,IAAIuF,EAAI,EAAGA,EAAI9E,KAAKjG,MAAMJ,OAAQmL,OAC/B9E,KAAKjG,MAAM+K,GAAGvF,IAAMA,GAAI,CACxBS,KAAKjG,MAAMkQ,OAAOnF,EAAG,GAAIA,eAK5BnC,MAAM,QAASlD,KAAK1F,OACzBH,aAAa+I,MAAM,iBAEvBgM,kBAAkBxG,YACR0G,SAAW,IAAO1G,KAAK5I,SAAaE,KAAKgJ,WAAehJ,KAAKgO,kBAAsBhO,KAAK4I,KAAK9I,WAC5F,cAAK,CAAC,CACTmB,WAAY,yCACZC,KAAM,OAAU,CAACkO,cACjB,GAAGhO,MAAMC,sBAAaC,YAE9B+N,YAAY3E,YACHiE,MAAMzV,UAAYwR,MAAMnL,UACxBoP,MAAMxP,KAAOuL,MAAMvL,MAE5BmQ,mBACSX,MAAMzV,UAAY,UAClByV,MAAMxP,KAAO,MAEtBoQ,iBAEWvP,KAAKyD,KAAK5G,QAAUmD,KAAKgO,UAAY,GAEhDwB,SAAS9G,YACE,CACHA,MAAM,EACNxP,WAAW,EACXqS,KAAM7C,KAAK6C,KACXpM,KAAMa,KAAKb,OAGnBsQ,UAAUtQ,aACFA,KAAKA,MAAQa,KAAKb,UACd,sBAAS,oBAAsBA,KAAKuJ,QAGxB,UAARvJ,MAEOA,KAAKoM,MAAQvL,KAAKuP,cAW7C5M,SAAW,mtGAgEf1J,IAAIC,UAAU,wBAAyB,CACnCgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,QAEVqE,KAAM,CACFtE,KAAMC,QAEVwJ,KAAM,CACFzJ,KAAMC,QAEVpD,OAAQ,CACJmD,KAAMC,QAEVsQ,QAAS,CACLvQ,KAAM0D,QAEV8M,OAAQ,CACJxQ,KAAMqH,iBACK,IAGnB5G,SAAU,CACNgQ,kBACUC,YAAcC,KAAKC,IAAI/P,KAAKyD,KAAK5G,QAASmD,KAAKhE,OAAOA,QAAUgE,KAAK1F,MAAMiR,KAAO,WACjFvL,KAAKyD,KAAKuM,WAAWH,YAAc,IAE9CI,4BACU1P,KAAOP,aACTO,KAAKkD,KAAKyM,WAIV3P,KAAKjG,OAA4B,UAAnBiG,KAAKjG,MAAM6E,MACzBoB,KAAKuO,YAAYC,YAAa,4BAAaxO,KAAKjG,MAAMyB,OAAON,UAAW8E,KAAKjG,MAAMyB,OAAOL,SAC1F6E,KAAKuO,YAAYlS,YAAa,4BAAa2D,KAAKvE,OAAOP,UAAW8E,KAAKqP,UAAUlU,SAC7E6E,KAAKuO,YAAYC,WAAWoB,MAAMC,WAAa7P,KAAKuO,YAAYlS,WAAWuT,MAAMC,WAC9E7P,KAAKuO,YAAYC,WAAWsB,KAAKD,WAAa7P,KAAKuO,YAAYlS,WAAWyT,KAAKD,YAMtFzW,MAAM2W,KAAK,uCAAwC/P,KAAKjG,MAAOiG,KAAKvE,OAAQuE,KAAKmP,UAC1E,MAInBnQ,KAAI,KACO,CAEHO,GAAIgQ,KAAKS,MAAMT,KAAKU,SAAWxK,KAAKyK,OAAOC,SAAS,IACpD/Q,KAAMnF,QAAQmB,aACdmT,YAAa,CACTC,WAAY,KACZnS,WAAY,KACZ+T,QAAS,UACM,EACX3B,eAAe,EACfC,UAAU,MAK1B5O,QAAS,CACLqM,6BACUnM,KAAOP,KAERO,KAAKkD,KAAKyM,SAuBXvW,MAAMqJ,KAAK,wDAAyDzC,KAAKuO,cAtBzEnV,MAAMqJ,KAAK,gCACLzC,KAAK0P,oBAkBPtW,MAAMqJ,KAAK,+BAAgCzC,KAAKuO,cAjBhDnV,MAAMqJ,KAAK,8CAEPzC,KAAKjG,MAAMyB,OAAO6U,gBACbrQ,KAAKoP,QAAWpP,KAAKuO,YAAY6B,QAAQtR,QAGnCkB,KAAKuO,YAAY6B,QAAQE,cAEhCtQ,KAAKuQ,0BAHAtQ,SAASoD,KAAK,4BAA8B5D,KAAKF,IAOrDS,KAAKoP,QAAWpP,KAAKuO,YAAY6B,QAAQ1B,eACrCzO,SAASoD,KAAK,2BAA6B5D,KAAKF,OAUzEgR,2BACUvQ,KAAOP,YAETO,KAAKuO,YAAY6B,QAAQtR,UACzBkB,KAAKuO,YAAY6B,QAAQE,cAAe,IAErC,cAAK,CAAC,CACT5P,WAAY,2CACZC,KAAM,WACeX,KAAKvE,OAAO8D,aACZE,KAAK1F,MAAMyB,OAAO+D,GAC/ByL,KAAMvL,KAAK1F,MAAMiR,SAEzB,GAAGnK,MAAMC,sBAAaC,WAAWN,MAAMG,WACvCZ,KAAKjG,MAAMyB,OAAON,UAAY0F,SAAS1F,UACvC8E,KAAKjG,MAAMyB,OAAOL,QAAUyF,SAASzF,QACrC6E,KAAKjG,MAAMyB,OAAOgV,OAAS5P,SAAS4P,OACpCxQ,KAAK2C,MAAM,QAAS3C,KAAKjG,WAIjC0W,oBAAoBhD,cAChBrU,MAAMqJ,KAAK,kBAAmBhD,KAAK4I,KAAKxB,MAAO4G,UAAWhO,KAAK1F,MAAM0O,OACjEhJ,KAAK4I,KAAKxB,MAAM4G,WAAY,OACtB1C,KAAOtL,KAAK4I,KAAKxB,MAAM4G,WAAW/E,YACnC,MAAMF,MAAMuC,QACTA,KAAKvC,IAAIC,OAAShJ,KAAK1F,MAAM0O,aAC7BrP,MAAMqJ,KAAK,QAASsI,KAAKvC,KAClBuC,KAAKvC,WAIjB,MAEXkI,oBAAoBjD,iBACVhF,MAAQhJ,KAAK1F,MAAM0O,MACnBsC,KAAOtL,KAAK4I,KAAKxB,MAAM4G,WAAW/E,QAClCiI,WAAa,OACd,MAAMnI,MAAMuC,KACb4F,WAAWtH,KAAK0B,KAAKvC,IAAIC,WAEzBmI,QAAUnI,MAAQ,OACfkI,WAAWxP,SAASyP,UACvBA,iBAEGA,SAEXC,oBAAoBpD,cAChBrU,MAAMqJ,KAAK,YAAahD,KAAK4I,KAAKxB,MAAO4G,UAAWhO,KAAK1F,MAAM0O,OAC3DhJ,KAAK4I,KAAKxB,MAAM4G,WAAY,OACtB1C,KAAOtL,KAAK4I,KAAKxB,MAAM4G,WAAWvP,YACnC,MAAMsK,MAAMuC,QACTA,KAAKvC,IAAIC,OAAShJ,KAAK1F,MAAM0O,aAC7BrP,MAAMqJ,KAAK,QAASsI,KAAKvC,KAClBuC,KAAKvC,WAIjB,MAEXsI,oBAAoBrD,iBACVhF,MAAQhJ,KAAK1F,MAAM0O,MACnBsC,KAAOtL,KAAK4I,KAAKxB,MAAM4G,WAAWvP,QAClCyS,WAAa,OACd,MAAMnI,MAAMuC,KACb4F,WAAWtH,KAAK0B,KAAKvC,IAAIC,WAEzBmI,QAAUnI,MAAQ,OACfkI,WAAWxP,SAASyP,UACvBA,iBAEGA,SAEXG,gBAAgB/F,YAEN2B,MAAQ,OACT,IAAI7H,EAAIrF,KAAK1F,MAAM6R,KAAM9G,EAAIrF,KAAK1F,MAAM6R,KAAOZ,KAAMlG,IAAK,OACrDkM,WAAavR,KAAKgR,oBAAoB3L,MACxCkM,WAAY,OACNJ,QAAUnR,KAAKiR,oBAAoB5L,GACzC6H,MAAMtD,KAAK,CACP9J,GAAIyR,WAAWzR,GACfkJ,MAAOmI,gBACInR,KAAK4I,KAAK9I,GACrBqM,KAAMoF,WAAWpF,OAErBoF,WAAWvI,MAAQmI,cAEjBK,WAAaxR,KAAKoR,oBAAoB/L,MACxCmM,YAAcA,WAAW1R,IAAME,KAAK1F,MAAMwF,GAAI,OACxCqR,QAAUnR,KAAKqR,oBAAoBhM,GACzC6H,MAAMtD,KAAK,CACP9J,GAAI0R,WAAW1R,GACfkJ,MAAOmI,gBACInR,KAAK4I,KAAK9I,GACrBqM,KAAMqF,WAAWrF,OAErBqF,WAAWxI,MAAQmI,SAGvBjE,MAAMhT,OAAS,kBACV,CAAC,CACF+G,WAAY,yCACZC,KAAM,CAACgM,MAAOA,UACd,GAAG9L,MAAMC,sBAAaC,YAGlCmQ,WAAWlG,YACDhL,KAAOP,iBACRsR,gBAAgB/F,OACd,cAAK,CAAC,CACTtK,WAAY,yCACZC,KAAM,CACEpB,GAAIS,KAAKjG,MAAMwF,GACfyL,KAAMA,SAEd,GAAGnK,MAAMC,sBAAaC,WAAWN,MAAMG,WACvCZ,KAAKjG,MAAMiR,KAAOpK,SAASoK,KAC3BhL,KAAK2C,MAAM,QAAS3C,KAAKjG,OACzBiG,KAAKiM,WAAU,KACXjM,KAAKmM,8BAKjBgF,eAAeC,SACPnJ,EAAI,UACS,GAAbmJ,IAAIvV,MACJoM,GAAM,KAAIxI,KAAKL,KAAKtD,SACbsV,IAAIvV,MAAQ,IACnBoM,GAAM,GAAEmJ,IAAIvV,SAAS4D,KAAKL,KAAKvD,WAElB,GAAbuV,IAAIrV,MACJkM,GAAM,KAAIxI,KAAKL,KAAKpD,SACboV,IAAIrV,MAAQ,IACnBkM,GAAM,GAAEmJ,IAAIrV,SAAS0D,KAAKL,KAAKrD,WAEnB,GAAZqV,IAAInV,KACJgM,GAAM,KAAIxI,KAAKL,KAAKlD,QACbkV,IAAInV,KAAO,IAClBgM,GAAM,GAAEmJ,IAAInV,QAAQwD,KAAKL,KAAKnD,UAG3BgM,EAAEoJ,sBAKjBjP,SAAW,uvNA8Gf1J,IAAIC,UAAU,SAAU,CACpBgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,IACI,MAGfwS,MAAO,CACH1S,KAAMqH,QACNnH,QAAO,KACI,GAGfkK,KAAM,CACFpK,KAAMC,OACNC,QAAO,IACI,MAGfuJ,KAAM,CACFzJ,KAAMC,OACNC,QAAO,IACI,MAGfoE,KAAM,CACFtE,KAAMC,OACNC,QAAO,IACI,MAGfrD,OAAQ,CACJmD,KAAMC,OACNC,QAAO,IACI,MAGfqQ,QAAS,CACLvQ,KAAM0D,OACNxD,QAAO,IACI,IAInBE,KAAI,KACO,CACHuS,SAAU,KACVC,kBAAmB,KACnBC,YAAY,EACZC,iBAAkB7X,WAAWC,WAC7BsF,KAAMnF,QAAQ8C,SACd4U,aAAa,EACbvH,MAAO,KAGftK,QAAS,CACL8R,UAAUzH,YAEDsH,YAAa,MACdI,MAAQjF,SAASkF,eAAe,aAAerS,KAAK1F,MAAMwF,IAC1DwS,YAAcnF,SAASkF,eAAe,gBAAkBrS,KAAK1F,MAAMwF,IACvEwS,YAAY3E,MAAM4E,SAAW,QAC7BD,YAAY3E,MAAM6E,KAAO9H,MAAM6H,SAASE,EAAI,KAC5CH,YAAY3E,MAAM+E,IAAMhI,MAAM6H,SAASI,EAAI,UACtCb,SAAW,IAAIc,uBAAWR,MAAOE,YAAa,CAC3C3L,MAAO,OACPkM,QAAS,CACLT,MAhiGX,IAiiGWU,IAjiGX,OAqiGD3F,SAAS4F,iBAAiB,YAAa/S,KAAKgT,cAEhDC,UAC0B,OAAlBjT,KAAK8R,eACAA,SAASoB,aAEdZ,YAAcnF,SAASkF,eAAe,gBAAkBrS,KAAK1F,MAAMwF,IACvEwS,YAAY3E,MAAMwF,eAAe,QACjCb,YAAY3E,MAAMwF,eAAe,OACjCb,YAAY3E,MAAMwF,eAAe,YACjChG,SAASiG,oBAAoB,YAAapT,KAAKgT,cAEnDA,YAAa,SAAStI,WACd4H,YAAcnF,SAASkF,eAAe,gBAAkBrS,KAAK1F,MAAMwF,IACvEwS,YAAY3E,MAAM4E,SAAW,QAC7BD,YAAY3E,MAAM6E,KAAO9H,MAAM2I,QAAU,KACzCf,YAAY3E,MAAM+E,IAAMhI,MAAM4I,QAAU,MAG5ClH,OAAO1B,WACC6I,OAAS7I,MAAMnL,KAAKO,GACpB0T,KAAOxT,KAAK1F,MAAMwF,QACjB2T,6BACA,CAAC,CACFxS,WAAY,yCACZC,KAAM,SAAYqS,aAAiBC,SACnC,GAAGxS,MAAMG,eACLuS,KAAO,IAAOvS,SAASrB,WAAeqB,SAASwS,cAAkBxS,SAASyS,OAC9EzZ,aAAa+I,MAAM,oBAAqBwQ,WACnCpZ,MAAMuZ,YAAYC,GAAGlK,KAAK8J,SAEhCtS,MAAMC,sBAAaC,YAE1ByS,WAAWL,YAEDtB,MAAQjF,SAASkF,eAAgB,aAAYqB,KAAKC,WAClDb,IAAM3F,SAASkF,eAAgB,aAAYqB,KAAKE,SAGlD5T,KAAK2K,MAAM+I,KAAKE,cACXjJ,MAAM+I,KAAKE,OAAOV,gBAChBlT,KAAK2K,MAAM+I,KAAKE,QAGb,OAAVxB,OAA0B,OAARU,KAAgBjZ,UAAUuY,QAAUvY,UAAUiZ,YAC3DnI,MAAM+I,KAAKE,OAAS,IAAIhB,uBAAWR,MAAOU,IAAK,CAChDnM,MAZU,iBAaVkM,QAAS,CACLT,MArlGX,IAslGWU,IAtlGX,SA4lGL5I,WAAWwJ,YACDnT,KAAOP,oBACR,CAAC,CACFiB,WAAY,4CACZC,KAAM,SAAYwS,KAAKC,cAAkBD,KAAKE,UAC9C,GAAG5S,MAAMG,cACLA,SAASlG,QAAS,MACb+Y,WAAWN,MAEhBvZ,aAAa+I,MAAM,yBAA0BwQ,UAEzCpJ,MAAQ/J,KAAKjG,MAAMuZ,YAAYI,IAAI1J,QAAQmJ,MAC/CnT,KAAKjG,MAAMuZ,YAAYI,IAAIzJ,OAAOF,MAAO,OAG9ClJ,MAAMC,sBAAaC,YAE1B4S,UAAUR,MACF1T,KAAK2K,MAAM+I,KAAKE,aACXjJ,MAAM+I,KAAKE,OAAOO,UAAU,CAACxN,MAAO,mBAGjDyN,UAAUV,MACF1T,KAAK2K,MAAM+I,KAAKE,aACXjJ,MAAM+I,KAAKE,OAAOO,UAAU,CAACxN,MAAO,oBAGjD0N,4BACS,CAAC,CACFpT,WAAY,oCACZC,KAAM,IACQlB,KAAK1F,MAAMwF,cACHE,KAAK1F,MAAMD,2BACN2F,KAAK1F,MAAMga,oBAEtC,GAAGlT,MAAMC,sBAAaC,YAE9BiT,cAAc7J,OACN1K,KAAKwU,kBACAtC,aAAc,EACnBxH,MAAM+J,mBAGdhB,iBACQzT,KAAK1F,MAAMuZ,aAAe7T,KAAK1F,MAAMuZ,YAAYI,QAC5C,IAAI5O,KAAKrF,KAAK1F,MAAMuZ,YAAYI,IAAK,KAClCP,KAAO1T,KAAK1F,MAAMuZ,YAAYI,IAAI5O,QACjC0O,WAAWL,QAM5BgB,oBAAoBhB,MACZA,KAAKC,SAAW3T,KAAK1F,MAAMwF,UACtBxF,MAAMuZ,YAAYI,IAAIrK,KAAK8J,WAC3BK,WAAWL,QAIxBiB,oBAAoBjB,SACZ1T,KAAK1F,MAAMuZ,aAAe7T,KAAK1F,MAAMuZ,YAAYI,QAC5C,IAAI5O,KAAKrF,KAAK1F,MAAMuZ,YAAYC,GAAI,KACjCc,IAAM5U,KAAK1F,MAAMuZ,YAAYC,GAAGzO,GAChCqO,KAAK5T,IAAM8U,IAAI9U,IACfS,KAAKjG,MAAMuZ,YAAYI,IAAIzJ,OAAOnF,EAAG,KAOrDwP,sBACSpB,eAITqB,gBAAgBC,SACR/U,KAAK1F,MAAMuZ,aAAe7T,KAAK1F,MAAMuZ,YAAYI,QAC5C,IAAI5O,KAAKrF,KAAK1F,MAAMuZ,YAAYI,IAAK,KAClCP,KAAO1T,KAAK1F,MAAMuZ,YAAYI,IAAI5O,GAClCqO,KAAKE,OAASmB,WACTf,WAAWN,QAQhCsB,cAAcC,cACJ1U,KAAOP,QACTA,KAAK1F,MAAMuZ,aAAe7T,KAAK1F,MAAMuZ,YAAYI,QAC5C,MAAM5O,KAAKrF,KAAK1F,MAAMuZ,YAAYI,IAAK,KACpCP,KAAO1T,KAAK1F,MAAMuZ,YAAYI,IAAI5O,GAClCqO,KAAKE,OAASqB,SACd1U,KAAKyT,WAAWN,MAChBnT,KAAKjG,MAAMuZ,YAAYI,IAAIzJ,OAAOnF,EAAG,OAI7CrF,KAAK1F,MAAMuZ,aAAe7T,KAAK1F,MAAMuZ,YAAYC,OAC5C,MAAMzO,KAAKrF,KAAK1F,MAAMuZ,YAAYC,GAAI,CAC5B9T,KAAK1F,MAAMuZ,YAAYC,GAAGzO,GAC5BsO,SAAWsB,QAChB1U,KAAKjG,MAAMuZ,YAAYI,IAAIzJ,OAAOnF,EAAG,KAMrD6P,qBACSzB,eAGTO,WAAWN,MACH1T,KAAK2K,MAAM+I,KAAKE,cACXjJ,MAAM+I,KAAKE,OAAOV,gBAChBlT,KAAK2K,MAAM+I,KAAKE,SAG/BuB,mBACU5U,KAAOP,KACPoV,UAAY,CACd1M,KAAM1I,KAAKL,KAAK,QAAUK,KAAK1F,MAAM6E,MAAMyS,oBAC3CzU,KAA0B,UAAnB6C,KAAK1F,MAAM6E,KAAoBa,KAAK1F,MAAMyB,OAAOsZ,YAAc,GACtEzM,KAAO5I,KAAK4I,KAAQ5I,KAAK4I,KAAKzL,KAAO,GACrCnB,OAASgE,KAAKhE,OAAUgE,KAAKhE,OAAOT,SAAWyE,KAAKuJ,KAAKpM,WAGxDqD,SAASC,eAAc,2BAAUT,KAAKL,KAAK2V,oBAAqBF,WAAY,CAC7EzU,UAAW,SACXC,QAASZ,KAAKL,KAAKpC,GACnBuD,YAAad,KAAKL,KAAKlC,SACxBuD,MAAK1G,QACAA,sBAEK,CAAC,CACF2G,WAAY,uCACZC,KAAM,IACIX,KAAKjG,MAAMwF,OAErB,GAAGkB,MAAMG,WACe,GAApBA,SAASlG,SACTsF,KAAK2C,MAAM,UAAW,CAAC3D,KAAMgB,KAAKjG,WAGvC8G,MAAMC,sBAAaC,cAG3BF,OAAMmU,MACL5b,MAAM6b,QAAQta,MAAMqa,UAIhC3V,SAAU,CACN6V,2BACa,CAAC,UAAU/T,SAAS1B,KAAK1F,MAAM6E,OAE5CuW,0BACa,CAAC,SAAShU,SAAS1B,KAAK1F,MAAM6E,OAE3CqV,mBACW,CAAC,QAAS,WAAY,UAAU9S,SAAS1B,KAAK1F,MAAM6E,QAGnE2I,UAIS9H,KAAK6R,QAGN1X,aAAa4N,IAAI,oBAAqB/H,KAAK0U,qBAE3Cva,aAAa4N,IAAI,oBAAqB/H,KAAK2U,qBAG3Cxa,aAAa4N,IAAI,eAAgB/H,KAAK6U,gBAGtC1a,aAAa4N,IAAI,gBAAiB/H,KAAK8U,iBAGvC3a,aAAa4N,IAAI,cAAe/H,KAAKgV,eACrC7a,aAAa4N,IAAI,cAAe/H,KAAKkV,iBAG7ClN,UAKShI,KAAK6R,aACD4B,cACLkC,YAAW,KACPxb,aAAa+I,MAAM,eAAgBlD,KAAK1F,MAAMwF,MAC/C,MAGX8V,oBACS5V,KAAK6R,MAAO,KACR,IAAIxM,KAAKrF,KAAK1F,MAAMuZ,YAAYI,IAAK,KAClCP,KAAO1T,KAAK1F,MAAMuZ,YAAYI,IAAI5O,QACjC2O,WAAWN,MAEpBvZ,aAAa+I,MAAM,gBAAiBlD,KAAK1F,MAAMwF,IAG/C3F,aAAa0b,KAAK,oBAAqB7V,KAAK0U,qBAC5Cva,aAAa0b,KAAK,oBAAqB7V,KAAK2U,qBAC5Cxa,aAAa0b,KAAK,eAAgB7V,KAAK6U,gBACvC1a,aAAa0b,KAAK,gBAAiB7V,KAAK8U,iBACxC3a,aAAa0b,KAAK,cAAe7V,KAAKgV,eACtC7a,aAAa0b,KAAK,cAAe7V,KAAKkV,iBAG9C9M,UACSpI,KAAK6R,YACD4B,eAGb9Q,SAAW,2mJAqGf1J,IAAIC,UAAU,iBAAkB,CAC5BgG,MAAO,OACM,CACLC,KAAMC,OACNC,QAAO,IACI,OAInBE,KAAI,KACO,CACHI,KAAMnF,QAAQmD,UAGtB0C,QAAS,GAETsC,SAAW,qvBAqBf1J,IAAIC,UAAU,gBAAiB,CAC3BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,IACI,MAGfkK,KAAM,CACFpK,KAAMC,OACNC,QAAO,IACI,MAGfuJ,KAAM,CACFzJ,KAAMC,OACNC,QAAO,IACI,MAGfoE,KAAM,CACFtE,KAAMC,OACNC,QAAO,IACI,MAGfrD,OAAQ,CACJmD,KAAMC,OACNC,QAAO,IACI,MAGfqQ,QAAS,CACLvQ,KAAM0D,OACNxD,QAAO,IACI,IAInBE,KAAI,KACO,CACH0S,iBAAkB7X,WAAWC,WAC7BsF,KAAMnF,QAAQgD,iBAGtBoC,SAAU,CACNkW,6BACQ9V,KAAKuJ,OAAQvJ,KAAKuJ,KAAKwM,uBAAqEtU,IAAjDzB,KAAKuJ,KAAKwM,iBAAiBD,oBAC/D9V,KAAKuJ,KAAKwM,iBAAiBD,mBAM1CE,4BACQhW,KAAKiW,aAAejW,KAAKkW,kBAAoBlW,KAAKmW,kBAC3C,kBAEA,sBAIfC,2BACQpW,KAAKiW,aAAejW,KAAKkW,kBAAoBlW,KAAKmW,kBAC3C,QAEA,sBAGf1a,mBACW,0BAAWuE,KAAK1F,MAAMyB,OAAON,YAExCC,iBACQsE,KAAK1F,MAAMyB,OAAOL,QAAU,GACrB,0BAAWsE,KAAK1F,MAAMyB,OAAOL,SAE7BsE,KAAKL,KAAKjC,WAGzB2Y,QAAO,IACIC,gBAAOD,SAItBhW,QAAS,CACL4V,eACQjW,KAAK1F,MAAMyB,OAAOwa,QAAUvW,KAAK1F,MAAMyB,OAAOwa,OAAOrc,OAAS,MACzD,MAAMsc,KAAKxW,KAAK1F,MAAMyB,OAAOwa,UAC1BC,EAAEtZ,gBACK,SAIZ,GAEXgZ,oBACQlW,KAAK1F,MAAMyB,OAAO6B,YAAcoC,KAAK1F,MAAMyB,OAAO6B,WAAWvD,eACxD,MAAMoc,UAAUzW,KAAK1F,MAAMyB,OAAO6B,WAAWvD,cAC1Coc,OAAOvJ,OAASuJ,OAAOvJ,MAAMhT,OAAS,SAC/B,SAIZ,GAEXic,2BACQnW,KAAK1F,MAAMyB,OAAO8B,aAAcmC,KAAK1F,MAAMyB,OAAO8B,WAAW6Y,eACrD1W,KAAK1F,MAAMyB,OAAO8B,WAAW6Y,aAAaxc,OAAS,GAInEyc,eAAeC,SAAUJ,kBAChB,CAAC,CACFvV,WAAY,oCACZC,KAAM,UACcsV,EAAE1W,WACHE,KAAK1F,MAAMwF,WACX8W,kBACCJ,EAAEzY,aAEtB,GAAGqD,MAAMC,sBAAaC,YAE9BuV,gBAAgBD,SAAUJ,kBACjB,CAAC,CACFvV,WAAY,oCACZC,KAAM,UACcsV,EAAE1W,WACHE,KAAK1F,MAAMwF,WACX0W,EAAEtZ,kBACD0Z,aAEpB,GAAGxV,MAAMC,sBAAaC,YAE9BwV,kCACS,CAAC,CACF7V,WAAY,oCACZC,KAAM,IACQlB,KAAK1F,MAAMwF,cACHE,KAAK1F,MAAMD,eAEjC,GAAG+G,MAAMC,sBAAaC,aAGlCqB,SAAW,u+JAgGf1J,IAAIC,UAAU,uBAAwB,CAClCgG,MAAO,OACM,CACLC,KAAMC,OACNC,QAAO,IACI,WAGP,CACJF,KAAMC,OACNC,QAAO,IACI,OAInBE,KAAI,KACO,CACH0S,iBAAkB7X,WAAWC,WAC7BsF,KAAMnF,QAAQgD,iBAGtBoC,SAAU,CACN0H,6BACQtH,KAAKuJ,OAAQvJ,KAAKuJ,KAAKwM,uBAAqEtU,IAAjDzB,KAAKuJ,KAAKwM,iBAAiBzO,oBAC/DtH,KAAKuJ,KAAKwM,iBAAiBzO,mBAK1CyP,qBACQzL,KAAO,OACN,IAAIvC,MAAM/I,KAAK1F,MAAMyB,OAAOwa,OAAQ,KACjCC,EAAIxW,KAAK1F,MAAMyB,OAAOwa,OAAOxN,IAC7ByN,EAAEtZ,UACFoO,KAAK1B,KAAK4M,UAGXlL,OAGfjL,QAAS,CACLsW,eAAeC,SAAUJ,kBAChB,CAAC,CACFvV,WAAY,oCACZC,KAAM,UACcsV,EAAE1W,WACHE,KAAK1F,MAAMwF,WACX8W,kBACCJ,EAAEzY,aAEtB,GAAGqD,MAAMC,sBAAaC,YAE9BuV,gBAAgBD,SAAUJ,kBACjB,CAAC,CACFvV,WAAY,oCACZC,KAAM,UACcsV,EAAE1W,WACHE,KAAK1F,MAAMwF,WACX0W,EAAEtZ,kBACD0Z,aAEpB,GAAGxV,MAAMC,sBAAaC,aAGlCqB,SAAW,inDA+Bf1J,IAAIC,UAAU,2BAA4B,CACtCgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,KAGf2X,UAAW,CACP7X,KAAMqH,iBACK,GAEfzK,OAAQ,CACJoD,KAAMC,OACNC,QAAO,KACI,MAInBE,KAAI,KACO,CACHI,KAAMnF,QAAQoD,aAGtBgC,SAAU,CACNsW,oBACQlW,KAAK1F,MAAMD,eACN,MAAMoc,UAAUzW,KAAK1F,MAAMD,cACxBoc,OAAOvJ,OAASuJ,OAAOvJ,MAAMhT,OAAS,SAC/B,SAIZ,GAEXmc,QAAO,IACIC,gBAAOD,SAGtBhW,QAAS,CACL4W,eAAerZ,mBACHA,gBACC,iBACM,yBACN,eAEA,sBACM,mBACN,sBACM,6BAEA,aAGnBsZ,cAAcT,QACHA,OAAO7Y,WAAa,YAAc,cAGjD+E,SAAW,u7CA+Bf1J,IAAIC,UAAU,2BAA4B,CACtCgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,KAGf2X,UAAW,CACP7X,KAAMqH,iBACK,GAEfkC,KAAM,CACFvJ,KAAMC,OACNC,QAAO,KACI,CAACS,GAAI,SAIxBP,KAAI,KACO,CACHI,KAAMnF,QAAQqD,aAGtB+B,SAAU,CACNsW,oBACQlW,KAAK1F,MAAMD,eACN,MAAMoc,UAAUzW,KAAK1F,MAAMD,cACxBoc,OAAOvJ,OAASuJ,OAAOvJ,MAAMhT,OAAS,SAC/B,SAIZ,GAEXmc,QAAO,IACIC,gBAAOD,SAGtBhW,QAAS,CACL8W,SAAStZ,kBACC4G,KAAO5G,WAAW4G,SACpB+D,EAAI,OACH,MAAMO,MAAMtE,KAAM,OACb2S,EAAI3S,KAAKsE,QAIXsO,IAHAtO,GAAK,IACLP,GAAK,OAIL6O,IADU,cAAVD,EAAEjY,KACK,gDAA+CiY,EAAEtX,KAEjD,yDAAwDsX,EAAEtX,oBAAoBsX,EAAEE,YAG3F9O,GAAM,YAAW6O,QAAQD,EAAExb,mBAExB4M,GAEXqO,gBAAgBD,SAAUpS,kBACjB,CAAC,CACFvD,WAAY,yCACZC,KAAM,eACmBsD,EAAE1E,WACRE,KAAK0I,KAAK5I,YACT8W,aAEpB,GAAGxV,MAAMC,sBAAaC,aAGlCqB,SAAW,6rHAoEf1J,IAAIC,UAAU,kBAAmB,CAC7BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,MAInBE,KAAI,KACO,CACH0S,iBAAkB7X,WAAWC,aAGrCgG,QAAS,GAGTsC,SAAW,uJAOf1J,IAAIC,UAAU,gBAAiB,CAC3BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,MAInBE,KAAI,KACO,IAGXc,QAAS,GAETsC,SAAW,mJAOf1J,IAAIC,UAAU,eAAgB,CAC1BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,MAInBE,KAAI,KACO,IAEXc,QAAS,GAETsC,SAAW,kJAOf1J,IAAIC,UAAU,eAAgB,CAC1BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,CACHlB,MAAO,OAKvBoB,KAAI,KACO,CACHgY,IAAK/c,QACLmF,KAAMnF,QAAQ8C,WAGtB+C,QAAS,GAETsC,SAAW,g1FAuDf1J,IAAIC,UAAU,mBAAoB,CAC9BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMgK,MACN9J,QAAO,KACI,MAInBE,KAAI,KACO,IAGXc,QAAS,GAETsC,SAAW,yRAUf1J,IAAIC,UAAU,wBAAyB,CACnCgG,MAAO,CACH5E,MAAO,CACH6E,KAAMC,OACNC,QAAO,KACI,MAInBE,KAAI,KACO,CACH2E,SAAS,IAIjBtE,SAAU,CACN4X,qBACWxX,KAAKyX,eAEhBC,oBACY1X,KAAK1F,MAAMqd,aAAe3X,KAAK1F,MAAMsd,aAIrDvX,QAAS,CACLoX,qBACYzX,KAAK1F,MAAMqd,cAAiB3X,KAAK1F,MAAMud,UAC1C7X,KAAK1F,MAAMsd,aAAgB5X,KAAK1F,MAAMmE,SAE/CqZ,sBACUvX,KAAOP,KACTA,KAAKyX,8BACA,CAAC,CACFxW,WAAY,mCACZC,KAAM,IACIlB,KAAK1F,MAAMwF,OAErB,GAAGkB,MAAMG,WACTZ,KAAK2C,MAAM,QAAS/B,aAErBC,MAAMC,sBAAaC,aAIlCqB,SAAW,k4CA2Bf1J,IAAIC,UAAU,gBAAiB,CAC3BgG,MAAO,CACH5E,MAAO,CACH6E,KAAMgK,MACN9J,QAAO,KACI,MAInBE,KAAI,KACO,IAGXc,QAAS,CACLmP,SAAQ,KACG,CACH9G,MAAM,EACNxP,WAAW,EACXqS,KAAM,EACNpM,KAAM,cAIlBwD,SAAW,2nBAkBf1J,IAAIC,UAAU,YAAa,CACvBgG,MAAO,CACH5E,MAAO,CACH6E,KAAMqH,iBACK,GAEfuR,WAAY,CACR5Y,KAAMC,OACNC,QAAO,IACI,MAGf6I,SAAU,CACN/I,KAAMqH,iBACK,GAEfwR,YAAa,CACT7Y,KAAM0D,eACK,IAGnBtD,KAAI,KACO,CACH0Y,eAAgB,sBAAS,eACzBtY,KAAMnF,QAAQ+D,QACd2Z,cAAe,GACfC,aAAc,GACd1Z,QAAS,GACTwK,QAAS,CACLxK,QAAS,GACT0Z,aAAc,GACdD,cAAe,IAEnBE,gBAAgB,EAChBC,kBAAmB,GACnBC,eAAgB,CACZJ,eAAe,EACfC,cAAc,KAI1BI,MAAO,CAEHR,kBACOS,wBAGXxQ,gBACUzH,KAAOP,UACRyY,kBAEAtQ,MAAMJ,IAAI,uBAAuB,CAAC2Q,WAAYC,eAC/CpY,KAAK+X,eAAeI,cAAgBC,WAApC,KAGR/Y,SAAU,CACNgZ,oBAAmB,KACR,CACHlQ,MAAM,EACNxP,WAAW,EACXqS,KAAM,EACNpM,KAAM,WAGd0Z,wBACUtY,KAAOP,YACTO,KAAK0I,QAAQxK,QACN8B,KAAKuY,iBAAiBvY,KAAK9B,SAE3B8B,KAAK9B,UAIxB4B,QAAS,CACLgI,UAAS,KACE,sBAAS,kBAEpByQ,iBAAiBC,eACPxY,KAAOP,KACPsL,KAAO,GACPjO,OAAS,IAAI2b,OAAQ,MAAKzY,KAAK0I,QAAQxK,aAAc,UACtD,MAAMwa,OAAOF,QAAS,OACjBG,MAAQ9Z,OAAO0K,OAAO,GAAImP,QAChCC,MAAMza,QAAU,GAEZwa,IAAIxa,YACC,MAAM1C,UAAUkd,IAAIxa,SACjBpB,OAAO8b,KAAKpd,OAAOP,YAAc6B,OAAO8b,KAAKpd,OAAOR,YACpD2d,MAAMza,QAAQmL,KAAK7N,aAGpBkd,IAAIrB,aAAgBrX,KAAK8X,kBAAkB3W,SAASuX,IAAInZ,MAC/DS,KAAK8X,kBAAkBzO,KAAKqP,IAAInZ,IAChCnG,MAAMqJ,KAAM,yBAAwBiW,IAAIG,SAASjc,OAAQ8b,oBACpD,CAAC,CACFhY,WAAY,mCACZC,KAAM,IACI+X,IAAInZ,OAEd,GAAGkB,MAAMG,WAETZ,KAAK8Y,KAAKJ,IAAK,WAAY9X,SAAS0W,UACpCtX,KAAK8Y,KAAKJ,IAAK,UAAW9X,SAAS1C,SACnC8B,KAAK8X,kBAAkB7N,OAAOjK,KAAK8X,kBAAkB9N,QAAQ0O,IAAInZ,IAAK,MAEvEsB,MAAMC,sBAAaC,YAGtB2X,IAAIpB,SACJqB,MAAMrB,SAAWtX,KAAKuY,iBAAiBG,IAAIpB,UACpCoB,IAAItB,cAAiBpX,KAAK8X,kBAAkB3W,SAASuX,IAAInZ,MAChES,KAAK8X,kBAAkBzO,KAAKqP,IAAInZ,IAChCnG,MAAMqJ,KAAM,yBAAwBiW,IAAIG,SAASjc,OAAQ8b,oBACpD,CAAC,CACFhY,WAAY,mCACZC,KAAM,IACI+X,IAAInZ,OAEd,GAAGkB,MAAMG,WAETZ,KAAK8Y,KAAKJ,IAAK,WAAY9X,SAAS0W,UACpCtX,KAAK8X,kBAAkB7N,OAAOjK,KAAK8X,kBAAkB9N,QAAQ0O,IAAInZ,IAAK,MAEvEsB,MAAMC,sBAAaC,aAGrB4X,MAAMrB,UAAYqB,MAAMrB,SAAS3d,QAAWgf,MAAMza,QAAQvE,SAC3DoR,KAAK1B,KAAKsP,cAIX5N,MAGXmN,mBACUlY,KAAOP,KACbO,KAAK6X,gBAAiB,iBACjB,CAAC,CACFnX,WAAY,qCACZC,KAAM,cACcX,KAAKyX,gBAEzB,GAAGhX,MAAMG,WACTZ,KAAK9B,QAAU0C,SACfZ,KAAK6X,gBAAiB,KAEvBhX,MAAMC,sBAAaC,gBACjBgY,0BACAd,uBAETc,2BACU/Y,KAAOP,oBACR,CAAC,CACFiB,WAAY,oCACZC,KAAM,CACF7D,OAAQ2C,KAAKiJ,QAAQkP,cAAgB,OAEzC,GAAGnX,MAAMG,WACTZ,KAAK4X,aAAehX,YAErBC,MAAMC,sBAAaC,YAE1BkX,4BACUjY,KAAOP,KACTA,KAAK+X,2BACA,CAAC,CACF9W,WAAY,4CACZC,KAAM,SACSlB,KAAK+X,WAAWjY,GAC3BzC,OAAQ2C,KAAKiJ,QAAQiP,eAAiB,OAE1C,GAAGlX,MAAMG,WACTZ,KAAK2X,cAAgB/W,YAEtBC,MAAMC,sBAAaC,YAG9BiY,yBACStQ,QAAQkP,aAAe,QACvBmB,sBAETE,0BACSvQ,QAAQiP,cAAgB,QACxBM,wBAGb7V,SAAW"}