import { sortBy } from 'lodash'; import { getIdFromGraphQLId } from '~/graphql_shared/utils'; import { ListType, NOT_FILTER } from './constants'; export function getMilestone() { return null; } export function updateListPosition(listObj) { const { listType } = listObj; let { position } = listObj; if (listType === ListType.closed) { position = Infinity; } else if (listType === ListType.backlog) { position = -Infinity; } return { ...listObj, position }; } export function formatBoardLists(lists) { return lists.nodes.reduce((map, list) => { return { ...map, [list.id]: updateListPosition(list), }; }, {}); } export function formatIssue(issue) { return { ...issue, labels: issue.labels?.nodes || [], assignees: issue.assignees?.nodes || [], }; } export function formatListIssues(listIssues) { const issues = {}; let listIssuesCount; const listData = listIssues.nodes.reduce((map, list) => { listIssuesCount = list.issues.count; let sortedIssues = list.issues.edges.map((issueNode) => ({ ...issueNode.node, })); sortedIssues = sortBy(sortedIssues, 'relativePosition'); return { ...map, [list.id]: sortedIssues.map((i) => { const id = getIdFromGraphQLId(i.id); const listIssue = { ...i, id, labels: i.labels?.nodes || [], assignees: i.assignees?.nodes || [], }; issues[id] = listIssue; return id; }), }; }, {}); return { listData, issues, listIssuesCount }; } export function formatListsPageInfo(lists) { const listData = lists.nodes.reduce((map, list) => { return { ...map, [list.id]: list.issues.pageInfo, }; }, {}); return listData; } export function fullBoardId(boardId) { return `gid://gitlab/Board/${boardId}`; } export function fullIterationId(id) { return `gid://gitlab/Iteration/${id}`; } export function fullUserId(id) { return `gid://gitlab/User/${id}`; } export function fullMilestoneId(id) { return `gid://gitlab/Milestone/${id}`; } export function fullLabelId(label) { if (label.project_id && label.project_id !== null) { return `gid://gitlab/ProjectLabel/${label.id}`; } return `gid://gitlab/GroupLabel/${label.id}`; } export function formatIssueInput(issueInput, boardConfig) { const { labelIds = [], assigneeIds = [] } = issueInput; const { labels, assigneeId, milestoneId } = boardConfig; return { milestoneId: milestoneId ? fullMilestoneId(milestoneId) : null, ...issueInput, labelIds: [...labelIds, ...(labels?.map((l) => fullLabelId(l)) || [])], assigneeIds: [...assigneeIds, ...(assigneeId ? [fullUserId(assigneeId)] : [])], }; } export function moveIssueListHelper(issue, fromList, toList) { const updatedIssue = issue; if ( toList.listType === ListType.label && !updatedIssue.labels.find((label) => label.id === toList.label.id) ) { updatedIssue.labels.push(toList.label); } if (fromList?.label && fromList.listType === ListType.label) { updatedIssue.labels = updatedIssue.labels.filter((label) => fromList.label.id !== label.id); } if ( toList.listType === ListType.assignee && !updatedIssue.assignees.find((assignee) => assignee.id === toList.assignee.id) ) { updatedIssue.assignees.push(toList.assignee); } if (fromList?.assignee && fromList.listType === ListType.assignee) { updatedIssue.assignees = updatedIssue.assignees.filter( (assignee) => assignee.id !== fromList.assignee.id, ); } return updatedIssue; } export function isListDraggable(list) { return list.listType !== ListType.backlog && list.listType !== ListType.closed; } export function transformNotFilters(filters) { return Object.keys(filters) .filter((key) => key.startsWith(NOT_FILTER)) .reduce((obj, key) => { return { ...obj, [key.substring(4, key.length - 1)]: filters[key], }; }, {}); } // EE-specific feature. Find the implementation in the `ee/`-folder export function transformBoardConfig() { return ''; } export default { getMilestone, formatIssue, formatListIssues, fullBoardId, fullLabelId, fullIterationId, isListDraggable, transformNotFilters, };