106 lines
3.7 KiB
JavaScript
106 lines
3.7 KiB
JavaScript
import produce from 'immer';
|
|
import VueApollo from 'vue-apollo';
|
|
import { defaultDataIdFromObject } from '@apollo/client/core';
|
|
import { concatPagination } from '@apollo/client/utilities';
|
|
import getIssueStateQuery from '~/issues/show/queries/get_issue_state.query.graphql';
|
|
import createDefaultClient from '~/lib/graphql';
|
|
import typeDefs from '~/work_items/graphql/typedefs.graphql';
|
|
import { WIDGET_TYPE_NOTES } from '~/work_items/constants';
|
|
|
|
export const config = {
|
|
typeDefs,
|
|
cacheConfig: {
|
|
// included temporarily until Vuex is removed from boards app
|
|
dataIdFromObject: (object) => {
|
|
// eslint-disable-next-line no-underscore-dangle
|
|
return object.__typename === 'BoardList' ? object.iid : defaultDataIdFromObject(object);
|
|
},
|
|
typePolicies: {
|
|
Project: {
|
|
fields: {
|
|
projectMembers: {
|
|
keyArgs: ['fullPath', 'search', 'relations', 'first'],
|
|
},
|
|
},
|
|
},
|
|
WorkItemWidgetNotes: {
|
|
fields: {
|
|
// If we add any key args, the discussions field becomes discussions({"filter":"ONLY_ACTIVITY","first":10}) and
|
|
// kills any possibility to handle it on the widget level without hardcoding a string.
|
|
discussions: {
|
|
keyArgs: false,
|
|
},
|
|
},
|
|
},
|
|
WorkItemWidgetProgress: {
|
|
fields: {
|
|
progress: {
|
|
// We want to show null progress as 0 as per https://gitlab.com/gitlab-org/gitlab/-/issues/386117
|
|
read(existing) {
|
|
return existing === null ? 0 : existing;
|
|
},
|
|
},
|
|
},
|
|
},
|
|
WorkItem: {
|
|
fields: {
|
|
// widgets policy because otherwise the subscriptions invalidate the cache
|
|
widgets: {
|
|
merge(existing = [], incoming, context) {
|
|
if (existing.length === 0) {
|
|
return incoming;
|
|
}
|
|
return existing.map((existingWidget) => {
|
|
const incomingWidget = incoming.find(
|
|
(w) => w.type && w.type === existingWidget.type,
|
|
);
|
|
// We don't want to override existing notes with empty widget on work item updates
|
|
if (incomingWidget?.type === WIDGET_TYPE_NOTES && !context.variables.pageSize) {
|
|
return existingWidget;
|
|
}
|
|
// we want to concat next page of discussions to the existing ones
|
|
if (incomingWidget?.type === WIDGET_TYPE_NOTES && context.variables.after) {
|
|
// concatPagination won't work because we were placing new widget here so we have to do this manually
|
|
return {
|
|
...incomingWidget,
|
|
discussions: {
|
|
...incomingWidget.discussions,
|
|
nodes: [
|
|
...existingWidget.discussions.nodes,
|
|
...incomingWidget.discussions.nodes,
|
|
],
|
|
},
|
|
};
|
|
}
|
|
return incomingWidget || existingWidget;
|
|
});
|
|
},
|
|
},
|
|
},
|
|
},
|
|
MemberInterfaceConnection: {
|
|
fields: {
|
|
nodes: concatPagination(),
|
|
},
|
|
},
|
|
},
|
|
},
|
|
};
|
|
|
|
export const resolvers = {
|
|
Mutation: {
|
|
updateIssueState: (_, { issueType = undefined, isDirty = false }, { cache }) => {
|
|
const sourceData = cache.readQuery({ query: getIssueStateQuery });
|
|
const data = produce(sourceData, (draftData) => {
|
|
draftData.issueState = { issueType, isDirty };
|
|
});
|
|
cache.writeQuery({ query: getIssueStateQuery, data });
|
|
},
|
|
},
|
|
};
|
|
|
|
export const defaultClient = createDefaultClient(resolvers, config);
|
|
|
|
export const apolloProvider = new VueApollo({
|
|
defaultClient,
|
|
});
|