mirror of
https://gitlab.dit.htwk-leipzig.de/htwk-software/htwkalender.git
synced 2025-08-02 17:59:14 +02:00
Merge branch 'main' into 52-studium-generale
# Conflicts: # backend/service/fetch/v1/fetchSeminarEventService.go
This commit is contained in:
@@ -1,4 +1,4 @@
|
|||||||
FROM golang:1.21.3-alpine
|
FROM golang:1.21-alpine
|
||||||
|
|
||||||
# Set the Current Working Directory inside the container
|
# Set the Current Working Directory inside the container
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
module htwkalender
|
module htwkalender
|
||||||
|
|
||||||
go 1.21.3
|
go 1.21
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/google/uuid v1.3.1
|
github.com/google/uuid v1.3.1
|
||||||
|
@@ -64,18 +64,22 @@ func GetSeminarGroupsEventsFromHTML(seminarGroupsLabel []string) []model.Seminar
|
|||||||
var seminarGroups []model.SeminarGroup
|
var seminarGroups []model.SeminarGroup
|
||||||
for _, seminarGroupLabel := range seminarGroupsLabel {
|
for _, seminarGroupLabel := range seminarGroupsLabel {
|
||||||
|
|
||||||
ssUrl := "https://stundenplan.htwk-leipzig.de/" + string("ss") + "/Berichte/Text-Listen;Studenten-Sets;name;" + seminarGroupLabel + "?template=sws_semgrp&weeks=1-65"
|
if (time.Now().Month() >= 3) && (time.Now().Month() <= 10) {
|
||||||
result, getError := fetch.GetHTML(ssUrl)
|
ssUrl := "https://stundenplan.htwk-leipzig.de/" + string("ss") + "/Berichte/Text-Listen;Studenten-Sets;name;" + seminarGroupLabel + "?template=sws_semgrp&weeks=1-65"
|
||||||
if getError == nil {
|
result, getError := fetch.GetHTML(ssUrl)
|
||||||
seminarGroup := parseSeminarGroup(result)
|
if getError == nil {
|
||||||
seminarGroups = append(seminarGroups, seminarGroup)
|
seminarGroup := parseSeminarGroup(result)
|
||||||
|
seminarGroups = append(seminarGroups, seminarGroup)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
wsUrl := "https://stundenplan.htwk-leipzig.de/" + string("ws") + "/Berichte/Text-Listen;Studenten-Sets;name;" + seminarGroupLabel + "?template=sws_semgrp&weeks=1-65"
|
if (time.Now().Month() >= 9) || (time.Now().Month() <= 4) {
|
||||||
result, getError = fetch.GetHTML(wsUrl)
|
wsUrl := "https://stundenplan.htwk-leipzig.de/" + string("ws") + "/Berichte/Text-Listen;Studenten-Sets;name;" + seminarGroupLabel + "?template=sws_semgrp&weeks=1-65"
|
||||||
if getError == nil {
|
result, getError := fetch.GetHTML(wsUrl)
|
||||||
seminarGroup := parseSeminarGroup(result)
|
if getError == nil {
|
||||||
seminarGroups = append(seminarGroups, seminarGroup)
|
seminarGroup := parseSeminarGroup(result)
|
||||||
|
seminarGroups = append(seminarGroups, seminarGroup)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return seminarGroups
|
return seminarGroups
|
||||||
|
@@ -1,11 +1,24 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import MenuBar from "./components/MenuBar.vue";
|
import MenuBar from "./components/MenuBar.vue";
|
||||||
import { RouterView } from "vue-router";
|
import {RouteRecordName, RouterView} from "vue-router";
|
||||||
|
import CalendarPreview from "./components/CalendarPreview.vue";
|
||||||
|
import moduleStore from "./store/moduleStore.ts";
|
||||||
|
|
||||||
|
const disabledPages = ["room-finder", "faq", "imprint", "privacy-policy", "edit"];
|
||||||
|
|
||||||
|
const store = moduleStore();
|
||||||
|
|
||||||
|
const isDisabled = (routeName: RouteRecordName | null | undefined) => {
|
||||||
|
return !disabledPages.includes(routeName as string) && store.modules.size > 0
|
||||||
|
};
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<MenuBar />
|
<MenuBar />
|
||||||
<RouterView />
|
<RouterView />
|
||||||
|
<!-- show CalendarPreview but only on specific router views -->
|
||||||
|
<CalendarPreview v-if="isDisabled($route.name)"/>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped></style>
|
<style scoped></style>
|
||||||
|
84
frontend/src/components/CalendarPreview.vue
Normal file
84
frontend/src/components/CalendarPreview.vue
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
<script lang="ts" setup>
|
||||||
|
import { Ref, computed, ref } from "vue";
|
||||||
|
import moduleStore from "../store/moduleStore";
|
||||||
|
import { useI18n } from "vue-i18n";
|
||||||
|
|
||||||
|
const dialogVisible: Ref<boolean> = ref(false);
|
||||||
|
const mobilePage = ref(true);
|
||||||
|
const { t } = useI18n({ useScope: "global" });
|
||||||
|
|
||||||
|
const store = moduleStore();
|
||||||
|
const tableData = computed(() =>
|
||||||
|
store.getAllModules().map((module) => {
|
||||||
|
return {
|
||||||
|
Course: module.course,
|
||||||
|
Module: module.userDefinedName,
|
||||||
|
};
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
const previewOn: Ref<boolean> = computed(() => {
|
||||||
|
return moduleStore().modules.size > 0;
|
||||||
|
});
|
||||||
|
|
||||||
|
const columns = computed(() => [
|
||||||
|
{ field: "Course", header: t("calendarPreview.course") },
|
||||||
|
{ field: "Module", header: t("calendarPreview.module") },
|
||||||
|
]);
|
||||||
|
|
||||||
|
const updateMobile = () => {
|
||||||
|
mobilePage.value = window.innerWidth <= 992;
|
||||||
|
};
|
||||||
|
|
||||||
|
updateMobile();
|
||||||
|
|
||||||
|
window.addEventListener("resize", updateMobile);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Dialog
|
||||||
|
id="calendar-dialog"
|
||||||
|
ref="calendar"
|
||||||
|
:visible="dialogVisible && previewOn"
|
||||||
|
:maximizable="!mobilePage"
|
||||||
|
:draggable="false"
|
||||||
|
:header="$t('calendarPreview.preview-long')"
|
||||||
|
class="w-full lg:w-30rem lg:h-auto m-0 lg:m-2"
|
||||||
|
position="bottomright"
|
||||||
|
@update:visible="dialogVisible = $event"
|
||||||
|
>
|
||||||
|
<DataTable
|
||||||
|
:value="tableData"
|
||||||
|
edit-mode="cell"
|
||||||
|
table-class="editable-cells-table"
|
||||||
|
responsive-layout="scroll"
|
||||||
|
>
|
||||||
|
<Column
|
||||||
|
v-for="column in columns"
|
||||||
|
:key="column.field"
|
||||||
|
:field="column.field"
|
||||||
|
:header="column.header"
|
||||||
|
:row-editor="false"
|
||||||
|
>
|
||||||
|
<template #body="{ data, field }">
|
||||||
|
{{ data[field] }}
|
||||||
|
</template>
|
||||||
|
</Column>
|
||||||
|
</DataTable>
|
||||||
|
</Dialog>
|
||||||
|
<SpeedDial
|
||||||
|
v-if="previewOn && !dialogVisible"
|
||||||
|
:style="{ position: 'fixed', bottom: '2rem', right: '2rem' }"
|
||||||
|
>
|
||||||
|
<template #button>
|
||||||
|
<Button
|
||||||
|
icon="pi pi-calendar"
|
||||||
|
:label="$t('calendarPreview.preview')"
|
||||||
|
class="p-button-rounded p-button-primary"
|
||||||
|
@click="dialogVisible = true"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</SpeedDial>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style scoped></style>
|
@@ -1,9 +1,10 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { computed, ComputedRef, PropType, Ref, ref, watch } from "vue";
|
import { computed, ComputedRef, PropType } from "vue";
|
||||||
import { Module } from "../model/module.ts";
|
import { Module } from "../model/module.ts";
|
||||||
import moduleStore from "../store/moduleStore";
|
import moduleStore from "../store/moduleStore";
|
||||||
import router from "../router";
|
import router from "../router";
|
||||||
|
|
||||||
|
const store = moduleStore();
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
modules: {
|
modules: {
|
||||||
type: Array as PropType<Module[]>,
|
type: Array as PropType<Module[]>,
|
||||||
@@ -11,48 +12,31 @@ const props = defineProps({
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
type ModuleWithSelection = { module: Module; selected: boolean };
|
const modules : ComputedRef<Module[]> = computed(() => props.modules);
|
||||||
|
|
||||||
// array of modules with boolean if selected with getter and setter
|
store.modules.clear();
|
||||||
const modulesWithSelection: Ref<ModuleWithSelection[]> = ref(
|
|
||||||
props.modules.map((propModule) => {
|
|
||||||
return {
|
|
||||||
module: propModule,
|
|
||||||
selected: moduleStore().modules.some((module: Module) =>
|
|
||||||
module.isEqual ? module.isEqual(propModule) : false,
|
|
||||||
),
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
const selectedModules: ComputedRef<Module[]> = computed(() =>
|
const allSelected : ComputedRef<boolean> =
|
||||||
modulesWithSelection.value
|
computed(() => props.modules.every((module) => store.hasModule(module)));
|
||||||
.filter((module) => module.selected)
|
|
||||||
.map((module) => module.module),
|
|
||||||
);
|
|
||||||
|
|
||||||
const currentModules = computed(() => props.modules);
|
function toggleAllModules(){
|
||||||
|
if (allSelected.value) {
|
||||||
function selectAllModules(selection: boolean) {
|
store.removeAllModules();
|
||||||
|
} else {
|
||||||
|
store.overwriteModules(props.modules);
|
||||||
|
}
|
||||||
console.debug(props.modules);
|
console.debug(props.modules);
|
||||||
modulesWithSelection.value.forEach((module) => {
|
|
||||||
module.selected = selection;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const allSelected: Ref<boolean> = ref(true);
|
function toggleModule(module: Module) {
|
||||||
|
if (store.hasModule(module)) {
|
||||||
watch(currentModules, (newValue: Module[]) => {
|
store.removeModule(module);
|
||||||
modulesWithSelection.value = newValue.map((module) => {
|
} else {
|
||||||
return { module: module, selected: false };
|
store.addModule(module);
|
||||||
});
|
}
|
||||||
});
|
}
|
||||||
|
|
||||||
function nextStep() {
|
function nextStep() {
|
||||||
selectedModules.value.forEach((module: Module) => {
|
|
||||||
moduleStore().addModule(module);
|
|
||||||
});
|
|
||||||
|
|
||||||
router.push("/additional-modules");
|
router.push("/additional-modules");
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -61,31 +45,29 @@ function nextStep() {
|
|||||||
<div class="flex flex-column card-container mx-8 mt-2">
|
<div class="flex flex-column card-container mx-8 mt-2">
|
||||||
<div class="flex align-items-center justify-content-center mb-3">
|
<div class="flex align-items-center justify-content-center mb-3">
|
||||||
<Button
|
<Button
|
||||||
:disabled="selectedModules.length < 1"
|
:disabled="store.isEmpty()"
|
||||||
class="col-4 justify-content-center"
|
class="col-4 justify-content-center"
|
||||||
@click="nextStep()"
|
@click="nextStep()"
|
||||||
>{{ $t("moduleSelection.nextStep") }}
|
>{{ $t("moduleSelection.nextStep") }}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex align-items-center justify-content-center">
|
<div class="flex align-items-center justify-content-center">
|
||||||
<DataView :value="modulesWithSelection" data-key="module">
|
<DataView :value="modules" data-key="uuid">
|
||||||
<template #header>
|
<template #header>
|
||||||
<div class="flex justify-content-between flex-wrap">
|
<div class="flex justify-content-between flex-wrap">
|
||||||
<div class="flex align-items-center justify-content-center">
|
<div class="flex align-items-center justify-content-center">
|
||||||
<h3>
|
<h3>
|
||||||
{{ $t("moduleSelection.modules") }} -
|
{{ $t("moduleSelection.modules") }} -
|
||||||
{{ selectedModules.length }}
|
{{ store.countModules() }}
|
||||||
</h3>
|
</h3>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex align-items-center justify-content-center">
|
<div class="flex align-items-center justify-content-center">
|
||||||
<ToggleButton
|
{{ allSelected ? $t('moduleSelection.deselectAll') : $t('moduleSelection.selectAll')}}
|
||||||
v-model="allSelected"
|
<InputSwitch
|
||||||
class="w-12rem"
|
class="mx-4"
|
||||||
off-icon="pi pi-times"
|
:disabled="modules.length === 0"
|
||||||
:off-label="$t('moduleSelection.deselectAll')"
|
:model-value="allSelected"
|
||||||
on-icon="pi pi-check"
|
@update:model-value="toggleAllModules()"
|
||||||
:on-label="$t('moduleSelection.selectAll')"
|
|
||||||
@click="selectAllModules(!allSelected)"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -106,18 +88,19 @@ function nextStep() {
|
|||||||
<div
|
<div
|
||||||
class="flex flex-column align-items-center justify-content-center sm:align-items-start gap-3"
|
class="flex flex-column align-items-center justify-content-center sm:align-items-start gap-3"
|
||||||
>
|
>
|
||||||
<p class="text-lg">{{ slotProps.data.module.name }}</p>
|
<p class="text-lg">{{ slotProps.data.name }}</p>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
class="flex sm:flex-column justify-content-center sm:align-items-end gap-3 sm:gap-2"
|
class="flex sm:flex-column justify-content-center sm:align-items-end gap-3 sm:gap-2"
|
||||||
>
|
>
|
||||||
<ToggleButton
|
<ToggleButton
|
||||||
v-model="modulesWithSelection[slotProps.index].selected"
|
|
||||||
class="w-9rem"
|
class="w-9rem"
|
||||||
off-icon="pi pi-times"
|
off-icon="pi pi-times"
|
||||||
:off-label="$t('moduleSelection.unselected')"
|
:off-label="$t('moduleSelection.unselected')"
|
||||||
on-icon="pi pi-check"
|
on-icon="pi pi-check"
|
||||||
:on-label="$t('moduleSelection.selected')"
|
:on-label="$t('moduleSelection.selected')"
|
||||||
|
:model-value="store.hasModule(slotProps.data)"
|
||||||
|
@update:model-value="toggleModule(slotProps.data)"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -68,10 +68,4 @@ const placeholders = computed(() => [
|
|||||||
</Dialog>
|
</Dialog>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped></style>
|
||||||
.small-button.p-button {
|
|
||||||
width: 2rem;
|
|
||||||
height: 2rem;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
@@ -9,8 +9,9 @@ import { onlyWhitespace } from "../helpers/strings.ts";
|
|||||||
import { useI18n } from "vue-i18n";
|
import { useI18n } from "vue-i18n";
|
||||||
const { t } = useI18n({ useScope: "global" });
|
const { t } = useI18n({ useScope: "global" });
|
||||||
|
|
||||||
|
const store = moduleStore();
|
||||||
const tableData = ref(
|
const tableData = ref(
|
||||||
moduleStore().modules.map((module) => {
|
store.getAllModules().map((module) => {
|
||||||
return {
|
return {
|
||||||
Course: module.course,
|
Course: module.course,
|
||||||
Module: module,
|
Module: module,
|
||||||
@@ -25,7 +26,7 @@ const columns = computed(() => [
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
async function finalStep() {
|
async function finalStep() {
|
||||||
const token: string = await createIndividualFeed(moduleStore().modules);
|
const token: string = await createIndividualFeed(store.getAllModules());
|
||||||
tokenStore().setToken(token);
|
tokenStore().setToken(token);
|
||||||
await router.push("/calendar-link");
|
await router.push("/calendar-link");
|
||||||
}
|
}
|
||||||
@@ -124,10 +125,4 @@ async function finalStep() {
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped></style>
|
||||||
.small-button.p-button {
|
|
||||||
width: 2rem;
|
|
||||||
height: 2rem;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
@@ -11,8 +11,9 @@ import { onlyWhitespace } from "../../helpers/strings.ts";
|
|||||||
import { useI18n } from "vue-i18n";
|
import { useI18n } from "vue-i18n";
|
||||||
const { t } = useI18n({ useScope: "global" });
|
const { t } = useI18n({ useScope: "global" });
|
||||||
|
|
||||||
|
const store = moduleStore();
|
||||||
const tableData = computed(() =>
|
const tableData = computed(() =>
|
||||||
moduleStore().modules.map((module: Module) => {
|
store.getAllModules().map((module: Module) => {
|
||||||
return {
|
return {
|
||||||
Course: module.course,
|
Course: module.course,
|
||||||
Module: module,
|
Module: module,
|
||||||
@@ -45,7 +46,7 @@ fetchedModules().then(
|
|||||||
);
|
);
|
||||||
|
|
||||||
async function finalStep() {
|
async function finalStep() {
|
||||||
await saveIndividualFeed(tokenStore().token, moduleStore().modules);
|
await saveIndividualFeed(tokenStore().token, store.getAllModules());
|
||||||
await router.push("/calendar-link");
|
await router.push("/calendar-link");
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
@@ -159,10 +160,4 @@ async function finalStep() {
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped></style>
|
||||||
.small-button.p-button {
|
|
||||||
width: 2rem;
|
|
||||||
height: 2rem;
|
|
||||||
padding: 0;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
@@ -38,7 +38,7 @@
|
|||||||
"semester": "Semester",
|
"semester": "Semester",
|
||||||
"module": "Modul",
|
"module": "Modul",
|
||||||
"day": "Tag",
|
"day": "Tag",
|
||||||
"start": "Begin",
|
"start": "Anfang",
|
||||||
"end": "Ende",
|
"end": "Ende",
|
||||||
"room": "Raum",
|
"room": "Raum",
|
||||||
"type": "Art",
|
"type": "Art",
|
||||||
@@ -57,8 +57,17 @@
|
|||||||
"dropDown": "Wähle weitere Module aus",
|
"dropDown": "Wähle weitere Module aus",
|
||||||
"module": "Modul",
|
"module": "Modul",
|
||||||
"modules": "Module",
|
"modules": "Module",
|
||||||
"dropDownFooterSelected": "ausgewählt",
|
"footerModulesSelected": "{count} Modul ausgewählt | {count} Module ausgewählt",
|
||||||
"nextStep": "Weiter"
|
"nextStep": "Weiter",
|
||||||
|
"professor": "Dozent",
|
||||||
|
"course": "Seminargruppe",
|
||||||
|
"info": "Info",
|
||||||
|
"info-long": "Information",
|
||||||
|
"paginator": {
|
||||||
|
"from": "",
|
||||||
|
"to": " bis ",
|
||||||
|
"of": " von insgesamt "
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"renameModules": {
|
"renameModules": {
|
||||||
"reminder": "Erinnerung",
|
"reminder": "Erinnerung",
|
||||||
@@ -90,6 +99,12 @@
|
|||||||
"toGoogleCalendar": "Google Kalender",
|
"toGoogleCalendar": "Google Kalender",
|
||||||
"toMicrosoftCalendar": "Microsoft Kalender"
|
"toMicrosoftCalendar": "Microsoft Kalender"
|
||||||
},
|
},
|
||||||
|
"calendarPreview": {
|
||||||
|
"preview": "Vorschau",
|
||||||
|
"preview-long": "Kalendervorschau",
|
||||||
|
"module": "Modul",
|
||||||
|
"course": "Seminargruppe"
|
||||||
|
},
|
||||||
"faqView": {
|
"faqView": {
|
||||||
"headline": "Fragen und Antworten",
|
"headline": "Fragen und Antworten",
|
||||||
"firstQuestion": "Wie funktioniert das Kalender erstellen mit dem HTWKalender?",
|
"firstQuestion": "Wie funktioniert das Kalender erstellen mit dem HTWKalender?",
|
||||||
|
@@ -53,12 +53,21 @@
|
|||||||
"noCalendarFound": "no calendar found"
|
"noCalendarFound": "no calendar found"
|
||||||
},
|
},
|
||||||
"additionalModules": {
|
"additionalModules": {
|
||||||
"subTitle": "Select additional Modules that are not listed in the regular semester for your Course",
|
"subTitle": "Select additional modules that are not listed in the regular semester for your course",
|
||||||
"dropDown": "Select additional modules",
|
"dropDown": "Select additional modules",
|
||||||
"module": "module",
|
"module": "module",
|
||||||
"modules": "modules",
|
"modules": "modules",
|
||||||
"dropDownFooterSelected": "selected",
|
"footerModulesSelected": "{count} module selected | {count} modules selected",
|
||||||
"nextStep": "next step"
|
"nextStep": "next step",
|
||||||
|
"professor": "professor",
|
||||||
|
"course": "course",
|
||||||
|
"info": "info",
|
||||||
|
"info-long": "information",
|
||||||
|
"paginator": {
|
||||||
|
"from": "",
|
||||||
|
"to": " to ",
|
||||||
|
"of": " of "
|
||||||
|
}
|
||||||
},
|
},
|
||||||
"renameModules": {
|
"renameModules": {
|
||||||
"reminder": "reminder",
|
"reminder": "reminder",
|
||||||
@@ -90,6 +99,12 @@
|
|||||||
"toGoogleCalendar": "to Google Calendar",
|
"toGoogleCalendar": "to Google Calendar",
|
||||||
"toMicrosoftCalendar": "to Microsoft Calendar"
|
"toMicrosoftCalendar": "to Microsoft Calendar"
|
||||||
},
|
},
|
||||||
|
"calendarPreview": {
|
||||||
|
"preview": "preview",
|
||||||
|
"preview-long": "calendar preview",
|
||||||
|
"module": "module",
|
||||||
|
"course": "course"
|
||||||
|
},
|
||||||
"faqView": {
|
"faqView": {
|
||||||
"headline": "faq",
|
"headline": "faq",
|
||||||
"firstQuestion": "How does calendar creation work with HTWKalender?",
|
"firstQuestion": "How does calendar creation work with HTWKalender?",
|
||||||
|
@@ -16,6 +16,7 @@ import "primevue/resources/themes/viva-dark/theme.css";
|
|||||||
import "primeicons/primeicons.css";
|
import "primeicons/primeicons.css";
|
||||||
import "primeflex/primeflex.css";
|
import "primeflex/primeflex.css";
|
||||||
import router from "./router";
|
import router from "./router";
|
||||||
|
import SpeedDial from "primevue/speeddial";
|
||||||
import TabView from "primevue/tabview";
|
import TabView from "primevue/tabview";
|
||||||
import TabPanel from "primevue/tabpanel";
|
import TabPanel from "primevue/tabpanel";
|
||||||
import Tag from "primevue/tag";
|
import Tag from "primevue/tag";
|
||||||
@@ -30,6 +31,7 @@ import Column from "primevue/column";
|
|||||||
import DynamicDialog from "primevue/dynamicdialog";
|
import DynamicDialog from "primevue/dynamicdialog";
|
||||||
import DialogService from "primevue/dialogservice";
|
import DialogService from "primevue/dialogservice";
|
||||||
import ProgressSpinner from "primevue/progressspinner";
|
import ProgressSpinner from "primevue/progressspinner";
|
||||||
|
import Checkbox from "primevue/checkbox";
|
||||||
import i18n from "./i18n";
|
import i18n from "./i18n";
|
||||||
|
|
||||||
const app = createApp(App);
|
const app = createApp(App);
|
||||||
@@ -52,6 +54,7 @@ app.component("InputSwitch", InputSwitch);
|
|||||||
app.component("Card", Card);
|
app.component("Card", Card);
|
||||||
app.component("DataView", DataView);
|
app.component("DataView", DataView);
|
||||||
app.component("ToggleButton", ToggleButton);
|
app.component("ToggleButton", ToggleButton);
|
||||||
|
app.component("SpeedDial", SpeedDial);
|
||||||
app.component("TabView", TabView);
|
app.component("TabView", TabView);
|
||||||
app.component("TabPanel", TabPanel);
|
app.component("TabPanel", TabPanel);
|
||||||
app.component("MultiSelect", MultiSelect);
|
app.component("MultiSelect", MultiSelect);
|
||||||
@@ -63,4 +66,5 @@ app.component("DataTable", DataTable);
|
|||||||
app.component("Column", Column);
|
app.component("Column", Column);
|
||||||
app.component("DynamicDialog", DynamicDialog);
|
app.component("DynamicDialog", DynamicDialog);
|
||||||
app.component("ProgressSpinner", ProgressSpinner);
|
app.component("ProgressSpinner", ProgressSpinner);
|
||||||
|
app.component("Checkbox", Checkbox);
|
||||||
app.mount("#app");
|
app.mount("#app");
|
||||||
|
@@ -12,7 +12,7 @@ export class Module {
|
|||||||
public events: Event[] = [],
|
public events: Event[] = [],
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
isEqual(module: Module): Boolean {
|
isEqual(module: Module): boolean {
|
||||||
return this.name === module.name && this.course === module.course;
|
return this.name === module.name && this.course === module.course;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -3,18 +3,42 @@ import { defineStore } from "pinia";
|
|||||||
|
|
||||||
const moduleStore = defineStore("moduleStore", {
|
const moduleStore = defineStore("moduleStore", {
|
||||||
state: () => ({
|
state: () => ({
|
||||||
modules: [] as Module[],
|
modules: new Map<string, Module>(),
|
||||||
}),
|
}),
|
||||||
actions: {
|
actions: {
|
||||||
addModule(module: Module) {
|
addModule(module: Module) {
|
||||||
this.modules.push(module);
|
this.modules.set(module.uuid, module);
|
||||||
},
|
},
|
||||||
removeModule(module: Module) {
|
removeModule(module: Module) {
|
||||||
this.modules.splice(this.modules.indexOf(module), 1);
|
this.modules.delete(module.uuid);
|
||||||
|
},
|
||||||
|
hasModule(module: Module): boolean {
|
||||||
|
return (
|
||||||
|
this.modules.has(module.uuid) &&
|
||||||
|
(this.modules.get(module.uuid)?.isEqual(module) ?? false)
|
||||||
|
);
|
||||||
},
|
},
|
||||||
removeAllModules() {
|
removeAllModules() {
|
||||||
this.modules = [];
|
this.modules.clear();
|
||||||
},
|
},
|
||||||
|
overwriteModules(modules: Module[]) {
|
||||||
|
this.modules.clear();
|
||||||
|
modules.forEach((module) => {
|
||||||
|
this.modules.set(module.uuid, module);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
isEmpty(): boolean {
|
||||||
|
return this.modules.size === 0;
|
||||||
|
},
|
||||||
|
countModules(): number {
|
||||||
|
return this.modules.size;
|
||||||
|
},
|
||||||
|
getAllModules(): Module[] {
|
||||||
|
return Array.from(this.modules.values());
|
||||||
|
},
|
||||||
|
containsModule(module: Module): boolean {
|
||||||
|
return this.modules.has(module.uuid);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -1,3 +1,10 @@
|
|||||||
body {
|
body {
|
||||||
font-family: var(--font-family);
|
font-family: var(--font-family);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.small-button.p-button.p-button-icon-only,
|
||||||
|
.small-button.p-button.p-button-icon-only.p-button-rounded {
|
||||||
|
width: 2rem;
|
||||||
|
height: 2rem;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
@@ -1,12 +1,13 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { defineAsyncComponent, ref, Ref } from "vue";
|
import { defineAsyncComponent, ref, Ref} from "vue";
|
||||||
import { Module } from "../model/module.ts";
|
import { Module } from "../model/module.ts";
|
||||||
import { fetchAllModules } from "../api/fetchCourse.ts";
|
import { fetchAllModules } from "../api/fetchCourse.ts";
|
||||||
import moduleStore from "../store/moduleStore.ts";
|
import moduleStore from "../store/moduleStore.ts";
|
||||||
import { MultiSelectAllChangeEvent } from "primevue/multiselect";
|
import { FilterMatchMode } from "primevue/api";
|
||||||
|
import { DataTableRowSelectEvent, DataTableRowUnselectEvent } from "primevue/datatable";
|
||||||
|
import { useDialog } from "primevue/usedialog";
|
||||||
import router from "../router";
|
import router from "../router";
|
||||||
import { fetchModule } from "../api/fetchModule.ts";
|
import { fetchModule } from "../api/fetchModule.ts";
|
||||||
import { useDialog } from "primevue/usedialog";
|
|
||||||
import { useI18n } from "vue-i18n";
|
import { useI18n } from "vue-i18n";
|
||||||
|
|
||||||
const dialog = useDialog();
|
const dialog = useDialog();
|
||||||
@@ -16,9 +17,26 @@ const fetchedModules = async () => {
|
|||||||
return await fetchAllModules();
|
return await fetchAllModules();
|
||||||
};
|
};
|
||||||
|
|
||||||
const modules: Ref<Module[]> = ref([]);
|
const store = moduleStore();
|
||||||
|
if (store.isEmpty()) {
|
||||||
|
router.replace("/");
|
||||||
|
}
|
||||||
|
|
||||||
const selectedModules: Ref<Module[]> = ref([] as Module[]);
|
const modules: Ref<Module[]> = ref([]);
|
||||||
|
const filters = ref({
|
||||||
|
course: {
|
||||||
|
value: null,
|
||||||
|
matchMode: FilterMatchMode.CONTAINS,
|
||||||
|
},
|
||||||
|
name: {
|
||||||
|
value: null,
|
||||||
|
matchMode: FilterMatchMode.CONTAINS,
|
||||||
|
},
|
||||||
|
prof: {
|
||||||
|
value: null,
|
||||||
|
matchMode: FilterMatchMode.CONTAINS,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
fetchedModules().then(
|
fetchedModules().then(
|
||||||
(data) =>
|
(data) =>
|
||||||
@@ -28,10 +46,6 @@ fetchedModules().then(
|
|||||||
);
|
);
|
||||||
|
|
||||||
async function nextStep() {
|
async function nextStep() {
|
||||||
selectedModules.value.forEach((module: Module) => {
|
|
||||||
moduleStore().addModule(module);
|
|
||||||
});
|
|
||||||
|
|
||||||
await router.push("/rename-modules");
|
await router.push("/rename-modules");
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,36 +74,14 @@ async function showInfo(module: Module) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const display = (module: Module) => module.name + " (" + module.course + ")";
|
function selectModule(event: DataTableRowSelectEvent) {
|
||||||
|
store.addModule(event.data);
|
||||||
const selectAll = ref(false);
|
|
||||||
|
|
||||||
const onSelectAllChange = (event: MultiSelectAllChangeEvent) => {
|
|
||||||
selectedModules.value = event.checked
|
|
||||||
? modules.value.map((module: Module) => module)
|
|
||||||
: [];
|
|
||||||
selectAll.value = event.checked;
|
|
||||||
};
|
|
||||||
|
|
||||||
function selectChange() {
|
|
||||||
selectAll.value = selectedModules.value.length === modules.value.length;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function itemsLabel(selectedModules: Module[]): string {
|
function unselectModule(event: DataTableRowUnselectEvent) {
|
||||||
return (selectedModules ? selectedModules.length : 0) != 1
|
store.removeModule(event.data);
|
||||||
? t("additionalModules.modules")
|
|
||||||
: t("additionalModules.module");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function itemsLabelWithNumber(selectedModules: Module[]): string {
|
|
||||||
return (
|
|
||||||
selectedModules.length.toString() +
|
|
||||||
" " +
|
|
||||||
itemsLabel(selectedModules) +
|
|
||||||
" " +
|
|
||||||
t("additionalModules.dropDownFooterSelected")
|
|
||||||
);
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
@@ -100,56 +92,103 @@ function itemsLabelWithNumber(selectedModules: Module[]): string {
|
|||||||
</h3>
|
</h3>
|
||||||
</div>
|
</div>
|
||||||
<div class="card flex align-items-center justify-content-center m-2">
|
<div class="card flex align-items-center justify-content-center m-2">
|
||||||
<MultiSelect
|
<DynamicDialog />
|
||||||
v-model="selectedModules"
|
<DataTable
|
||||||
:max-selected-labels="1"
|
v-model:filters="filters"
|
||||||
:option-label="display"
|
:selection="store.getAllModules()"
|
||||||
:options="modules"
|
:value="modules"
|
||||||
:select-all="selectAll"
|
data-key="uuid"
|
||||||
:virtual-scroller-options="{ itemSize: 70 }"
|
paginator
|
||||||
class="custom-multiselect"
|
:rows="10"
|
||||||
filter
|
:rows-per-page-options="[5, 10, 20, 50]"
|
||||||
:placeholder="$t('additionalModules.dropDown')"
|
paginator-template="FirstPageLink PrevPageLink CurrentPageReport NextPageLink LastPageLink RowsPerPageDropdown"
|
||||||
:auto-filter-focus="true"
|
:current-page-report-template="
|
||||||
:show-toggle-all="false"
|
$t('additionalModules.paginator.from') +
|
||||||
:selected-items-label="itemsLabelWithNumber(selectedModules)"
|
'{first}' +
|
||||||
@change="selectChange()"
|
$t('additionalModules.paginator.to') +
|
||||||
@selectall-change="onSelectAllChange($event)"
|
'{last}' +
|
||||||
|
$t('additionalModules.paginator.of') +
|
||||||
|
'{totalRecords}'
|
||||||
|
"
|
||||||
|
filter-display="row"
|
||||||
|
:loading="!modules.length"
|
||||||
|
loading-icon="pi pi-spinner"
|
||||||
|
:show-gridlines="true"
|
||||||
|
:striped-rows="true"
|
||||||
|
:select-all="false"
|
||||||
|
class="w-10"
|
||||||
|
@row-select="selectModule"
|
||||||
|
@row-unselect="unselectModule"
|
||||||
>
|
>
|
||||||
<template #option="slotProps">
|
<Column selection-mode="multiple">
|
||||||
<div class="flex justify-content-between w-full">
|
</Column>
|
||||||
<div class="flex align-items-center justify-content-center">
|
|
||||||
<p class="text-1xl white-space-normal p-mb-0">
|
<Column
|
||||||
{{ display(slotProps.option) }}
|
field="course"
|
||||||
</p>
|
:header="$t('additionalModules.course')"
|
||||||
</div>
|
:show-clear-button="false"
|
||||||
<div class="flex align-items-center justify-content-center ml-2">
|
:show-filter-menu="false"
|
||||||
<Button
|
>
|
||||||
class="small-button"
|
<template #filter="{ filterModel, filterCallback }">
|
||||||
icon="pi pi-info"
|
<InputText
|
||||||
severity="secondary"
|
v-model="filterModel.value"
|
||||||
rounded
|
type="text"
|
||||||
outlined
|
class="p-column-filter max-w-10rem"
|
||||||
aria-label="Information"
|
@input="filterCallback()"
|
||||||
@click.stop="showInfo(slotProps.option)"
|
/>
|
||||||
></Button>
|
</template>
|
||||||
<DynamicDialog />
|
</Column>
|
||||||
</div>
|
<Column
|
||||||
</div>
|
field="name"
|
||||||
</template>
|
:header="$t('additionalModules.module')"
|
||||||
|
:show-clear-button="false"
|
||||||
|
:show-filter-menu="false"
|
||||||
|
>
|
||||||
|
<template #filter="{ filterModel, filterCallback }">
|
||||||
|
<InputText
|
||||||
|
v-model="filterModel.value"
|
||||||
|
type="text"
|
||||||
|
class="p-column-filter"
|
||||||
|
@input="filterCallback()"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</Column>
|
||||||
|
<Column
|
||||||
|
field="prof"
|
||||||
|
:header="$t('additionalModules.professor')"
|
||||||
|
:show-clear-button="false"
|
||||||
|
:show-filter-menu="false"
|
||||||
|
>
|
||||||
|
</Column>
|
||||||
|
<Column
|
||||||
|
:header="$t('additionalModules.info')"
|
||||||
|
>
|
||||||
|
<template #body="slotProps">
|
||||||
|
<Button
|
||||||
|
icon="pi pi-info"
|
||||||
|
severity="secondary"
|
||||||
|
rounded
|
||||||
|
outlined
|
||||||
|
:aria-label="$t('additionalModules.info-long')"
|
||||||
|
class="small-button"
|
||||||
|
@click.stop="showInfo(slotProps.data)"
|
||||||
|
></Button>
|
||||||
|
</template>
|
||||||
|
</Column>
|
||||||
<template #footer>
|
<template #footer>
|
||||||
<div class="py-2 px-3">
|
<div class="py-2 px-3">
|
||||||
<b>{{ selectedModules ? selectedModules.length : 0 }}</b>
|
{{
|
||||||
item{{
|
t('additionalModules.footerModulesSelected', { count: store?.countModules() ?? 0 })
|
||||||
(selectedModules ? selectedModules.length : 0) > 1 ? "s" : ""
|
}}
|
||||||
}}
|
|
||||||
selected.
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</MultiSelect>
|
</DataTable>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex align-items-center justify-content-center h-4rem m-2">
|
<div class="flex align-items-center justify-content-center h-4rem m-2">
|
||||||
<Button @click="nextStep()">{{
|
<Button
|
||||||
|
:disabled="store.isEmpty()"
|
||||||
|
@click="nextStep()"
|
||||||
|
>{{
|
||||||
$t("additionalModules.nextStep")
|
$t("additionalModules.nextStep")
|
||||||
}}</Button>
|
}}</Button>
|
||||||
</div>
|
</div>
|
||||||
@@ -170,4 +209,8 @@ function itemsLabelWithNumber(selectedModules: Module[]): string {
|
|||||||
height: 2rem;
|
height: 2rem;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:deep(.p-filter-column .p-checkbox .p-component) {
|
||||||
|
display: none !important;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@@ -12,7 +12,7 @@ const { t } = useI18n({ useScope: "global" });
|
|||||||
const toast = useToast();
|
const toast = useToast();
|
||||||
|
|
||||||
const token: Ref<string> = ref("");
|
const token: Ref<string> = ref("");
|
||||||
const modules: Ref<Module[]> = ref(moduleStore().modules);
|
const modules: Ref<Map<string, Module>> = ref(moduleStore().modules);
|
||||||
|
|
||||||
function extractToken(token: string): string {
|
function extractToken(token: string): string {
|
||||||
const tokenRegex = /^[a-z0-9]{15}$/;
|
const tokenRegex = /^[a-z0-9]{15}$/;
|
||||||
@@ -51,7 +51,7 @@ function loadCalendar(): void {
|
|||||||
data.forEach((module) => {
|
data.forEach((module) => {
|
||||||
moduleStore().addModule(module);
|
moduleStore().addModule(module);
|
||||||
});
|
});
|
||||||
modules.value = data;
|
modules.value = moduleStore().modules;
|
||||||
router.push("/edit-additional-modules");
|
router.push("/edit-additional-modules");
|
||||||
} else {
|
} else {
|
||||||
toast.add({
|
toast.add({
|
||||||
|
Reference in New Issue
Block a user