Compare commits
5 Commits
5d9dbb0653
...
ea0bc82c49
| Author | SHA1 | Date | |
|---|---|---|---|
|
ea0bc82c49
|
|||
|
15ef8435f6
|
|||
|
4c2cae7149
|
|||
|
ffaf31bbeb
|
|||
|
6ab1aa26b1
|
@@ -13,7 +13,6 @@
|
|||||||
>
|
>
|
||||||
<TaskListComponent :tasks="task.subtasks" />
|
<TaskListComponent :tasks="task.subtasks" />
|
||||||
</q-expansion-item>
|
</q-expansion-item>
|
||||||
<!-- TODO: Add date formatting Mixin? https://jerickson.net/how-to-format-dates-in-vue-3/ -->
|
|
||||||
</q-card>
|
</q-card>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
<template>
|
<template>
|
||||||
<q-form @submit="onSubmit" @reset="onReset" class="q-gutter-md">
|
<q-form @submit="onSubmit" class="q-gutter-md">
|
||||||
<q-input
|
<q-input
|
||||||
filled
|
filled
|
||||||
v-model="modifiedTask.title"
|
v-model="modifiedTask.title"
|
||||||
@@ -52,13 +52,14 @@
|
|||||||
use-input
|
use-input
|
||||||
use-chips
|
use-chips
|
||||||
multiple
|
multiple
|
||||||
|
clearable
|
||||||
|
emit-value
|
||||||
|
map-options
|
||||||
input-debounce="250"
|
input-debounce="250"
|
||||||
@new-value="addTag"
|
|
||||||
:options="skillTagOptions"
|
:options="skillTagOptions"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
option-value="$id"
|
option-value="$id"
|
||||||
@filter="filterSkillTags"
|
@filter="filterSkillTags"
|
||||||
new-value-mode="add-unique"
|
|
||||||
>
|
>
|
||||||
</q-select>
|
</q-select>
|
||||||
</div>
|
</div>
|
||||||
@@ -70,13 +71,14 @@
|
|||||||
use-input
|
use-input
|
||||||
use-chips
|
use-chips
|
||||||
multiple
|
multiple
|
||||||
|
clearable
|
||||||
|
emit-value
|
||||||
|
map-options
|
||||||
input-debounce="250"
|
input-debounce="250"
|
||||||
@new-value="addTag"
|
|
||||||
:options="taskTagOptions"
|
:options="taskTagOptions"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
option-value="$id"
|
option-value="$id"
|
||||||
@filter="filterTaskTags"
|
@filter="filterTaskTags"
|
||||||
new-value-mode="add-unique"
|
|
||||||
>
|
>
|
||||||
</q-select>
|
</q-select>
|
||||||
</div>
|
</div>
|
||||||
@@ -108,6 +110,9 @@
|
|||||||
v-model="modifiedTask.depends_on"
|
v-model="modifiedTask.depends_on"
|
||||||
use-input
|
use-input
|
||||||
multiple
|
multiple
|
||||||
|
clearable
|
||||||
|
emit-value
|
||||||
|
map-options
|
||||||
input-debounce="250"
|
input-debounce="250"
|
||||||
:options="tasks"
|
:options="tasks"
|
||||||
option-label="title"
|
option-label="title"
|
||||||
@@ -122,6 +127,9 @@
|
|||||||
hint="Add a boat, if applicable"
|
hint="Add a boat, if applicable"
|
||||||
v-model="modifiedTask.boat"
|
v-model="modifiedTask.boat"
|
||||||
use-input
|
use-input
|
||||||
|
clearable
|
||||||
|
emit-value
|
||||||
|
map-options
|
||||||
input-debounce="250"
|
input-debounce="250"
|
||||||
:options="boatList"
|
:options="boatList"
|
||||||
option-label="name"
|
option-label="name"
|
||||||
@@ -137,7 +145,6 @@
|
|||||||
flat
|
flat
|
||||||
class="q-ml-sm"
|
class="q-ml-sm"
|
||||||
/>
|
/>
|
||||||
<q-btn label="Reset" type="reset" color="primary" flat class="q-ml-sm" />
|
|
||||||
<q-btn
|
<q-btn
|
||||||
label="Cancel"
|
label="Cancel"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
@@ -150,7 +157,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { reactive, ref, Ref } from 'vue';
|
import { computed, reactive, ref, Ref } from 'vue';
|
||||||
import { useRouter } from 'vue-router';
|
import { useRouter } from 'vue-router';
|
||||||
import { useTaskStore, TASKSTATUS } from 'src/stores/task';
|
import { useTaskStore, TASKSTATUS } from 'src/stores/task';
|
||||||
import type { TaskTag, SkillTag, Task } from 'src/stores/task';
|
import type { TaskTag, SkillTag, Task } from 'src/stores/task';
|
||||||
@@ -173,39 +180,45 @@ const defaultTask = <Task>{
|
|||||||
depends_on: [],
|
depends_on: [],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
taskStore.fetchTasks();
|
||||||
|
|
||||||
const { taskId } = props;
|
const { taskId } = props;
|
||||||
const targetTask = taskId && taskStore.tasks.find((t) => t.$id === taskId);
|
const targetTask = taskId && taskStore.tasks.find((t) => t.$id === taskId);
|
||||||
const modifiedTask = reactive(targetTask ? targetTask : defaultTask);
|
const modifiedTask = reactive(targetTask ? targetTask : defaultTask);
|
||||||
|
|
||||||
taskStore.fetchSkillTags();
|
let tasks = taskStore.tasks;
|
||||||
taskStore.fetchTaskTags();
|
|
||||||
taskStore.fetchTasks();
|
|
||||||
|
|
||||||
const tasks = ref<Task[]>(taskStore.tasks);
|
|
||||||
|
|
||||||
const boatList = ref<Boat[]>(useBoatStore().boats);
|
const boatList = ref<Boat[]>(useBoatStore().boats);
|
||||||
|
|
||||||
const skillTagOptions = ref<SkillTag[]>();
|
const skillTagOptions = ref<SkillTag[]>(taskStore.skillTags);
|
||||||
const taskTagOptions = ref<TaskTag[]>();
|
const taskTagOptions = ref<TaskTag[]>(taskStore.taskTags);
|
||||||
|
|
||||||
function filterSkillTags(val: string, update: (cb: () => void) => void): void {
|
const filterSkillTags = computed(
|
||||||
|
() =>
|
||||||
|
(val: string, update: (cb: () => void) => void): void => {
|
||||||
return filterTags(skillTagOptions, taskStore.skillTags, val, update);
|
return filterTags(skillTagOptions, taskStore.skillTags, val, update);
|
||||||
}
|
}
|
||||||
function filterTaskTags(val: string, update: (cb: () => void) => void): void {
|
);
|
||||||
|
const filterTaskTags = computed(
|
||||||
|
() =>
|
||||||
|
(val: string, update: (cb: () => void) => void): void => {
|
||||||
return filterTags(taskTagOptions, taskStore.taskTags, val, update);
|
return filterTags(taskTagOptions, taskStore.taskTags, val, update);
|
||||||
}
|
}
|
||||||
function filterTasks(val: string, update: (cb: () => void) => void): void {
|
);
|
||||||
|
const filterTasks = computed(
|
||||||
|
() =>
|
||||||
|
(val: string, update: (cb: () => void) => void): void => {
|
||||||
if (val === '') {
|
if (val === '') {
|
||||||
update(() => {
|
update(() => {
|
||||||
tasks.value = taskStore.tasks;
|
tasks = taskStore.tasks;
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
update(() => {
|
update(() => {
|
||||||
tasks.value = taskStore.filterTasks(val);
|
tasks = taskStore.filterTasksByTitle(val);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
);
|
||||||
|
|
||||||
function filterTags(
|
function filterTags(
|
||||||
optionVar: Ref<(SkillTag | TaskTag)[] | undefined>,
|
optionVar: Ref<(SkillTag | TaskTag)[] | undefined>,
|
||||||
@@ -227,9 +240,6 @@ function filterTags(
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function addTag(tag: string) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// Method to update the model in ISO 8601 format
|
// Method to update the model in ISO 8601 format
|
||||||
const updateDateISO = (value: string) => {
|
const updateDateISO = (value: string) => {
|
||||||
modifiedTask.due_date = date.formatDate(value, 'YYYY-MM-DD');
|
modifiedTask.due_date = date.formatDate(value, 'YYYY-MM-DD');
|
||||||
@@ -242,17 +252,18 @@ const dateRule = (val: string) => {
|
|||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
async function onSubmit() {
|
async function onSubmit() {
|
||||||
console.log(modifiedTask);
|
//console.log(modifiedTask);
|
||||||
try {
|
try {
|
||||||
|
if (modifiedTask.$id) {
|
||||||
|
await taskStore.updateTask(modifiedTask);
|
||||||
|
console.log('Updated Task: ' + modifiedTask.$id);
|
||||||
|
} else {
|
||||||
await taskStore.addTask(modifiedTask);
|
await taskStore.addTask(modifiedTask);
|
||||||
console.log('Created Task');
|
console.log('Created Task');
|
||||||
|
}
|
||||||
router.go(-1);
|
router.go(-1);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to create new Task: ', error);
|
console.error('Failed to create new Task: ', error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function onReset() {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -3,30 +3,55 @@
|
|||||||
<q-table
|
<q-table
|
||||||
:rows="tasks"
|
:rows="tasks"
|
||||||
:columns="columns"
|
:columns="columns"
|
||||||
|
:grid="$q.screen.xs"
|
||||||
dense
|
dense
|
||||||
row-key="$id"
|
row-key="$id"
|
||||||
|
flatten
|
||||||
no-data-label="I didn't find anything for you"
|
no-data-label="I didn't find anything for you"
|
||||||
no-results-label="The filter didn't uncover any results"
|
no-results-label="The filter didn't uncover any results"
|
||||||
selection="multiple"
|
selection="multiple"
|
||||||
v-model:selected="selected"
|
v-model:selected="selected"
|
||||||
|
:filter="searchFilter"
|
||||||
|
:filter-method="filterRows"
|
||||||
>
|
>
|
||||||
<template v-slot:top>
|
<template v-slot:top>
|
||||||
<q-btn
|
<q-select
|
||||||
color="primary"
|
style="width: 250px"
|
||||||
:disable="loading"
|
multiple
|
||||||
label="New Task"
|
use-chips
|
||||||
to="/task/edit"
|
clearable
|
||||||
/>
|
label="Skills Filter"
|
||||||
<q-btn
|
input-debounce="250"
|
||||||
v-if="tasks.length !== 0"
|
:options="skillTagOptions"
|
||||||
class="q-ml-sm"
|
v-model="searchFilter.skillTags"
|
||||||
color="primary"
|
option-label="name"
|
||||||
:disable="loading"
|
option-value="$id"
|
||||||
label="Delete task(s)"
|
>
|
||||||
@click="deleteTasks"
|
</q-select>
|
||||||
/>
|
<q-select
|
||||||
|
style="width: 250px"
|
||||||
|
multiple
|
||||||
|
use-chips
|
||||||
|
clearable
|
||||||
|
label="Tag Filter"
|
||||||
|
input-debounce="250"
|
||||||
|
:options="taskTagOptions"
|
||||||
|
v-model="searchFilter.taskTags"
|
||||||
|
option-label="name"
|
||||||
|
option-value="$id"
|
||||||
|
>
|
||||||
|
<template v-slot: prepend>
|
||||||
|
<q-icon name="wrench"></q-icon>
|
||||||
|
</template>
|
||||||
|
</q-select>
|
||||||
<q-space />
|
<q-space />
|
||||||
<q-input borderless debounce="300" color="primary" v-model="filter">
|
<q-input
|
||||||
|
flatten
|
||||||
|
debounce="300"
|
||||||
|
color="primary"
|
||||||
|
clearable
|
||||||
|
v-model="searchFilter.title"
|
||||||
|
>
|
||||||
<template v-slot:append>
|
<template v-slot:append>
|
||||||
<q-icon name="search" />
|
<q-icon name="search" />
|
||||||
</template>
|
</template>
|
||||||
@@ -43,51 +68,160 @@
|
|||||||
</q-tr>
|
</q-tr>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<template v-slot:body="props">
|
<template v-slot:body-cell-skills="props">
|
||||||
<q-tr :props="props">
|
<q-td :props="props" class="q-gutter-sm">
|
||||||
<q-td key="desc" auto-width>
|
|
||||||
<q-checkbox dense v-model="props.selected"></q-checkbox>
|
|
||||||
</q-td>
|
|
||||||
<q-td v-for="col in props.cols" :key="col.name" :props="props">
|
|
||||||
<div v-if="col.name == 'skills'" class="q-gutter-sm">
|
|
||||||
<q-badge
|
<q-badge
|
||||||
v-for="skill in props.row.required_skills"
|
v-for="skill in props.value"
|
||||||
:key="skill"
|
:key="skill"
|
||||||
:color="skill.tagColour"
|
:color="skill.tagColour"
|
||||||
text-color="white"
|
text-color="white"
|
||||||
>
|
:label="skill.name"
|
||||||
{{ skill.name }}
|
/>
|
||||||
</q-badge>
|
</q-td>
|
||||||
</div>
|
</template>
|
||||||
<div v-else-if="col.name == 'tags'" class="q-gutter-sm">
|
|
||||||
|
<template v-slot:body-cell-tags="props">
|
||||||
|
<q-td :props="props" class="q-gutter-sm">
|
||||||
<q-badge
|
<q-badge
|
||||||
v-for="tag in props.row.tags"
|
v-for="tag in props.value"
|
||||||
:key="tag"
|
:key="tag"
|
||||||
:color="tag.colour"
|
:color="tag.colour"
|
||||||
text-color="white"
|
text-color="white"
|
||||||
>
|
:label="tag.name"
|
||||||
{{ tag.name }}
|
/>
|
||||||
</q-badge>
|
|
||||||
</div>
|
|
||||||
<div v-else>
|
|
||||||
{{ col.value }}
|
|
||||||
</div>
|
|
||||||
</q-td>
|
</q-td>
|
||||||
</q-tr>
|
</template>
|
||||||
|
<template v-slot:body-cell-actions="props">
|
||||||
|
<q-td :props="props" class="q-gutter-sm">
|
||||||
|
<q-btn
|
||||||
|
label="Sign Up"
|
||||||
|
size="sm"
|
||||||
|
color="primary"
|
||||||
|
:to="{ name: 'signup-task', params: { id: props.value } }"
|
||||||
|
/>
|
||||||
|
<q-btn
|
||||||
|
label="Edit"
|
||||||
|
size="sm"
|
||||||
|
color="primary"
|
||||||
|
:to="{ name: 'edit-task', params: { id: props.value } }"
|
||||||
|
/>
|
||||||
|
</q-td>
|
||||||
|
</template>
|
||||||
|
<template v-slot:item="props">
|
||||||
|
<div
|
||||||
|
class="q-pa-xs col-xs-12 col-sm-6 col-md-4 col-lg-3 grid-style-transition"
|
||||||
|
:style="props.selected ? 'transform: scale(0.95);' : ''"
|
||||||
|
>
|
||||||
|
<q-card
|
||||||
|
bordered
|
||||||
|
flat
|
||||||
|
:class="
|
||||||
|
props.selected
|
||||||
|
? $q.dark.isActive
|
||||||
|
? 'bg-grey-9'
|
||||||
|
: 'bg-grey-2'
|
||||||
|
: ''
|
||||||
|
"
|
||||||
|
>
|
||||||
|
<q-card-section>
|
||||||
|
<q-checkbox
|
||||||
|
dense
|
||||||
|
v-model="props.selected"
|
||||||
|
:label="props.row.name"
|
||||||
|
/>
|
||||||
|
</q-card-section>
|
||||||
|
<q-separator />
|
||||||
|
<q-list dense>
|
||||||
|
<q-item
|
||||||
|
v-for="col in props.cols.filter((col) => col.name !== 'desc')"
|
||||||
|
:key="col.name"
|
||||||
|
>
|
||||||
|
<q-item-section>
|
||||||
|
<q-item-label>{{ col.label }}</q-item-label>
|
||||||
|
</q-item-section>
|
||||||
|
<q-item-section side>
|
||||||
|
<q-item-label caption v-if="col.name === 'skills'">
|
||||||
|
<q-chip
|
||||||
|
size="sm"
|
||||||
|
v-for="skill in col.value"
|
||||||
|
outline
|
||||||
|
color="primary"
|
||||||
|
:key="skill.$id"
|
||||||
|
>{{ skill.name }}</q-chip
|
||||||
|
></q-item-label
|
||||||
|
>
|
||||||
|
<q-item-label caption v-else-if="col.name === 'tags'">
|
||||||
|
<q-chip
|
||||||
|
size="sm"
|
||||||
|
v-for="tag in col.value"
|
||||||
|
outline
|
||||||
|
color="primary"
|
||||||
|
:key="tag.$id"
|
||||||
|
>{{ tag.name }}</q-chip
|
||||||
|
></q-item-label
|
||||||
|
>
|
||||||
|
<q-item-label caption v-else-if="col.name === 'actions'">
|
||||||
|
<q-btn
|
||||||
|
label="Sign Up"
|
||||||
|
size="sm"
|
||||||
|
color="primary"
|
||||||
|
:to="{ name: 'signup-task', params: { id: col.value } }"
|
||||||
|
/>
|
||||||
|
<q-btn
|
||||||
|
label="Edit"
|
||||||
|
size="sm"
|
||||||
|
color="primary"
|
||||||
|
:to="{ name: 'edit-task', params: { id: col.value } }"
|
||||||
|
/>
|
||||||
|
</q-item-label>
|
||||||
|
<q-item-label v-else caption>{{ col.value }}</q-item-label>
|
||||||
|
</q-item-section>
|
||||||
|
</q-item>
|
||||||
|
</q-list>
|
||||||
|
</q-card>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</q-table>
|
</q-table>
|
||||||
|
<q-page-sticky position="bottom-right" :offset="[18, 18]">
|
||||||
|
<q-fab
|
||||||
|
v-model="fabShow"
|
||||||
|
vertical-actions-align="right"
|
||||||
|
color="primary"
|
||||||
|
glossy
|
||||||
|
icon="keyboard_arrow_up"
|
||||||
|
direction="up"
|
||||||
|
>
|
||||||
|
<q-fab-action
|
||||||
|
color="primary"
|
||||||
|
:disable="loading"
|
||||||
|
label="New Task"
|
||||||
|
to="/task/edit"
|
||||||
|
icon="add"
|
||||||
|
/>
|
||||||
|
<q-fab-action
|
||||||
|
v-if="tasks.length !== 0"
|
||||||
|
class="q-ml-sm"
|
||||||
|
color="primary"
|
||||||
|
:disable="loading"
|
||||||
|
label="Delete task(s)"
|
||||||
|
@click="deleteTasks"
|
||||||
|
icon="delete"
|
||||||
|
/> </q-fab
|
||||||
|
></q-page-sticky>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { defineProps, ref } from 'vue';
|
import { computed, defineProps, ref } from 'vue';
|
||||||
import { useTaskStore, Task } from 'src/stores/task';
|
import { useTaskStore, Task, SkillTag, TaskTag } from 'src/stores/task';
|
||||||
import { QTableProps, date, useQuasar } from 'quasar';
|
import { QTableProps, date, useQuasar } from 'quasar';
|
||||||
import { useBoatStore } from 'src/stores/boat';
|
import { useBoatStore } from 'src/stores/boat';
|
||||||
import BoatPickerComponent from '../boat/BoatPickerComponent.vue';
|
import { useRouter } from 'vue-router';
|
||||||
|
|
||||||
|
const router = useRouter();
|
||||||
const selected = ref([]);
|
const selected = ref([]);
|
||||||
const loading = ref(false); // Placeholder
|
const loading = ref(false); // Placeholder
|
||||||
|
const fabShow = ref(false);
|
||||||
const columns = <QTableProps['columns']>[
|
const columns = <QTableProps['columns']>[
|
||||||
{
|
{
|
||||||
name: 'title',
|
name: 'title',
|
||||||
@@ -116,14 +250,15 @@ const columns = <QTableProps['columns']>[
|
|||||||
name: 'skills',
|
name: 'skills',
|
||||||
align: 'left',
|
align: 'left',
|
||||||
label: 'Skills',
|
label: 'Skills',
|
||||||
field: 'required_skills',
|
field: (row) =>
|
||||||
|
row.required_skills.map((s: string) => taskStore.getSkillById(s)),
|
||||||
sortable: false,
|
sortable: false,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'tags',
|
name: 'tags',
|
||||||
align: 'left',
|
align: 'left',
|
||||||
label: 'Tags',
|
label: 'Tags',
|
||||||
field: 'tags',
|
field: (row) => row.tags.map((s: string) => taskStore.getTaskTagById(s)),
|
||||||
sortable: false,
|
sortable: false,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -132,7 +267,7 @@ const columns = <QTableProps['columns']>[
|
|||||||
label: 'Boat',
|
label: 'Boat',
|
||||||
field: (row) =>
|
field: (row) =>
|
||||||
useBoatStore().boats.find((boat) => boat.$id === row.boat)?.name,
|
useBoatStore().boats.find((boat) => boat.$id === row.boat)?.name,
|
||||||
sortable: false,
|
sortable: true,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: 'volunteers',
|
name: 'volunteers',
|
||||||
@@ -156,24 +291,56 @@ const columns = <QTableProps['columns']>[
|
|||||||
format: (val) => {
|
format: (val) => {
|
||||||
return (
|
return (
|
||||||
val
|
val
|
||||||
.map((t: string) => lookupTaskFromId(t))
|
.map((t: string) => taskStore.getTaskById(t))
|
||||||
.filter((t: Task) => t)
|
.filter((t: Task) => t)
|
||||||
.map((t: Task) => t.title)
|
.map((t: Task) => t.title)
|
||||||
.join(', ') || null
|
.join(', ') || null
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
{ name: 'actions', align: 'center', label: 'Actions', field: '$id' },
|
||||||
];
|
];
|
||||||
|
|
||||||
const props = defineProps<{ tasks: Task[] }>();
|
const props = defineProps<{ tasks: Task[] }>();
|
||||||
const taskStore = useTaskStore();
|
const taskStore = useTaskStore();
|
||||||
const $q = useQuasar();
|
const $q = useQuasar();
|
||||||
taskStore.fetchTaskTags();
|
|
||||||
taskStore.fetchSkillTags();
|
|
||||||
|
|
||||||
function lookupTaskFromId(id: string): Task {
|
const searchFilter = ref({
|
||||||
return taskStore.tasks.find((t) => t.$id === id) || undefined;
|
title: '',
|
||||||
|
skillTags: <SkillTag[]>[],
|
||||||
|
taskTags: <TaskTag[]>[],
|
||||||
|
});
|
||||||
|
|
||||||
|
const skillTagOptions = ref<SkillTag[]>(taskStore.skillTags);
|
||||||
|
const taskTagOptions = ref<TaskTag[]>(taskStore.taskTags);
|
||||||
|
|
||||||
|
function onRowClick(evt: Event, row: Task) {
|
||||||
|
router.push({ name: 'edit-task', params: { id: row.$id } });
|
||||||
}
|
}
|
||||||
|
// TODO: Implement server side search
|
||||||
|
const filterRows = computed(
|
||||||
|
() => (rows: readonly Task[], terms: any, cols: any, cellValueFn: any) => {
|
||||||
|
let result = rows;
|
||||||
|
result = rows.filter((row) =>
|
||||||
|
terms.title
|
||||||
|
? row.title.toLowerCase().includes(terms.title.toLowerCase())
|
||||||
|
: true
|
||||||
|
);
|
||||||
|
result = result.filter((row) =>
|
||||||
|
terms.skillTags && terms.skillTags.length > 0
|
||||||
|
? row.required_skills.some((req_skill) =>
|
||||||
|
terms.skillTags.map((t) => t.$id).includes(req_skill)
|
||||||
|
)
|
||||||
|
: true
|
||||||
|
);
|
||||||
|
result = result.filter((row) =>
|
||||||
|
terms.taskTags && terms.taskTags.length > 0
|
||||||
|
? row.tags.some((tag) => terms.taskTags.map((t) => t.$id).includes(tag))
|
||||||
|
: true
|
||||||
|
);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
function deleteTasks() {
|
function deleteTasks() {
|
||||||
confirmDelete(selected.value);
|
confirmDelete(selected.value);
|
||||||
@@ -192,5 +359,4 @@ function confirmDelete(tasks: Task[]) {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
const filter = ref('');
|
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -2,12 +2,15 @@
|
|||||||
<ToolbarComponent pageTitle="Tasks" />
|
<ToolbarComponent pageTitle="Tasks" />
|
||||||
<q-page padding>
|
<q-page padding>
|
||||||
<div class="q-pa-md" style="max-width: 400px">
|
<div class="q-pa-md" style="max-width: 400px">
|
||||||
<TaskEditComponent />
|
<TaskEditComponent :taskId="taskId" />
|
||||||
</div>
|
</div>
|
||||||
</q-page>
|
</q-page>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
const taskId = useRoute().params.id as string;
|
||||||
|
console.log(taskId);
|
||||||
import ToolbarComponent from 'src/components/ToolbarComponent.vue';
|
import ToolbarComponent from 'src/components/ToolbarComponent.vue';
|
||||||
import TaskEditComponent from 'src/components/task/TaskEditComponent.vue';
|
import TaskEditComponent from 'src/components/task/TaskEditComponent.vue';
|
||||||
|
import { useRoute } from 'vue-router';
|
||||||
</script>
|
</script>
|
||||||
|
|||||||
@@ -1,15 +1,13 @@
|
|||||||
<template>
|
<template>
|
||||||
<toolbar-component pageTitle="Tasks" />
|
<toolbar-component pageTitle="Tasks" />
|
||||||
<q-page padding>
|
<q-page padding>
|
||||||
<TaskListComponent v-if="$q.screen.lt.sm" :tasks="taskStore.tasks" />
|
<TaskTableComponent :tasks="taskStore.tasks" />
|
||||||
<TaskTableComponent v-else :tasks="taskStore.tasks" />
|
|
||||||
</q-page>
|
</q-page>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { useTaskStore } from 'stores/task';
|
import { useTaskStore } from 'stores/task';
|
||||||
import ToolbarComponent from 'src/components/ToolbarComponent.vue';
|
import ToolbarComponent from 'src/components/ToolbarComponent.vue';
|
||||||
import TaskListComponent from 'src/components/task/TaskListComponent.vue';
|
|
||||||
import TaskTableComponent from 'src/components/task/TaskTableComponent.vue';
|
import TaskTableComponent from 'src/components/task/TaskTableComponent.vue';
|
||||||
|
|
||||||
const taskStore = useTaskStore();
|
const taskStore = useTaskStore();
|
||||||
|
|||||||
@@ -57,7 +57,7 @@ const routes: RouteRecordRaw[] = [
|
|||||||
name: 'task-index',
|
name: 'task-index',
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: 'edit',
|
path: '/:id/edit',
|
||||||
component: () => import('pages/task/TaskEditPage.vue'),
|
component: () => import('pages/task/TaskEditPage.vue'),
|
||||||
name: 'edit-task',
|
name: 'edit-task',
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,23 +1,27 @@
|
|||||||
import { defineStore } from 'pinia';
|
import { defineStore } from 'pinia';
|
||||||
import { AppwriteIds, databases, ID } from 'src/boot/appwrite';
|
import { AppwriteIds, databases, ID } from 'src/boot/appwrite';
|
||||||
import type { Models } from 'appwrite';
|
import type { Models } from 'appwrite';
|
||||||
import { Boat } from './boat';
|
|
||||||
|
|
||||||
export const TASKSTATUS = ['ready', 'complete', 'waiting', 'archived'];
|
export const TASKSTATUS = ['ready', 'complete', 'waiting', 'archived'];
|
||||||
|
|
||||||
export interface Task extends Partial<Models.Document> {
|
export interface Task extends Partial<Models.Document> {
|
||||||
title: string;
|
title: string;
|
||||||
description: string;
|
description: string;
|
||||||
required_skills: SkillTag[];
|
/* Array of Appwrite Document IDs */
|
||||||
tags: TaskTag[];
|
required_skills: string[];
|
||||||
|
/* Array of Appwrite Document IDs */
|
||||||
|
tags: string[];
|
||||||
due_date: string;
|
due_date: string;
|
||||||
duration: number;
|
duration: number;
|
||||||
|
/* Array of Appwrite Document IDs */
|
||||||
volunteers: string[];
|
volunteers: string[];
|
||||||
volunteers_required: number;
|
volunteers_required: number;
|
||||||
status: string;
|
status: string;
|
||||||
depends_on: Task[];
|
/* Array of Appwrite Document IDs */
|
||||||
boat?: Boat;
|
depends_on: string[];
|
||||||
} // TODO: convert some of these strings into objects.
|
/* Appwrite ID of a Boat resource */
|
||||||
|
boat?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
export interface TaskTag extends Models.Document {
|
export interface TaskTag extends Models.Document {
|
||||||
name: string;
|
name: string;
|
||||||
@@ -46,24 +50,7 @@ export const useTaskStore = defineStore('tasks', {
|
|||||||
AppwriteIds.databaseId,
|
AppwriteIds.databaseId,
|
||||||
AppwriteIds.collectionIdTask
|
AppwriteIds.collectionIdTask
|
||||||
);
|
);
|
||||||
this.tasks = response.documents.map((document) => {
|
this.tasks = response.documents as Task[];
|
||||||
// TODO: Should this be a GraphQL query, instead?
|
|
||||||
// Map over `required_skills` and replace each skill ID with the corresponding skill object from `this.skillTags`
|
|
||||||
const updatedRequiredSkills = document.required_skills.map(
|
|
||||||
(skillId: string) =>
|
|
||||||
this.skillTags.find((skillTag) => skillTag.$id === skillId) || {}
|
|
||||||
);
|
|
||||||
const updatedTaskTags = document.tags.map((tagid: string) =>
|
|
||||||
this.taskTags.find((taskTag) => taskTag.$id === tagid)
|
|
||||||
);
|
|
||||||
|
|
||||||
// Update the `required_skills` property of the document with the new array of skill objects
|
|
||||||
return {
|
|
||||||
...document,
|
|
||||||
required_skills: updatedRequiredSkills,
|
|
||||||
tags: updatedTaskTags,
|
|
||||||
};
|
|
||||||
}) as Task[];
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to fetch tasks', error);
|
console.error('Failed to fetch tasks', error);
|
||||||
}
|
}
|
||||||
@@ -112,10 +99,6 @@ export const useTaskStore = defineStore('tasks', {
|
|||||||
},
|
},
|
||||||
async addTask(task: Task) {
|
async addTask(task: Task) {
|
||||||
const newTask = <Models.Document>{ ...task };
|
const newTask = <Models.Document>{ ...task };
|
||||||
newTask.required_skills = task.required_skills.map((s) => s['$id']);
|
|
||||||
newTask.tags = task.tags.map((s) => s['$id']);
|
|
||||||
newTask.depends_on = task.depends_on.map((d) => d['$id']);
|
|
||||||
newTask.boat = task.boat?.$id;
|
|
||||||
try {
|
try {
|
||||||
const response = await databases.createDocument(
|
const response = await databases.createDocument(
|
||||||
AppwriteIds.databaseId,
|
AppwriteIds.databaseId,
|
||||||
@@ -128,32 +111,48 @@ export const useTaskStore = defineStore('tasks', {
|
|||||||
console.error('Failed to add task:', error);
|
console.error('Failed to add task:', error);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
async updateTask(task: Task) {
|
||||||
|
const newTask = <Partial<Models.Document>>{
|
||||||
|
...task,
|
||||||
|
id: undefined,
|
||||||
|
$databaseId: undefined,
|
||||||
|
$collectionId: undefined,
|
||||||
|
};
|
||||||
|
if (!task.$id) {
|
||||||
|
console.error('No Task ID!');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
const response = await databases.updateDocument(
|
||||||
|
AppwriteIds.databaseId,
|
||||||
|
AppwriteIds.collectionIdTask,
|
||||||
|
task.$id,
|
||||||
|
newTask
|
||||||
|
);
|
||||||
|
this.tasks.push(response as Task);
|
||||||
|
} catch (error) {
|
||||||
|
console.error('Failed to update task:', error);
|
||||||
|
}
|
||||||
|
},
|
||||||
// TODO: Enhance this store to include offline caching, and subscription notification when items change on the server.
|
// TODO: Enhance this store to include offline caching, and subscription notification when items change on the server.
|
||||||
|
},
|
||||||
filterTasks(searchQuery: string) {
|
// Add more actions as needed (e.g., updateTask, deleteTask)
|
||||||
const result = this.tasks.filter((task) =>
|
getters: {
|
||||||
|
getTaskById: (state) => (id: string) => {
|
||||||
|
return state.tasks.find((task) => task.$id === id) || null;
|
||||||
|
},
|
||||||
|
getTaskTagById: (state) => (id: string) => {
|
||||||
|
return state.taskTags.find((tag) => tag.$id === id) || null;
|
||||||
|
},
|
||||||
|
getSkillById: (state) => (id: string) => {
|
||||||
|
return state.skillTags.find((tag) => tag.$id === id) || null;
|
||||||
|
},
|
||||||
|
filterTasksByTitle: (state) => (searchQuery: string) => {
|
||||||
|
const result = state.tasks.filter((task) =>
|
||||||
task.title.toLowerCase().includes(searchQuery.toLowerCase())
|
task.title.toLowerCase().includes(searchQuery.toLowerCase())
|
||||||
);
|
);
|
||||||
console.log(result);
|
console.log(result);
|
||||||
return result;
|
return result;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
// Add more actions as needed (e.g., updateTask, deleteTask)
|
|
||||||
getters: {
|
|
||||||
// A getter to reconstruct the hierarchical structure from flat task data
|
|
||||||
taskHierarchy: (state) => {
|
|
||||||
function buildHierarchy(
|
|
||||||
tasks: Task[],
|
|
||||||
parentId: string | null = null
|
|
||||||
): Task[] {
|
|
||||||
return tasks
|
|
||||||
.filter((task) => task.parentId === parentId)
|
|
||||||
.map((task) => ({
|
|
||||||
...task,
|
|
||||||
subtasks: buildHierarchy(tasks, task.$id), // Assuming $id is the task ID field
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
return buildHierarchy(state.tasks);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user