7aede352f1
Co-authored-by: kolaente <k@knt.li> Reviewed-on: https://kolaente.dev/vikunja/frontend/pulls/920 Reviewed-by: dpschen <dpschen@noreply.kolaente.de> Co-authored-by: konrad <k@knt.li> Co-committed-by: konrad <k@knt.li>
359 lines
No EOL
8.9 KiB
JavaScript
359 lines
No EOL
8.9 KiB
JavaScript
import cloneDeep from 'lodash.clonedeep'
|
|
|
|
import {findById, findIndexById} from '@/helpers/utils'
|
|
import {i18n} from '@/i18n'
|
|
import {success} from '@/message'
|
|
|
|
import BucketService from '../../services/bucket'
|
|
import {setLoading} from '../helper'
|
|
import TaskCollectionService from '@/services/taskCollection'
|
|
|
|
const TASKS_PER_BUCKET = 25
|
|
|
|
function getTaskIndicesById(state, taskId) {
|
|
let taskIndex
|
|
const bucketIndex = state.buckets.findIndex(({ tasks }) => {
|
|
taskIndex = findIndexById(tasks, taskId)
|
|
return taskIndex !== -1
|
|
})
|
|
|
|
return {
|
|
bucketIndex: bucketIndex !== -1 ? bucketIndex : null,
|
|
taskIndex: taskIndex !== -1 ? taskIndex : null,
|
|
}
|
|
}
|
|
|
|
const addTaskToBucketAndSort = (state, task) => {
|
|
const bucketIndex = findIndexById(state.buckets, task.bucketId)
|
|
state.buckets[bucketIndex].tasks.push(task)
|
|
state.buckets[bucketIndex].tasks.sort((a, b) => a.kanbanPosition > b.kanbanPosition ? 1 : -1)
|
|
}
|
|
|
|
/**
|
|
* This store is intended to hold the currently active kanban view.
|
|
* It should hold only the current buckets.
|
|
*/
|
|
export default {
|
|
namespaced: true,
|
|
|
|
state: () => ({
|
|
buckets: [],
|
|
listId: 0,
|
|
bucketLoading: {},
|
|
taskPagesPerBucket: {},
|
|
allTasksLoadedForBucket: {},
|
|
}),
|
|
|
|
mutations: {
|
|
setListId(state, listId) {
|
|
state.listId = parseInt(listId)
|
|
},
|
|
|
|
setBuckets(state, buckets) {
|
|
state.buckets = buckets
|
|
buckets.forEach(b => {
|
|
state.taskPagesPerBucket[b.id] = 1
|
|
state.allTasksLoadedForBucket[b.id] = false
|
|
})
|
|
},
|
|
|
|
addBucket(state, bucket) {
|
|
state.buckets.push(bucket)
|
|
},
|
|
|
|
removeBucket(state, bucket) {
|
|
const bucketIndex = findIndexById(state.buckets, bucket.id)
|
|
state.buckets.splice(bucketIndex, 1)
|
|
},
|
|
|
|
setBucketById(state, bucket) {
|
|
const bucketIndex = findIndexById(state.buckets, bucket.id)
|
|
state.buckets[bucketIndex] = bucket
|
|
},
|
|
|
|
setBucketByIndex(state, {bucketIndex, bucket}) {
|
|
state.buckets[bucketIndex] = bucket
|
|
},
|
|
|
|
setTaskInBucketByIndex(state, {bucketIndex, taskIndex, task}) {
|
|
const bucket = state.buckets[bucketIndex]
|
|
bucket.tasks[taskIndex] = task
|
|
state.buckets[bucketIndex] = bucket
|
|
},
|
|
|
|
setTasksInBucketByBucketId(state, {bucketId, tasks}) {
|
|
const bucketIndex = findIndexById(state.buckets, bucketId)
|
|
state.buckets[bucketIndex] = {
|
|
...state.buckets[bucketIndex],
|
|
tasks,
|
|
}
|
|
},
|
|
|
|
setTaskInBucket(state, task) {
|
|
// If this gets invoked without any tasks actually loaded, we can save the hassle of finding the task
|
|
if (state.buckets.length === 0) {
|
|
return
|
|
}
|
|
|
|
let found = false
|
|
|
|
const findAndUpdate = b => {
|
|
for (const t in state.buckets[b].tasks) {
|
|
if (state.buckets[b].tasks[t].id === task.id) {
|
|
const bucket = state.buckets[b]
|
|
bucket.tasks[t] = task
|
|
|
|
if (bucket.id !== task.bucketId) {
|
|
bucket.tasks.splice(t, 1)
|
|
addTaskToBucketAndSort(state, task)
|
|
}
|
|
|
|
state.buckets[b] = bucket
|
|
|
|
found = true
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const b in state.buckets) {
|
|
if (state.buckets[b].id === task.bucketId) {
|
|
findAndUpdate(b)
|
|
if (found) {
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const b in state.buckets) {
|
|
findAndUpdate(b)
|
|
if (found) {
|
|
return
|
|
}
|
|
}
|
|
},
|
|
|
|
addTaskToBucket(state, task) {
|
|
const bucketIndex = findIndexById(state.buckets, task.bucketId)
|
|
const oldBucket = state.buckets[bucketIndex]
|
|
const newBucket = {
|
|
...oldBucket,
|
|
tasks: [
|
|
...oldBucket.tasks,
|
|
task,
|
|
],
|
|
}
|
|
state.buckets[bucketIndex] = newBucket
|
|
},
|
|
|
|
addTasksToBucket(state, {tasks, bucketId}) {
|
|
const bucketIndex = findIndexById(state.buckets, bucketId)
|
|
const oldBucket = state.buckets[bucketIndex]
|
|
const newBucket = {
|
|
...oldBucket,
|
|
tasks: [
|
|
...oldBucket.tasks,
|
|
...tasks,
|
|
],
|
|
}
|
|
state.buckets[bucketIndex] = newBucket
|
|
},
|
|
|
|
removeTaskInBucket(state, task) {
|
|
// If this gets invoked without any tasks actually loaded, we can save the hassle of finding the task
|
|
if (state.buckets.length === 0) {
|
|
return
|
|
}
|
|
|
|
const { bucketIndex, taskIndex } = getTaskIndicesById(state, task.id)
|
|
|
|
if (
|
|
state.buckets[bucketIndex]?.id !== task.bucketId ||
|
|
state.buckets[bucketIndex]?.tasks[taskIndex]?.id !== task.id
|
|
) {
|
|
return
|
|
}
|
|
|
|
state.buckets[bucketIndex].tasks.splice(taskIndex, 1)
|
|
},
|
|
|
|
setBucketLoading(state, {bucketId, loading}) {
|
|
state.bucketLoading[bucketId] = loading
|
|
},
|
|
|
|
setTasksLoadedForBucketPage(state, {bucketId, page}) {
|
|
state.taskPagesPerBucket[bucketId] = page
|
|
},
|
|
|
|
setAllTasksLoadedForBucket(state, bucketId) {
|
|
state.allTasksLoadedForBucket[bucketId] = true
|
|
},
|
|
},
|
|
|
|
getters: {
|
|
getBucketById(state) {
|
|
return (bucketId) => findById(state.buckets, bucketId)
|
|
},
|
|
|
|
getTaskById(state) {
|
|
return (id) => {
|
|
const { bucketIndex, taskIndex } = getTaskIndicesById(state, id)
|
|
|
|
return {
|
|
bucketIndex,
|
|
taskIndex,
|
|
task: state.buckets[bucketIndex]?.tasks?.[taskIndex] || null,
|
|
}
|
|
}
|
|
},
|
|
},
|
|
|
|
actions: {
|
|
async loadBucketsForList(ctx, {listId, params}) {
|
|
const cancel = setLoading(ctx, 'kanban')
|
|
|
|
// Clear everything to prevent having old buckets in the list if loading the buckets from this list takes a few moments
|
|
ctx.commit('setBuckets', [])
|
|
|
|
params.per_page = TASKS_PER_BUCKET
|
|
|
|
const bucketService = new BucketService()
|
|
try {
|
|
const response = await bucketService.getAll({listId: listId}, params)
|
|
ctx.commit('setBuckets', response)
|
|
ctx.commit('setListId', listId)
|
|
return response
|
|
} finally {
|
|
cancel()
|
|
}
|
|
},
|
|
|
|
async loadNextTasksForBucket(ctx, {listId, ps = {}, bucketId}) {
|
|
const isLoading = ctx.state.bucketLoading[bucketId] ?? false
|
|
if (isLoading) {
|
|
return
|
|
}
|
|
|
|
const page = (ctx.state.taskPagesPerBucket[bucketId] ?? 1) + 1
|
|
|
|
const alreadyLoaded = ctx.state.allTasksLoadedForBucket[bucketId] ?? false
|
|
if (alreadyLoaded) {
|
|
return
|
|
}
|
|
|
|
const cancel = setLoading(ctx, 'kanban')
|
|
ctx.commit('setBucketLoading', {bucketId: bucketId, loading: true})
|
|
|
|
const params = JSON.parse(JSON.stringify(ps))
|
|
|
|
params.sort_by = 'kanban_position'
|
|
params.order_by = 'asc'
|
|
|
|
let hasBucketFilter = false
|
|
for (const f in params.filter_by) {
|
|
if (params.filter_by[f] === 'bucket_id') {
|
|
hasBucketFilter = true
|
|
if (params.filter_value[f] !== bucketId) {
|
|
params.filter_value[f] = bucketId
|
|
}
|
|
break
|
|
}
|
|
}
|
|
|
|
if (!hasBucketFilter) {
|
|
params.filter_by = [...(params.filter_by ?? []), 'bucket_id']
|
|
params.filter_value = [...(params.filter_value ?? []), bucketId]
|
|
params.filter_comparator = [...(params.filter_comparator ?? []), 'equals']
|
|
}
|
|
|
|
params.per_page = TASKS_PER_BUCKET
|
|
|
|
const taskService = new TaskCollectionService()
|
|
try {
|
|
const tasks = await taskService.getAll({listId: listId}, params, page)
|
|
ctx.commit('addTasksToBucket', {tasks, bucketId: bucketId})
|
|
ctx.commit('setTasksLoadedForBucketPage', {bucketId, page})
|
|
if (taskService.totalPages <= page) {
|
|
ctx.commit('setAllTasksLoadedForBucket', bucketId)
|
|
}
|
|
return tasks
|
|
} finally {
|
|
cancel()
|
|
ctx.commit('setBucketLoading', {bucketId, loading: false})
|
|
}
|
|
},
|
|
|
|
async createBucket(ctx, bucket) {
|
|
const cancel = setLoading(ctx, 'kanban')
|
|
|
|
const bucketService = new BucketService()
|
|
try {
|
|
const createdBucket = await bucketService.create(bucket)
|
|
ctx.commit('addBucket', createdBucket)
|
|
return createdBucket
|
|
} finally {
|
|
cancel()
|
|
}
|
|
},
|
|
|
|
async deleteBucket(ctx, {bucket, params}) {
|
|
const cancel = setLoading(ctx, 'kanban')
|
|
|
|
const bucketService = new BucketService()
|
|
try {
|
|
const response = await bucketService.delete(bucket)
|
|
ctx.commit('removeBucket', bucket)
|
|
// We reload all buckets because tasks are being moved from the deleted bucket
|
|
ctx.dispatch('loadBucketsForList', {listId: bucket.listId, params: params})
|
|
return response
|
|
} finally {
|
|
cancel()
|
|
}
|
|
},
|
|
|
|
async updateBucket(ctx, updatedBucketData) {
|
|
const cancel = setLoading(ctx, 'kanban')
|
|
|
|
const bucketIndex = findIndexById(ctx.state.buckets, updatedBucketData.id)
|
|
const oldBucket = cloneDeep(ctx.state.buckets[bucketIndex])
|
|
|
|
const updatedBucket = {
|
|
...ctx.state.buckets[bucketIndex],
|
|
...updatedBucketData,
|
|
}
|
|
|
|
ctx.commit('setBucketByIndex', {bucketIndex, bucket: updatedBucket})
|
|
|
|
const bucketService = new BucketService()
|
|
try {
|
|
const returnedBucket = await bucketService.update(updatedBucket)
|
|
ctx.commit('setBucketByIndex', {bucketIndex, bucket: returnedBucket})
|
|
return returnedBucket
|
|
} catch(e) {
|
|
// restore original state
|
|
ctx.commit('setBucketByIndex', {bucketIndex, bucket: oldBucket})
|
|
|
|
throw e
|
|
} finally {
|
|
cancel()
|
|
}
|
|
},
|
|
|
|
async updateBucketTitle(ctx, { id, title }) {
|
|
const bucket = findById(ctx.state.buckets, id)
|
|
|
|
if (bucket.title === title) {
|
|
// bucket title has not changed
|
|
return
|
|
}
|
|
|
|
const updatedBucketData = {
|
|
id,
|
|
title,
|
|
}
|
|
|
|
await ctx.dispatch('updateBucket', updatedBucketData)
|
|
success({message: i18n.global.t('list.kanban.bucketTitleSavedSuccess')})
|
|
},
|
|
},
|
|
} |