debian-mirror-gitlab/app/assets/javascripts/sidebar/mount_sidebar.js
2023-06-20 00:43:36 +05:30

819 lines
21 KiB
JavaScript

import Vue from 'vue';
import VueApollo from 'vue-apollo';
import { TYPENAME_ISSUE, TYPENAME_MERGE_REQUEST } from '~/graphql_shared/constants';
import { convertToGraphQLId, getIdFromGraphQLId } from '~/graphql_shared/utils';
import initInviteMembersModal from '~/invite_members/init_invite_members_modal';
import initInviteMembersTrigger from '~/invite_members/init_invite_members_trigger';
import { TYPE_ISSUE, TYPE_MERGE_REQUEST, WORKSPACE_PROJECT } from '~/issues/constants';
import { gqlClient } from '~/issues/list/graphql';
import {
isInDesignPage,
isInIncidentPage,
isInIssuePage,
isInMRPage,
parseBoolean,
} from '~/lib/utils/common_utils';
import { __ } from '~/locale';
import { apolloProvider } from '~/graphql_shared/issuable_client';
import Translate from '~/vue_shared/translate';
import UserSelect from '~/vue_shared/components/user_select/user_select.vue';
import CollapsedAssigneeList from './components/assignees/collapsed_assignee_list.vue';
import SidebarAssignees from './components/assignees/sidebar_assignees.vue';
import SidebarAssigneesWidget from './components/assignees/sidebar_assignees_widget.vue';
import SidebarConfidentialityWidget from './components/confidential/sidebar_confidentiality_widget.vue';
import CopyEmailToClipboard from './components/copy/copy_email_to_clipboard.vue';
import SidebarDueDateWidget from './components/date/sidebar_date_widget.vue';
import SidebarEscalationStatus from './components/incidents/sidebar_escalation_status.vue';
import { DropdownVariant } from './components/labels/labels_select_vue/constants';
import LabelsSelectWidget from './components/labels/labels_select_widget/labels_select_root.vue';
import IssuableLockForm from './components/lock/issuable_lock_form.vue';
import MilestoneDropdown from './components/milestone/milestone_dropdown.vue';
import MoveIssuesButton from './components/move/move_issues_button.vue';
import SidebarParticipantsWidget from './components/participants/sidebar_participants_widget.vue';
import SidebarReferenceWidget from './components/copy/sidebar_reference_widget.vue';
import SidebarReviewers from './components/reviewers/sidebar_reviewers.vue';
import SidebarReviewersInputs from './components/reviewers/sidebar_reviewers_inputs.vue';
import SidebarSeverityWidget from './components/severity/sidebar_severity_widget.vue';
import SidebarDropdownWidget from './components/sidebar_dropdown_widget.vue';
import StatusDropdown from './components/status/status_dropdown.vue';
import SidebarSubscriptionsWidget from './components/subscriptions/sidebar_subscriptions_widget.vue';
import SubscriptionsDropdown from './components/subscriptions/subscriptions_dropdown.vue';
import SidebarTimeTracking from './components/time_tracking/sidebar_time_tracking.vue';
import SidebarTodoWidget from './components/todo_toggle/sidebar_todo_widget.vue';
import { IssuableAttributeType } from './constants';
import CrmContacts from './components/crm_contacts/crm_contacts.vue';
import trackShowInviteMemberLink from './track_invite_members';
import MoveIssueButton from './components/move/move_issue_button.vue';
Vue.use(Translate);
Vue.use(VueApollo);
function getSidebarOptions(sidebarOptEl = document.querySelector('.js-sidebar-options')) {
return JSON.parse(sidebarOptEl.innerHTML);
}
function mountSidebarTodoWidget() {
const el = document.querySelector('.js-sidebar-todo-widget-root');
if (!el) {
return null;
}
const { projectPath, iid, id } = el.dataset;
return new Vue({
el,
name: 'SidebarTodoWidgetRoot',
apolloProvider,
provide: {
isClassicSidebar: true,
},
render: (createElement) =>
createElement(SidebarTodoWidget, {
props: {
fullPath: projectPath,
issuableId:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? convertToGraphQLId(TYPENAME_ISSUE, id)
: convertToGraphQLId(TYPENAME_MERGE_REQUEST, id),
issuableIid: iid,
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
},
}),
});
}
function getSidebarAssigneeAvailabilityData() {
const sidebarAssigneeEl = document.querySelectorAll('.js-sidebar-assignee-data input');
return Array.from(sidebarAssigneeEl)
.map((el) => el.dataset)
.reduce(
(acc, { username, availability = '' }) => ({
...acc,
[username]: availability,
}),
{},
);
}
function mountSidebarAssigneesDeprecated(mediator) {
const el = document.querySelector('.js-sidebar-assignees-root');
if (!el) {
return null;
}
const { id, iid, fullPath } = getSidebarOptions();
const assigneeAvailabilityStatus = getSidebarAssigneeAvailabilityData();
return new Vue({
el,
name: 'SidebarAssigneesRoot',
apolloProvider,
render: (createElement) =>
createElement(SidebarAssignees, {
props: {
mediator,
issuableIid: String(iid),
projectPath: fullPath,
field: el.dataset.field,
signedIn: Object.prototype.hasOwnProperty.call(el.dataset, 'signedIn'),
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
issuableId: id,
assigneeAvailabilityStatus,
},
}),
});
}
function mountSidebarAssigneesWidget() {
const el = document.querySelector('.js-sidebar-assignees-root');
if (!el) {
return;
}
const { id, iid, fullPath, editable } = getSidebarOptions();
const isIssuablePage = isInIssuePage() || isInIncidentPage() || isInDesignPage();
const issuableType = isIssuablePage ? TYPE_ISSUE : TYPE_MERGE_REQUEST;
// eslint-disable-next-line no-new
new Vue({
el,
name: 'SidebarAssigneesRoot',
apolloProvider,
provide: {
canUpdate: parseBoolean(editable),
directlyInviteMembers: Object.prototype.hasOwnProperty.call(
el.dataset,
'directlyInviteMembers',
),
},
render: (createElement) =>
createElement(SidebarAssigneesWidget, {
props: {
iid: String(iid),
fullPath,
issuableType,
issuableId: id,
allowMultipleAssignees: !el.dataset.maxAssignees || el.dataset.maxAssignees > 1,
editable: parseBoolean(editable),
},
scopedSlots: {
collapsed: ({ users }) =>
createElement(CollapsedAssigneeList, {
props: {
users,
issuableType,
},
}),
},
}),
});
const assigneeDropdown = document.querySelector('.js-sidebar-assignee-dropdown');
if (assigneeDropdown) {
trackShowInviteMemberLink(assigneeDropdown);
}
}
function mountSidebarReviewers(mediator) {
const el = document.querySelector('.js-sidebar-reviewers-root');
if (!el) {
return;
}
const { iid, fullPath } = getSidebarOptions();
// eslint-disable-next-line no-new
new Vue({
el,
name: 'SidebarReviewersRoot',
apolloProvider,
render: (createElement) =>
createElement(SidebarReviewers, {
props: {
mediator,
issuableIid: String(iid),
projectPath: fullPath,
field: el.dataset.field,
issuableType: isInIssuePage() || isInDesignPage() ? TYPE_ISSUE : TYPE_MERGE_REQUEST,
},
}),
});
const reviewersInputEl = document.querySelector('.js-reviewers-inputs');
if (reviewersInputEl) {
// eslint-disable-next-line no-new
new Vue({
el: reviewersInputEl,
render(createElement) {
return createElement(SidebarReviewersInputs);
},
});
}
const reviewerDropdown = document.querySelector('.js-sidebar-reviewer-dropdown');
if (reviewerDropdown) {
trackShowInviteMemberLink(reviewerDropdown);
}
}
function mountSidebarCrmContacts() {
const el = document.querySelector('.js-sidebar-crm-contacts-root');
if (!el) {
return null;
}
const { issueId, groupIssuesPath } = el.dataset;
return new Vue({
el,
name: 'SidebarCrmContactsRoot',
apolloProvider,
render: (createElement) =>
createElement(CrmContacts, {
props: {
issueId,
groupIssuesPath,
},
}),
});
}
function mountSidebarMilestoneWidget() {
const el = document.querySelector('.js-sidebar-milestone-widget-root');
if (!el) {
return null;
}
const { canEdit, projectPath, issueIid } = el.dataset;
return new Vue({
el,
name: 'SidebarMilestoneWidgetRoot',
apolloProvider,
provide: {
canUpdate: parseBoolean(canEdit),
isClassicSidebar: true,
},
render: (createElement) =>
createElement(SidebarDropdownWidget, {
props: {
attrWorkspacePath: projectPath,
workspacePath: projectPath,
iid: issueIid,
issuableType: isInIssuePage() || isInDesignPage() ? TYPE_ISSUE : TYPE_MERGE_REQUEST,
issuableAttribute: IssuableAttributeType.Milestone,
icon: 'clock',
},
}),
});
}
export function mountMilestoneDropdown() {
const el = document.querySelector('.js-milestone-dropdown-root');
if (!el) {
return null;
}
Vue.use(VueApollo);
const {
canAdminMilestone,
fullPath,
inputName,
milestoneId,
milestoneTitle,
projectMilestonesPath,
workspaceType,
} = el.dataset;
return new Vue({
el,
name: 'MilestoneDropdownRoot',
apolloProvider,
render(createElement) {
return createElement(MilestoneDropdown, {
props: {
attrWorkspacePath: fullPath,
canAdminMilestone,
inputName,
issuableType: isInIssuePage() ? TYPE_ISSUE : TYPE_MERGE_REQUEST,
milestoneId,
milestoneTitle,
projectMilestonesPath,
workspaceType,
},
});
},
});
}
export function mountSidebarLabelsWidget() {
const el = document.querySelector('.js-sidebar-labels-widget-root');
if (!el) {
return null;
}
return new Vue({
el,
name: 'SidebarLabelsWidgetRoot',
apolloProvider,
provide: {
...el.dataset,
canUpdate: parseBoolean(el.dataset.canEdit),
allowLabelCreate: parseBoolean(el.dataset.allowLabelCreate),
allowLabelEdit: parseBoolean(el.dataset.canEdit),
allowScopedLabels: parseBoolean(el.dataset.allowScopedLabels),
isClassicSidebar: true,
},
render: (createElement) =>
createElement(LabelsSelectWidget, {
props: {
iid: String(el.dataset.iid),
fullPath: el.dataset.projectPath,
allowLabelRemove: parseBoolean(el.dataset.canEdit),
allowMultiselect: true,
footerCreateLabelTitle: __('Create project label'),
footerManageLabelTitle: __('Manage project labels'),
labelsCreateTitle: __('Create project label'),
labelsFilterBasePath: el.dataset.projectIssuesPath,
variant: DropdownVariant.Sidebar,
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
workspaceType: WORKSPACE_PROJECT,
attrWorkspacePath: el.dataset.projectPath,
labelCreateType: WORKSPACE_PROJECT,
},
class: ['block labels js-labels-block'],
scopedSlots: {
default: () => __('None'),
},
}),
});
}
function mountSidebarConfidentialityWidget() {
const el = document.querySelector('.js-sidebar-confidential-widget-root');
if (!el) {
return null;
}
const { fullPath, iid } = getSidebarOptions();
const dataNode = document.getElementById('js-confidential-issue-data');
const initialData = JSON.parse(dataNode.innerHTML);
return new Vue({
el,
name: 'SidebarConfidentialityWidgetRoot',
apolloProvider,
provide: {
canUpdate: initialData.is_editable,
isClassicSidebar: true,
},
render: (createElement) =>
createElement(SidebarConfidentialityWidget, {
props: {
iid: String(iid),
fullPath,
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
},
}),
});
}
function mountSidebarDueDateWidget() {
const el = document.querySelector('.js-sidebar-due-date-widget-root');
if (!el) {
return null;
}
const { fullPath, iid, editable } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarDueDateWidgetRoot',
apolloProvider,
provide: {
canUpdate: parseBoolean(editable),
},
render: (createElement) =>
createElement(SidebarDueDateWidget, {
props: {
iid: String(iid),
fullPath,
issuableType: TYPE_ISSUE,
},
}),
});
}
function mountSidebarReferenceWidget() {
const el = document.querySelector('.js-sidebar-reference-widget-root');
if (!el) {
return null;
}
const { fullPath, iid } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarReferenceWidgetRoot',
apolloProvider,
provide: {
iid: String(iid),
fullPath,
},
render: (createElement) =>
createElement(SidebarReferenceWidget, {
props: {
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
},
}),
});
}
function mountIssuableLockForm(store) {
const el = document.querySelector('.js-sidebar-lock-root');
if (!el || !store) {
return null;
}
const { fullPath, editable } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarLockRoot',
store,
provide: {
fullPath,
},
render: (createElement) =>
createElement(IssuableLockForm, {
props: {
isEditable: parseBoolean(editable),
},
}),
});
}
function mountSidebarParticipantsWidget() {
const el = document.querySelector('.js-sidebar-participants-widget-root');
if (!el) {
return null;
}
const { fullPath, iid } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarParticipantsWidgetRoot',
apolloProvider,
render: (createElement) =>
createElement(SidebarParticipantsWidget, {
props: {
iid: String(iid),
fullPath,
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
},
}),
});
}
function mountSidebarSubscriptionsWidget() {
const el = document.querySelector('.js-sidebar-subscriptions-widget-root');
if (!el) {
return null;
}
const { fullPath, iid, editable } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarSubscriptionsWidgetRoot',
apolloProvider,
provide: {
canUpdate: parseBoolean(editable),
},
render: (createElement) =>
createElement(SidebarSubscriptionsWidget, {
props: {
iid: String(iid),
fullPath,
issuableType:
isInIssuePage() || isInIncidentPage() || isInDesignPage()
? TYPE_ISSUE
: TYPE_MERGE_REQUEST,
},
}),
});
}
function mountSidebarTimeTracking() {
const el = document.querySelector('.js-sidebar-time-tracking-root');
const {
id,
iid,
fullPath,
issuableType,
timeTrackingLimitToHours,
canCreateTimelogs,
} = getSidebarOptions();
if (!el) {
return null;
}
return new Vue({
el,
name: 'SidebarTimeTrackingRoot',
apolloProvider,
provide: { issuableType },
render: (createElement) =>
createElement(SidebarTimeTracking, {
props: {
fullPath,
issuableId: id.toString(),
issuableIid: iid.toString(),
limitToHours: timeTrackingLimitToHours,
canAddTimeEntries: canCreateTimelogs,
},
}),
});
}
function mountSidebarSeverityWidget() {
const el = document.querySelector('.js-sidebar-severity-widget-root');
if (!el) {
return null;
}
const { fullPath, iid, severity, editable } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarSeverityWidgetRoot',
apolloProvider,
provide: {
canUpdate: parseBoolean(editable),
},
render: (createElement) =>
createElement(SidebarSeverityWidget, {
props: {
projectPath: fullPath,
iid: String(iid),
initialSeverity: severity.toUpperCase(),
},
}),
});
}
function mountSidebarEscalationStatus() {
const el = document.querySelector('.js-sidebar-escalation-status-root');
if (!el) {
return null;
}
const { issuableType } = getSidebarOptions();
const { canUpdate, issueIid, projectPath } = el.dataset;
return new Vue({
el,
name: 'SidebarEscalationStatusRoot',
apolloProvider,
provide: {
canUpdate: parseBoolean(canUpdate),
},
render: (createElement) =>
createElement(SidebarEscalationStatus, {
props: {
iid: issueIid,
issuableType,
projectPath,
},
}),
});
}
function mountCopyEmailToClipboard() {
const el = document.querySelector('.js-sidebar-copy-email-root');
if (!el) {
return null;
}
const { createNoteEmail } = getSidebarOptions();
return new Vue({
el,
name: 'SidebarCopyEmailRoot',
render: (createElement) =>
createElement(CopyEmailToClipboard, { props: { issueEmailAddress: createNoteEmail } }),
});
}
export async function mountMoveIssuesButton() {
const el = document.querySelector('.js-move-issues');
if (!el) {
return null;
}
Vue.use(VueApollo);
return new Vue({
el,
name: 'MoveIssuesRoot',
apolloProvider: new VueApollo({
defaultClient: await gqlClient(),
}),
render: (createElement) =>
createElement(MoveIssuesButton, {
props: {
projectFullPath: el.dataset.projectFullPath,
projectsFetchPath: el.dataset.projectsFetchPath,
},
}),
});
}
export function mountStatusDropdown() {
const el = document.querySelector('.js-status-dropdown');
if (!el) {
return null;
}
return new Vue({
el,
name: 'StatusDropdownRoot',
render: (createElement) => createElement(StatusDropdown),
});
}
export function mountSubscriptionsDropdown() {
const el = document.querySelector('.js-subscriptions-dropdown');
if (!el) {
return null;
}
return new Vue({
el,
name: 'SubscriptionsDropdownRoot',
render: (createElement) => createElement(SubscriptionsDropdown),
});
}
export function mountMoveIssueButton() {
const el = document.querySelector('.js-sidebar-move-issue-block');
if (!el) {
return null;
}
const { projectsAutocompleteEndpoint } = getSidebarOptions();
const { projectFullPath, issueIid } = el.dataset;
Vue.use(VueApollo);
return new Vue({
el,
name: 'MoveIssueDropdownRoot',
apolloProvider,
provide: {
projectsAutocompleteEndpoint,
projectFullPath,
issueIid,
},
render: (createElement) => createElement(MoveIssueButton),
});
}
export function mountAssigneesDropdown() {
const el = document.querySelector('.js-assignee-dropdown');
const assigneeIdsInput = document.querySelector('.js-assignee-ids-input');
if (!el || !assigneeIdsInput) {
return null;
}
const { fullPath } = el.dataset;
const currentUser = {
id: gon?.current_user_id,
username: gon?.current_username,
name: gon?.current_user_fullname,
avatarUrl: gon?.current_user_avatar_url,
};
return new Vue({
el,
apolloProvider,
data() {
return {
selectedUserName: '',
value: [],
};
},
methods: {
onSelectedUnassigned() {
assigneeIdsInput.value = 0;
this.value = [];
this.selectedUserName = __('Unassigned');
},
onSelected(selected) {
assigneeIdsInput.value = selected.map((user) => getIdFromGraphQLId(user.id));
this.value = selected;
this.selectedUserName = selected.map((user) => user.name).join(', ');
},
},
render(h) {
const component = this;
return h(UserSelect, {
props: {
text: component.selectedUserName || __('Select assignee'),
headerText: __('Assign to'),
fullPath,
currentUser,
value: component.value,
},
on: {
input(selected) {
if (!selected.length) {
component.onSelectedUnassigned();
return;
}
component.onSelected(selected);
},
},
class: 'gl-w-full',
});
},
});
}
const isAssigneesWidgetShown =
(isInIssuePage() || isInDesignPage() || isInMRPage()) && gon.features.issueAssigneesWidget;
export function mountSidebar(mediator, store) {
initInviteMembersModal();
initInviteMembersTrigger();
mountSidebarTodoWidget();
if (isAssigneesWidgetShown) {
mountSidebarAssigneesWidget();
} else {
mountSidebarAssigneesDeprecated(mediator);
}
mountSidebarReviewers(mediator);
mountSidebarCrmContacts();
mountSidebarLabelsWidget();
mountSidebarMilestoneWidget();
mountSidebarConfidentialityWidget();
mountSidebarDueDateWidget();
mountSidebarReferenceWidget();
mountIssuableLockForm(store);
mountSidebarParticipantsWidget();
mountSidebarSubscriptionsWidget();
mountCopyEmailToClipboard();
mountSidebarTimeTracking();
mountSidebarSeverityWidget();
mountSidebarEscalationStatus();
mountMoveIssueButton();
}
export { getSidebarOptions };