15 refactor module selection to use store

This commit is contained in:
survellow
2023-11-29 12:00:22 +01:00
parent d41e6f4a3c
commit 1f41abff8b
3 changed files with 52 additions and 82 deletions

View File

@@ -1,5 +1,5 @@
<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";
@@ -12,47 +12,29 @@ const props = defineProps({
}, },
}); });
const modules : ComputedRef<Module[]> = computed(() => props.modules);
store.modules.clear(); store.modules.clear();
type ModuleWithSelection = { module: Module; selected: boolean }; const allSelected : ComputedRef<boolean> =
computed(() => props.modules.every((module) => store.hasModule(module)));
// array of modules with boolean if selected with getter and setter function toggleAllModules(){
const modulesWithSelection: Ref<ModuleWithSelection[]> = ref( if (allSelected.value) {
props.modules.map((propModule) => { store.removeAllModules();
return { } else {
module: propModule, store.overwriteModules(props.modules);
selected: store.hasModule(propModule), }
};
}),
);
const selectedModules: ComputedRef<Module[]> = computed(() =>
modulesWithSelection.value
.filter((module) => module.selected)
.map((module) => module.module),
);
const currentModules = computed(() => props.modules);
function selectAllModules(selection: boolean) {
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);
}); }
}); }
// TODO: Refactor and directly use store hashmap
watch(selectedModules, (newValue: Module[]) => {
store.overwriteModules(newValue);
});
function nextStep() { function nextStep() {
router.push("/additional-modules"); router.push("/additional-modules");
@@ -63,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>
@@ -108,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>

View File

@@ -27,6 +27,9 @@ const moduleStore = defineStore("moduleStore", {
this.modules.set(module.uuid, module); this.modules.set(module.uuid, module);
}); });
}, },
isEmpty(): boolean {
return this.modules.size === 0;
},
countModules(): number { countModules(): number {
return this.modules.size; return this.modules.size;
}, },

View File

@@ -1,16 +1,15 @@
<script lang="ts" setup> <script lang="ts" setup>
import { defineAsyncComponent, ref, Ref, watch} 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 { FilterMatchMode } from "primevue/api"; import { FilterMatchMode } from "primevue/api";
import { DataTableRowSelectEvent, DataTableRowUnselectEvent } from "primevue/datatable";
import { useDialog } from "primevue/usedialog"; 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 { useI18n } from "vue-i18n";
const dialog = useDialog(); const dialog = useDialog();
const { t } = useI18n({ useScope: "global" });
const fetchedModules = async () => { const fetchedModules = async () => {
return await fetchAllModules(); return await fetchAllModules();
@@ -18,10 +17,6 @@ const fetchedModules = async () => {
const store = moduleStore(); const store = moduleStore();
const selectedModules: Ref<Module[]> = ref(
store.getAllModules()
);
const modules: Ref<Module[]> = ref([]); const modules: Ref<Module[]> = ref([]);
const filters = ref({ const filters = ref({
course: { course: {
@@ -38,35 +33,12 @@ const filters = ref({
}, },
}); });
//const selectedModules: Ref<Module[]> = ref([] as Module[]);
//const additionalModules: Ref<Map<string, Module>> = ref(new Map());
fetchedModules().then( fetchedModules().then(
(data) => (data) =>
(modules.value = data.map((module: Module) => { (modules.value = data.map((module: Module) => {
return module; return module;
})), })),
).finally(() => { );
// init selected modules from store with fetched modules and set selected modules
selectedModules.value = store.getAllModules().filter((module: Module) => {
return modules.value.some((m: Module) => m.uuid === module.uuid);
});
});
watch(selectedModules, () => {
//add missing modules to store
selectedModules.value.forEach((module: Module) => {
if (!store.getAllModules().some((m: Module) => m.uuid === module.uuid)) {
store.addModule(module);
}
});
//remove modules from store that are not selected anymore
store.getAllModules().forEach((module: Module) => {
if (!selectedModules.value.some((m: Module) => m.uuid === module.uuid)) {
store.removeModule(module);
}
});
});
async function nextStep() { async function nextStep() {
await router.push("/rename-modules"); await router.push("/rename-modules");
@@ -96,6 +68,15 @@ async function showInfo(module: Module) {
}, },
}); });
} }
function selectModule(event: DataTableRowSelectEvent) {
store.addModule(event.data);
}
function unselectModule(event: DataTableRowUnselectEvent) {
store.removeModule(event.data);
}
</script> </script>
<template> <template>
@@ -109,7 +90,7 @@ async function showInfo(module: Module) {
<DynamicDialog /> <DynamicDialog />
<DataTable <DataTable
v-model:filters="filters" v-model:filters="filters"
v-model:selection="selectedModules" :selection="store.getAllModules()"
:value="modules" :value="modules"
data-key="uuid" data-key="uuid"
paginator paginator
@@ -124,6 +105,8 @@ async function showInfo(module: Module) {
:striped-rows="true" :striped-rows="true"
:select-all="false" :select-all="false"
class="w-10" class="w-10"
@row-select="selectModule"
@row-unselect="unselectModule"
> >
<Column selection-mode="multiple"> <Column selection-mode="multiple">
</Column> </Column>
@@ -196,7 +179,10 @@ async function showInfo(module: Module) {
</DataTable> </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>