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/src/treestudyplan-components.js

622 lines
23 KiB
JavaScript
Raw Normal View History

2024-06-03 23:24:16 +02:00
/* eslint no-var: "error"*/
/* eslint no-console: "off"*/
/* eslint camelcase: "off" */
/* eslint-disable no-trailing-spaces */
/* eslint-env es6*/
// Put this file in path/to/plugin/amd/src
2024-06-03 23:24:16 +02:00
import {loadStrings} from './util/string-helper';
import {formatDate, studyplanDates, studyplanTiming} from './util/date-helper';
2024-03-10 15:56:35 +01:00
import FitTextVue from './util/fittext-vue';
2024-08-08 19:27:51 +02:00
import {settings} from "./util/settings";
export default {
2024-06-03 23:24:16 +02:00
install(Vue /* ,options */) {
2024-03-10 15:56:35 +01:00
Vue.use(FitTextVue);
2024-06-03 23:24:16 +02:00
let strings = loadStrings({
studyplancard: {
open: "open",
noenddate: "noenddate",
idnumber: "studyplan_idnumber",
description: "studyplan_description",
completed: "completed",
details: "studyplan_details",
2024-03-10 15:56:35 +01:00
suspended: "suspended",
},
details: {
details: "studyplan_details",
},
extrafields: {
show: "show@core"
},
prevnext: {
prev: "prev@core",
previous: "previous@core",
next: "next@core",
select: "selectanoptions@core",
}
});
2023-07-28 00:04:24 +02:00
// Create new eventbus for interaction between item components
const ItemEventBus = new Vue();
Vue.component('s-studyplan-card', {
props: {
value: {
type: Object,
},
open: {
type: Boolean
},
2024-03-10 15:56:35 +01:00
ignoresuspend: {
type: Boolean,
2024-06-03 23:24:16 +02:00
'default': false,
2024-03-10 15:56:35 +01:00
},
},
data() {
return {
text: strings.studyplancard
};
},
computed: {
timeless() {
const plan = this.value;
if (!plan.pages || plan.pages.length == 0 || plan.pages[0].timeless) {
return true;
} else {
return false;
}
},
2024-06-03 23:24:16 +02:00
timing() {
2023-11-13 22:18:28 +01:00
return studyplanTiming(this.value);
},
2024-06-03 23:24:16 +02:00
dates() {
const dates = studyplanDates(this.value);
return {
2024-06-03 23:24:16 +02:00
start: formatDate(dates.start),
end: (dates.end) ? formatDate(dates.end) : this.text.noenddate,
};
},
2024-03-10 15:56:35 +01:00
suspended() {
return (this.value.suspended && !this.ignoresuspend);
}
},
methods: {
onOpenClick(e) {
2024-06-03 23:24:16 +02:00
this.$emit('open', e);
}
},
template: `
2023-09-08 12:47:29 +02:00
<b-card
2024-03-10 15:56:35 +01:00
:class="'s-studyplan-card timing-' + timing + (suspended?' s-suspended':'')"
>
<template #header></template>
<div class='s-studyplan-card-content'>
<div class='s-studyplan-card-icon'><img :src='value.icon'></div>
<div class='s-studyplan-card-info'>
<div class='s-studyplan-card-titlebar'>
<b-card-title>
2024-03-10 15:56:35 +01:00
<a class='title' v-if='open && !suspended'
href='#' @click.prevent='onOpenClick($event)'>{{value.name}}</a>
<template v-else>{{value.name}}</template>
2024-03-10 15:56:35 +01:00
<div v-if="suspended" class='text-danger'
><fittext maxsize="12pt">{{text.suspended}}</fittext></div>
</b-card-title>
<div class='s-studyplan-card-titleslot'><slot name='title'></slot></div>
</div>
<div class='s-studyplan-card-idnumber' v-if='value.idnumber'>
{{ text.idnumber }}: {{ value.idnumber }}
</div>
<s-progress-bar
2024-03-10 15:56:35 +01:00
v-if='!suspended && value.progress !== undefined && value.progress !== null'
v-model="value.progress"
></s-progress-bar>
</div>
</div>
<slot></slot>
<template #footer>
<span v-if="!timeless" :class="'t-timing-'+timing" v-html="dates.start + ' - '+ dates.end"></span>
<span class="s-studyplan-card-buttons">
<slot name='footer'></slot>
<s-studyplan-details
v-model="value"
v-if="value.description"
><i class='fa fa-info-circle'></i></s-studyplan-details>
2024-03-10 15:56:35 +01:00
<b-button style="float:right;" v-if='open && !suspended' variant='primary'
@click.prevent='onOpenClick($event)'>{{ text.open }}</b-button>
</span>
</template>
</b-card>
`,
});
2023-07-28 00:04:24 +02:00
Vue.component('s-progress-bar', {
props: {
value: {
type: Number,
},
min: {
type: Number,
2024-06-03 23:24:16 +02:00
default() {
return 0;
}
},
max: {
type: Number,
2024-06-03 23:24:16 +02:00
default() {
return 1;
}
}
},
data() {
return {
text: strings.studyplancard
};
},
computed: {
width_completed() {
2024-06-03 23:24:16 +02:00
if (this.value) {
const v = ((this.value - this.min) / (this.max - this.min));
return v * 100;
} else {
return 0;
}
},
width_incomplete() {
2024-06-03 23:24:16 +02:00
if (this.value) {
const v = ((this.value - this.min) / (this.max - this.min));
return (1 - v) * 100;
} else {
return 100;
}
},
percentage_complete() {
2024-06-03 23:24:16 +02:00
if (this.value) {
const v = ((this.value - this.min) / (this.max - this.min));
return Math.round(v * 100) + "%";
} else {
return "0%";
}
}
},
template: `
<div class='s-studyplan-card-progress' >
<div class="s-studyplan-card-progressbar"
><span v-if="width_completed > 0"
:style="{width: width_completed+'%'}"
class='s-studyplan-card-progress-segment s-studyplan-card-progress-completed'
></span
><span :style="{width: width_incomplete+'%'}"
class='s-studyplan-card-progress-segment s-studyplan-card-progress-incomplete'
></span
></div>
<div class="s-studyplan-card-progresstext">
{{ percentage_complete}} {{ text.completed.toLowerCase() }}
</div>
</div>
`,
});
2023-07-28 00:04:24 +02:00
/*
* S-STUDYLINE-HEADER-HEADING
* The only reasing this is not a simple empty div, is the fact that the header height
* needs to match that of the period headers
*/
Vue.component('s-studyline-header-heading', {
props: {
identifier: {
type: Number, // Page reference.
2024-06-03 23:24:16 +02:00
default() {
return 0;
}
}
2023-07-28 00:04:24 +02:00
},
data() {
return {
layerHeights: {}
};
},
created() {
// Listener for the signal that a new connection was made and needs to be drawn
// Sent by the incoming item - By convention, outgoing items are responsible for drawing the lines
ItemEventBus.$on('headerHeightChange', this.onHeaderHeightChange);
},
computed: {
2023-09-08 12:47:29 +02:00
2023-07-28 00:04:24 +02:00
},
methods: {
2024-06-03 23:24:16 +02:00
onHeaderHeightChange(newheight, identifier) {
if (this.identifier == identifier) {
2024-06-03 23:24:16 +02:00
if (this.$refs.main) {
this.$refs.main.style.height = `${newheight}px`;
}
2023-08-08 22:47:36 +02:00
}
2023-07-28 00:04:24 +02:00
}
},
template: `
2024-02-23 09:20:10 +01:00
<div class="s-studyline-header-heading" ref="main" :data-id="identifier"><slot></slot></div>
2023-07-28 00:04:24 +02:00
`,
});
Vue.component('s-studyline-header-period', {
props: {
value: {
2024-06-03 23:24:16 +02:00
type: Object, // Dict with layer as index
2023-07-28 00:04:24 +02:00
},
identifier: {
type: Number, // Page reference.
2024-06-03 23:24:16 +02:00
default() {
return 0;
}
2024-04-21 23:08:03 +02:00
},
mode: {
type: String,
2024-06-03 23:24:16 +02:00
'default': "view",
}
2023-07-28 00:04:24 +02:00
},
mounted() {
2024-06-03 23:24:16 +02:00
const self = this;
if (self.value.period == 1) {
2023-07-28 00:04:24 +02:00
self.resizeListener = new ResizeObserver(() => {
2024-06-03 23:24:16 +02:00
if (self.$refs.main) {
2023-07-28 00:04:24 +02:00
const size = self.$refs.main.getBoundingClientRect();
ItemEventBus.$emit('headerHeightChange', size.height, self.identifier);
2023-07-28 00:04:24 +02:00
}
}).observe(self.$refs.main);
}
},
unmounted() {
2024-06-03 23:24:16 +02:00
if (this.resizeListener) {
2023-07-28 00:04:24 +02:00
this.resizeListener.disconnect();
}
},
computed: {
2024-06-03 23:24:16 +02:00
startdate() {
return formatDate(this.value.startdate);
},
2024-06-03 23:24:16 +02:00
enddate() {
return formatDate(this.value.enddate);
2023-08-03 18:44:57 +02:00
},
2024-06-03 23:24:16 +02:00
current() {
if (this.value && this.value.startdate && this.value.enddate) {
2023-08-03 18:44:57 +02:00
const now = new Date();
const pstart = new Date(this.value.startdate);
const pend = new Date(this.value.enddate);
return (now >= pstart && now < pend);
2024-06-03 23:24:16 +02:00
} else {
2023-08-03 18:44:57 +02:00
return false;
}
}
2023-08-03 18:44:57 +02:00
2023-07-28 00:04:24 +02:00
},
data() {
return {
};
},
template: `
<div :class="'s-studyline-header-period '" ref="main" :data-id="identifier"
><p><abbr :id="'s-period-'+value.id" :title="value.fullname">{{ value.shortname }}</abbr>
<b-tooltip
:target="'s-period-'+value.id" triggers="hover"
>{{ value.fullname }}<br>
<span class="s-studyline-header-period-datespan">
<span class="date">{{ startdate }}</span> - <span class="date">{{ enddate }}</span>
</span>
</b-tooltip>
<slot></slot
><p v-if="value.startdate > 0"
class="s-studyline-header-period-datespan small">
2024-05-10 15:22:52 +02:00
<span class="date">{{ startdate }}</span>
- <span class="date" v-if="this.value.enddate > 0">{{ enddate }}</span><span class="date" v-else>&infin;</span>
</p>
2024-04-21 23:08:03 +02:00
<div v-if="current && mode == 'view'" class='s-studyline-period-highlight'></div>
2023-07-28 00:04:24 +02:00
</div>
`,
});
Vue.component('s-studyplan-details', {
props: {
value: {
type: Object,
},
variant: {
type: String,
2024-06-03 23:24:16 +02:00
default() {
return "info";
}
},
pill: {
type: Boolean,
2024-06-03 23:24:16 +02:00
default() {
return false;
}
},
size: {
type: String,
2024-06-03 23:24:16 +02:00
default() {
return "";
}
}
},
data() {
return {
text: strings.details,
};
},
template: `
<span>
<b-button
:size="size"
:pill="pill"
:variant="variant"
v-b-modal="'modal-description-'+value.id"
><slot><i class='fa fa-info-circle'></i>
{{ text.details}}</slot>
</b-button>
<b-modal
:title="value.name"
scrollable
centered
ok-only
content-class="s-studyplan-details"
:id="'modal-description-'+value.id"
>
<b-container>
<b-row>
<b-col cols="4"><img :src='value.icon'></b-col>
<b-col cols="8" class="pl-1">
<span v-html="value.description"></span>
</b-col>
</b-row>
</b-container>
</b-modal>
</span>
`,
});
Vue.component('s-course-extrafields', {
props: {
value: {
type: Array,
},
variant: {
type: String,
2024-06-03 23:24:16 +02:00
default() {
return "info";
}
},
position: {
type: String,
2024-06-03 23:24:16 +02:00
default() {
return "below";
}
},
size: {
type: String,
2024-06-03 23:24:16 +02:00
default() {
return "";
}
}
},
data() {
return {
text: strings.extrafields,
};
},
computed: {
fields() {
const fields = [];
for (const ix in this.value) {
const field = this.value[ix];
if (field.position == this.position && field.value && field.value.length > 0) {
fields.push(field);
}
}
return fields;
},
},
methods: {
displaydate(field) {
2024-06-03 23:24:16 +02:00
return formatDate(field.value, false);
},
},
template: `
<div :class="'s-course-extrafields ' + ((fields.length>0)?position:'empty')">
<table v-if="fields.length > 0">
<tr v-for="field in fields">
<td><span class='title' v-if='field.title'>{{ field.title}}</span></td>
<td>
<span v-if='field.type == "date"' class="value date">{{ displaydate(field) }}</span>
<span v-else-if='field.type == "checkbox"'
:class="'value ' + (field.checked?'true':'false')">{{ field.value }}</span>
<span v-else-if='field.type == "textarea"'>
2024-06-12 21:12:00 +02:00
<a class='text-info'
@click.prevent.stop=""
href='#'
v-b-modal="field.courseid+'_'+field.fieldname">{{text.show}}...</a>
<b-modal
:id="field.courseid+'_'+field.fieldname"
:title="field.title""
scrollable
centered
ok-only
><span v-html="field.value"></span
></b-modal>
</span>
<span v-else class="value"><span v-html="field.value"></span></span>
</td>
</tr>
</table>
</div>
`,
});
Vue.component('s-prevnext-selector', {
props: {
value: {
type: Object,
2024-06-03 23:24:16 +02:00
default() {
return null;
}
},
options: {
type: Array,
},
grouped: {
type: Boolean,
2024-06-03 23:24:16 +02:00
'default': false,
},
titlefield: {
type: String,
2024-06-03 23:24:16 +02:00
'default': "title",
},
labelfield: {
type: String,
2024-06-03 23:24:16 +02:00
'default': "label",
},
optionsfield: {
type: String,
2024-06-03 23:24:16 +02:00
'default': "options",
},
defaultselectable: {
type: Boolean,
2024-06-03 23:24:16 +02:00
'default': false,
},
variant: {
type: String,
2024-06-03 23:24:16 +02:00
'default': "",
},
arrows: {
type: Boolean,
2024-06-03 23:24:16 +02:00
'default': false,
},
},
data() {
return {
2024-08-08 19:27:51 +02:00
showarrows: settings("showprevnextarrows"),
text: strings.prevnext,
};
},
computed: {
fields() {
const f = [];
2024-06-03 23:24:16 +02:00
if (this.defaultselectable) {
f.push(null);
}
if (this.grouped) {
2024-06-03 23:24:16 +02:00
for (const gix in this.options) {
const group = this.options[gix];
2024-06-03 23:24:16 +02:00
for (const ix in group[this.optionsfield]) {
const v = group[this.optionsfield][ix];
f.push(v);
}
}
} else {
2024-06-03 23:24:16 +02:00
for (const ix in this.options) {
const v = this.options[ix];
f.push(v);
}
}
return f;
},
bubblevalue: {
2024-06-03 23:24:16 +02:00
get() {
return (this.value) ? this.value : null;
},
set(v) {
this.$emit('input', (v) ? v : null);
},
}
},
methods: {
atFirst() {
if (this.fields.length > 0) {
return this.fields[0] == this.value;
}
return true; // Since it disables the button, do so if pressing it would make no sense.
},
atLast() {
if (this.fields.length > 0) {
2024-06-03 23:24:16 +02:00
const l = this.fields.length - 1;
return this.fields[l] == this.value;
}
return true; // Since it disables the button, do so if pressing it would make no sense.
},
prev() {
if (this.fields.length > 0) {
const index = this.fields.indexOf(this.value);
if (index > 0) {
2024-06-03 23:24:16 +02:00
this.$emit("input", this.fields[index - 1]);
this.$emit("change", this.fields[index - 1]);
}
}
},
next() {
if (this.fields.length > 0) {
const index = this.fields.indexOf(this.value);
2024-06-03 23:24:16 +02:00
const l = this.fields.length - 1;
if (index >= 0 && index < l) {
2024-06-03 23:24:16 +02:00
this.$emit("input", this.fields[index + 1]);
this.$emit("change", this.fields[index + 1]);
}
}
},
selectedchanged(value) {
2024-06-03 23:24:16 +02:00
this.$emit("change", value);
}
},
template: `
<div :class="'s-prevnext-selector '">
2024-08-08 19:27:51 +02:00
<template v-if="showarrows"
><b-button :variant='variant' @click="prev" :disabled="atFirst()"
><i v-if='arrows' class='fa fa-caret-left s-prevnext-arrow'></i
2024-08-08 19:27:51 +02:00
><template v-else>{{text.prev}}</template></b-button
>&nbsp;</template
><b-form-select v-model="bubblevalue"
@change='selectedchanged'
>
<b-form-select-option
:disabled="!defaultselectable"
:value="null"
:class="(defaultselectable)?'font-italic text-primary ':'font-italic'"
><slot name="defaultlabel">{{text.select}}</slot></b-form-select-option>
</b-form-select-option-group>
<template v-if="grouped">
<template v-for="(g,gi) in this.options">
<b-form-select-option-group
v-if="g[optionsfield] && g[optionsfield].length > 0"
:label="g[labelfield]"
>
<b-form-select-option
v-for="(o,i) in g[optionsfield]"
:key="i"
:value="o"
><slot :value="o">{{o[titlefield]}}</slot></b-form-select-option>
</b-form-select-option-group>
</template>
</template>
<template v-else>
<b-form-select-option
v-for="(o,i) in this.options"
:key="i"
:value="o"
><slot :value="o">{{o[titlefield]}}</slot></b-form-select-option>
</template>
2024-08-08 19:27:51 +02:00
</b-form-select
><template v-if="showarrows"
>&nbsp;<b-button :variant='variant' @click="next" :disabled="atLast()"
><i v-if='arrows' class='fa fa-caret-right s-prevnext-arrow'></i
2024-08-08 19:27:51 +02:00
><template v-else>{{text.next}}</template></b-button
></template>
</div>
`,
});
}
};