Commit 8f0c9efc authored by Nathan Friend's avatar Nathan Friend

Merge branch '224622-requirement-description-support' into 'master'

Add Requirement description support

See merge request gitlab-org/gitlab!44902
parents 1fb57db4 52a83481
<script> <script>
import { GlDrawer, GlFormGroup, GlFormTextarea, GlFormCheckbox, GlButton } from '@gitlab/ui'; import '~/behaviors/markdown/render_gfm';
import $ from 'jquery';
import {
GlDrawer,
GlFormGroup,
GlFormTextarea,
GlButton,
GlFormCheckbox,
GlTooltipDirective,
GlSafeHtmlDirective as SafeHtml,
} from '@gitlab/ui';
import { isEmpty } from 'lodash'; import { isEmpty } from 'lodash';
import { __, sprintf } from '~/locale'; import { __, sprintf } from '~/locale';
import ZenMode from '~/zen_mode';
import MarkdownField from '~/vue_shared/components/markdown/field.vue';
import RequirementStatusBadge from './requirement_status_badge.vue';
import RequirementMeta from '../mixins/requirement_meta';
import { MAX_TITLE_LENGTH, TestReportStatus } from '../constants'; import { MAX_TITLE_LENGTH, TestReportStatus } from '../constants';
export default { export default {
events: {
drawerClose: 'drawer-close',
disableEdit: 'disable-edit',
enableEdit: 'enable-edit',
},
titleInvalidMessage: sprintf(__('Requirement title cannot have more than %{limit} characters.'), { titleInvalidMessage: sprintf(__('Requirement title cannot have more than %{limit} characters.'), {
limit: MAX_TITLE_LENGTH, limit: MAX_TITLE_LENGTH,
}), }),
...@@ -15,7 +35,15 @@ export default { ...@@ -15,7 +35,15 @@ export default {
GlFormTextarea, GlFormTextarea,
GlFormCheckbox, GlFormCheckbox,
GlButton, GlButton,
MarkdownField,
RequirementStatusBadge,
},
directives: {
GlTooltip: GlTooltipDirective,
SafeHtml,
}, },
mixins: [RequirementMeta],
inject: ['descriptionPreviewPath', 'descriptionHelpPath'],
props: { props: {
drawerOpen: { drawerOpen: {
type: Boolean, type: Boolean,
...@@ -26,6 +54,11 @@ export default { ...@@ -26,6 +54,11 @@ export default {
required: false, required: false,
default: null, default: null,
}, },
enableRequirementEdit: {
type: Boolean,
required: false,
default: false,
},
requirementRequestActive: { requirementRequestActive: {
type: Boolean, type: Boolean,
required: true, required: true,
...@@ -33,8 +66,10 @@ export default { ...@@ -33,8 +66,10 @@ export default {
}, },
data() { data() {
return { return {
zenModeEnabled: false,
title: this.requirement?.title || '', title: this.requirement?.title || '',
satisfied: this.requirement?.satisfied || false, satisfied: this.requirement?.satisfied || false,
description: this.requirement?.description || '',
}; };
}, },
computed: { computed: {
...@@ -48,19 +83,17 @@ export default { ...@@ -48,19 +83,17 @@ export default {
return this.isCreate ? __('Create requirement') : __('Save changes'); return this.isCreate ? __('Create requirement') : __('Save changes');
}, },
titleInvalid() { titleInvalid() {
return this.title.length > MAX_TITLE_LENGTH; return this.title?.length > MAX_TITLE_LENGTH;
}, },
disableSaveButton() { disableSaveButton() {
return this.title === '' || this.titleInvalid || this.requirementRequestActive; return this.title === '' || this.titleInvalid || this.requirementRequestActive;
}, },
reference() {
return `REQ-${this.requirement?.iid}`;
},
}, },
watch: { watch: {
requirement: { requirement: {
handler(value) { handler(value) {
this.title = value?.title || ''; this.title = value?.title || '';
this.description = value?.description || '';
this.satisfied = value?.satisfied || false; this.satisfied = value?.satisfied || false;
}, },
deep: true, deep: true,
...@@ -69,10 +102,25 @@ export default { ...@@ -69,10 +102,25 @@ export default {
// Clear `title` and `satisfied` value on drawer close. // Clear `title` and `satisfied` value on drawer close.
if (!value) { if (!value) {
this.title = ''; this.title = '';
this.description = '';
this.satisfied = false; this.satisfied = false;
} }
}, },
}, },
mounted() {
this.zenMode = new ZenMode();
$(this.$refs.gfmContainer).renderGFM();
$(document).on('zen_mode:enter', () => {
this.zenModeEnabled = true;
});
$(document).on('zen_mode:leave', () => {
this.zenModeEnabled = false;
});
},
beforeDestroy() {
$(document).off('zen_mode:enter');
$(document).off('zen_mode:leave');
},
methods: { methods: {
getDrawerHeaderHeight() { getDrawerHeaderHeight() {
const wrapperEl = document.querySelector('.js-requirements-container-wrapper'); const wrapperEl = document.querySelector('.js-requirements-container-wrapper');
...@@ -100,31 +148,83 @@ export default { ...@@ -100,31 +148,83 @@ export default {
return null; return null;
}, },
handleSave() { handleFormInputKeyDown() {
if (this.isCreate) { if (this.zenModeEnabled) {
this.$emit('save', this.title); // Exit Zen mode, don't close the drawer.
this.zenModeEnabled = false;
this.zenMode.exit();
} else { } else {
this.$emit('save', { this.$emit(this.$options.events.disableEdit);
iid: this.requirement.iid,
title: this.title,
lastTestReportState: this.newLastTestReportState(),
});
} }
}, },
handleSave() {
const { title, description } = this;
const eventParams = {
title,
description,
};
if (!this.isCreate) {
eventParams.iid = this.requirement.iid;
eventParams.lastTestReportState = this.newLastTestReportState();
}
this.$emit('save', eventParams);
},
handleCancel() {
this.$emit(
this.isCreate ? this.$options.events.drawerClose : this.$options.events.disableEdit,
);
},
}, },
}; };
</script> </script>
<template> <template>
<gl-drawer :open="drawerOpen" :header-height="getDrawerHeaderHeight()" @close="$emit('cancel')"> <gl-drawer
:open="drawerOpen"
:header-height="getDrawerHeaderHeight()"
:class="{ 'zen-mode gl-absolute': zenModeEnabled }"
class="requirement-form-drawer"
@close="$emit($options.events.drawerClose)"
>
<template #header> <template #header>
<h4 class="gl-m-0">{{ fieldLabel }}</h4> <h4 v-if="isCreate" class="gl-m-0">{{ __('New Requirement') }}</h4>
<div v-else class="gl-display-flex gl-align-items-center">
<strong class="gl-text-gray-500">{{ reference }}</strong>
<requirement-status-badge
v-if="testReport"
:test-report="testReport"
:last-test-report-manually-created="requirement.lastTestReportManuallyCreated"
class="gl-ml-3"
/>
</div>
</template> </template>
<template> <template>
<div class="requirement-form"> <div v-if="!enableRequirementEdit && !isCreate" class="requirement-details">
<span v-if="!isCreate" class="text-muted">{{ reference }}</span> <div
class="title-container gl-display-flex gl-border-b-1 gl-border-b-solid gl-border-gray-100"
>
<h3 v-safe-html="titleHtml" class="title qa-title gl-flex-grow-1 gl-m-0 gl-mb-3"></h3>
<gl-button
v-if="canUpdate && !isArchived"
v-gl-tooltip.bottom
data-testid="edit"
:title="__('Edit title and description')"
icon="pencil"
class="btn-edit gl-align-self-start"
@click="$emit($options.events.enableEdit, $event)"
/>
</div>
<div data-testid="descriptionContainer" class="description-container gl-mt-3">
<div ref="gfmContainer" v-safe-html="descriptionHtml" class="md"></div>
</div>
</div>
<div v-else class="requirement-form">
<div class="requirement-form-container" :class="{ 'gl-flex-grow-1 gl-mt-2': !isCreate }"> <div class="requirement-form-container" :class="{ 'gl-flex-grow-1 gl-mt-2': !isCreate }">
<div data-testid="form-error-container" class="flash-container"></div>
<gl-form-group <gl-form-group
data-testid="title"
:label="__('Title')" :label="__('Title')"
:invalid-feedback="$options.titleInvalidMessage" :invalid-feedback="$options.titleInvalidMessage"
:state="!titleInvalid" :state="!titleInvalid"
...@@ -137,12 +237,39 @@ export default { ...@@ -137,12 +237,39 @@ export default {
autofocus autofocus
resize resize
:disabled="requirementRequestActive" :disabled="requirementRequestActive"
:placeholder="__('Describe the requirement here')" :placeholder="__('Requirement title')"
max-rows="25" max-rows="25"
class="requirement-form-textarea" class="requirement-form-textarea"
:class="{ 'gl-field-error-outline': titleInvalid }" :class="{ 'gl-field-error-outline': titleInvalid }"
@keyup.escape.exact="$emit('cancel')" @keydown.escape.exact.stop="handleFormInputKeyDown"
@keydown.meta.enter="handleSave"
@keydown.ctrl.enter="handleSave"
/> />
</gl-form-group>
<gl-form-group data-testid="description" class="common-note-form">
<label for="requirementDescription" class="d-block col-form-label gl-pb-0!">
{{ __('Description') }}
</label>
<markdown-field
:markdown-preview-path="descriptionPreviewPath"
:markdown-docs-path="descriptionHelpPath"
:enable-autocomplete="false"
:textarea-value="description"
>
<template #textarea>
<textarea
id="requirementDescription"
v-model="description"
:data-supports-quick-actions="false"
:aria-label="__('Description')"
:placeholder="__('Describe the requirement here')"
class="note-textarea js-gfm-input js-autosize markdown-area qa-description-textarea"
@keydown.escape.exact.stop="handleFormInputKeyDown"
@keydown.meta.enter="handleSave"
@keydown.ctrl.enter="handleSave"
></textarea>
</template>
</markdown-field>
<gl-form-checkbox v-if="!isCreate" v-model="satisfied" class="gl-mt-6">{{ <gl-form-checkbox v-if="!isCreate" v-model="satisfied" class="gl-mt-6">{{
__('Satisfied') __('Satisfied')
}}</gl-form-checkbox> }}</gl-form-checkbox>
...@@ -162,7 +289,7 @@ export default { ...@@ -162,7 +289,7 @@ export default {
variant="default" variant="default"
category="primary" category="primary"
class="js-requirement-cancel" class="js-requirement-cancel"
@click="$emit('cancel')" @click="handleCancel"
> >
{{ __('Cancel') }} {{ __('Cancel') }}
</gl-button> </gl-button>
......
<script> <script>
import { escape } from 'lodash';
import { GlPopover, GlLink, GlAvatar, GlButton, GlTooltipDirective } from '@gitlab/ui'; import { GlPopover, GlLink, GlAvatar, GlButton, GlTooltipDirective } from '@gitlab/ui';
import { __, sprintf } from '~/locale';
import { getTimeago } from '~/lib/utils/datetime_utility';
import timeagoMixin from '~/vue_shared/mixins/timeago'; import timeagoMixin from '~/vue_shared/mixins/timeago';
import RequirementStatusBadge from './requirement_status_badge.vue'; import RequirementStatusBadge from './requirement_status_badge.vue';
import RequirementMeta from '../mixins/requirement_meta';
import { FilterState } from '../constants'; import { FilterState } from '../constants';
export default { export default {
...@@ -20,7 +18,7 @@ export default { ...@@ -20,7 +18,7 @@ export default {
directives: { directives: {
GlTooltip: GlTooltipDirective, GlTooltip: GlTooltipDirective,
}, },
mixins: [timeagoMixin], mixins: [RequirementMeta, timeagoMixin],
props: { props: {
requirement: { requirement: {
type: Object, type: Object,
...@@ -42,36 +40,13 @@ export default { ...@@ -42,36 +40,13 @@ export default {
required: false, required: false,
default: false, default: false,
}, },
active: {
type: Boolean,
required: false,
default: false,
}, },
computed: {
reference() {
return `REQ-${this.requirement.iid}`;
},
canUpdate() {
return this.requirement.userPermissions.updateRequirement;
},
canArchive() {
return this.requirement.userPermissions.adminRequirement;
},
createdAt() {
return sprintf(__('created %{timeAgo}'), {
timeAgo: escape(getTimeago().format(this.requirement.createdAt)),
});
},
updatedAt() {
return sprintf(__('updated %{timeAgo}'), {
timeAgo: escape(getTimeago().format(this.requirement.updatedAt)),
});
},
isArchived() {
return this.requirement?.state === FilterState.archived;
},
author() {
return this.requirement.author;
},
testReport() {
return this.requirement.testReports.nodes[0];
}, },
computed: {
showIssuableMetaActions() { showIssuableMetaActions() {
return Boolean(this.canUpdate || this.canArchive || this.testReport); return Boolean(this.canUpdate || this.canArchive || this.testReport);
}, },
...@@ -105,7 +80,11 @@ export default { ...@@ -105,7 +80,11 @@ export default {
</script> </script>
<template> <template>
<li class="issue requirement" :class="{ 'disabled-content': stateChangeRequestActive }"> <li
class="issue requirement gl-cursor-pointer"
:class="{ 'disabled-content': stateChangeRequestActive, 'gl-bg-blue-50': active }"
@click="$emit('show-click', requirement)"
>
<div class="issue-box"> <div class="issue-box">
<div class="issuable-info-container"> <div class="issuable-info-container">
<span class="issuable-reference text-muted d-none d-sm-block mr-2">{{ reference }}</span> <span class="issuable-reference text-muted d-none d-sm-block mr-2">{{ reference }}</span>
...@@ -119,7 +98,7 @@ export default { ...@@ -119,7 +98,7 @@ export default {
<span <span
v-gl-tooltip:tooltipcontainer.bottom v-gl-tooltip:tooltipcontainer.bottom
:title="tooltipTitle(requirement.createdAt)" :title="tooltipTitle(requirement.createdAt)"
>{{ createdAt }}</span >{{ createdAtFormatted }}</span
> >
{{ __('by') }} {{ __('by') }}
<gl-link ref="authorLink" class="author-link js-user-link" :href="author.webUrl"> <gl-link ref="authorLink" class="author-link js-user-link" :href="author.webUrl">
...@@ -130,7 +109,7 @@ export default { ...@@ -130,7 +109,7 @@ export default {
v-gl-tooltip:tooltipcontainer.bottom v-gl-tooltip:tooltipcontainer.bottom
:title="tooltipTitle(requirement.updatedAt)" :title="tooltipTitle(requirement.updatedAt)"
class="issuable-updated-at" class="issuable-updated-at"
>&middot; {{ updatedAt }}</span >&middot; {{ updatedAtFormatted }}</span
> >
</div> </div>
<requirement-status-badge <requirement-status-badge
...@@ -154,7 +133,7 @@ export default { ...@@ -154,7 +133,7 @@ export default {
v-gl-tooltip v-gl-tooltip
icon="pencil" icon="pencil"
:title="__('Edit')" :title="__('Edit')"
@click="$emit('editClick', requirement)" @click="$emit('edit-click', requirement)"
/> />
</li> </li>
<li v-if="canArchive && !isArchived" class="requirement-archive d-sm-block"> <li v-if="canArchive && !isArchived" class="requirement-archive d-sm-block">
...@@ -164,7 +143,7 @@ export default { ...@@ -164,7 +143,7 @@ export default {
icon="archive" icon="archive"
:loading="stateChangeRequestActive" :loading="stateChangeRequestActive"
:title="__('Archive')" :title="__('Archive')"
@click="handleArchiveClick" @click.stop="handleArchiveClick"
/> />
</li> </li>
<li v-if="canArchive && isArchived" class="requirement-reopen d-sm-block"> <li v-if="canArchive && isArchived" class="requirement-reopen d-sm-block">
......
...@@ -53,7 +53,7 @@ export default { ...@@ -53,7 +53,7 @@ export default {
:description="emptyStateDescription" :description="emptyStateDescription"
> >
<template v-if="emptyStateDescription && canCreateRequirement" #actions> <template v-if="emptyStateDescription && canCreateRequirement" #actions>
<gl-button category="primary" variant="success" @click="$emit('clickNewRequirement')">{{ <gl-button category="primary" variant="success" @click="$emit('click-new-requirement')">{{
__('New requirement') __('New requirement')
}}</gl-button> }}</gl-button>
</template> </template>
......
<script> <script>
import { GlPagination } from '@gitlab/ui'; import { GlPagination } from '@gitlab/ui';
import * as Sentry from '~/sentry/wrapper';
import { __, sprintf } from '~/locale'; import { __, sprintf } from '~/locale';
import Api from '~/api'; import Api from '~/api';
import { deprecatedCreateFlash as createFlash } from '~/flash'; import createFlash from '~/flash';
import { urlParamsToObject } from '~/lib/utils/common_utils'; import { urlParamsToObject } from '~/lib/utils/common_utils';
import { updateHistory, setUrlParams } from '~/lib/utils/url_utility'; import { updateHistory, setUrlParams } from '~/lib/utils/url_utility';
...@@ -153,9 +152,11 @@ export default { ...@@ -153,9 +152,11 @@ export default {
pageInfo: requirementsRoot?.pageInfo || {}, pageInfo: requirementsRoot?.pageInfo || {},
}; };
}, },
error: e => { error() {
createFlash(__('Something went wrong while fetching requirements list.')); createFlash({
Sentry.captureException(e); message: __('Something went wrong while fetching requirements list.'),
captureError: true,
});
}, },
}, },
requirementsCount: { requirementsCount: {
...@@ -174,9 +175,11 @@ export default { ...@@ -174,9 +175,11 @@ export default {
ALL: opened + archived, ALL: opened + archived,
}; };
}, },
error: e => { error() {
createFlash(__('Something went wrong while fetching requirements count.')); createFlash({
Sentry.captureException(e); message: __('Something went wrong while fetching requirements count.'),
captureError: true,
});
}, },
}, },
}, },
...@@ -186,8 +189,9 @@ export default { ...@@ -186,8 +189,9 @@ export default {
textSearch: this.initialTextSearch, textSearch: this.initialTextSearch,
authorUsernames: this.initialAuthorUsernames, authorUsernames: this.initialAuthorUsernames,
sortBy: this.initialSortBy, sortBy: this.initialSortBy,
showCreateForm: false, showRequirementCreateDrawer: false,
showEditForm: false, showRequirementViewDrawer: false,
enableRequirementEdit: false,
editedRequirement: null, editedRequirement: null,
createRequirementRequestActive: false, createRequirementRequestActive: false,
stateChangeRequestActiveFor: 0, stateChangeRequestActiveFor: 0,
...@@ -225,7 +229,7 @@ export default { ...@@ -225,7 +229,7 @@ export default {
return this.requirementsCount[this.filterBy]; return this.requirementsCount[this.filterBy];
}, },
showEmptyState() { showEmptyState() {
return this.requirementsListEmpty && !this.showCreateForm; return this.requirementsListEmpty && !this.showRequirementCreateDrawer;
}, },
showPaginationControls() { showPaginationControls() {
const { hasPreviousPage, hasNextPage } = this.requirements.pageInfo; const { hasPreviousPage, hasNextPage } = this.requirements.pageInfo;
...@@ -337,7 +341,8 @@ export default { ...@@ -337,7 +341,8 @@ export default {
replace: true, replace: true,
}); });
}, },
updateRequirement({ iid, title, state, lastTestReportState, errorFlashMessage }) { updateRequirement(requirement = {}, { errorFlashMessage, flashMessageContainer } = {}) {
const { iid, title, description, state, lastTestReportState } = requirement;
const updateRequirementInput = { const updateRequirementInput = {
projectPath: this.projectPath, projectPath: this.projectPath,
iid, iid,
...@@ -346,6 +351,9 @@ export default { ...@@ -346,6 +351,9 @@ export default {
if (title) { if (title) {
updateRequirementInput.title = title; updateRequirementInput.title = title;
} }
if (description) {
updateRequirementInput.description = description;
}
if (state) { if (state) {
updateRequirementInput.state = state; updateRequirementInput.state = state;
} }
...@@ -361,8 +369,12 @@ export default { ...@@ -361,8 +369,12 @@ export default {
}, },
}) })
.catch(e => { .catch(e => {
createFlash(errorFlashMessage); createFlash({
Sentry.captureException(e); message: errorFlashMessage,
parent: flashMessageContainer,
captureError: true,
});
throw e;
}); });
}, },
handleTabClick({ filterBy }) { handleTabClick({ filterBy }) {
...@@ -382,13 +394,18 @@ export default { ...@@ -382,13 +394,18 @@ export default {
this.$nextTick(() => this.$apollo.queries.requirements.refetch()); this.$nextTick(() => this.$apollo.queries.requirements.refetch());
}, },
handleNewRequirementClick() { handleNewRequirementClick() {
this.showCreateForm = true; this.showRequirementCreateDrawer = true;
},
handleShowRequirementClick(requirement) {
this.showRequirementViewDrawer = true;
this.editedRequirement = requirement;
}, },
handleEditRequirementClick(requirement) { handleEditRequirementClick(requirement) {
this.showEditForm = true; this.showRequirementViewDrawer = true;
this.enableRequirementEdit = true;
this.editedRequirement = requirement; this.editedRequirement = requirement;
}, },
handleNewRequirementSave(title) { handleNewRequirementSave({ title, description }) {
this.createRequirementRequestActive = true; this.createRequirementRequestActive = true;
return this.$apollo return this.$apollo
.mutate({ .mutate({
...@@ -397,87 +414,105 @@ export default { ...@@ -397,87 +414,105 @@ export default {
createRequirementInput: { createRequirementInput: {
projectPath: this.projectPath, projectPath: this.projectPath,
title, title,
description,
}, },
}, },
}) })
.then(({ data }) => { .then(res => {
if (!data.createRequirement.errors.length) { const createReqMutation = res?.data?.createRequirement || {};
if (createReqMutation.errors?.length === 0) {
this.$apollo.queries.requirementsCount.refetch(); this.$apollo.queries.requirementsCount.refetch();
this.$apollo.queries.requirements.refetch(); this.$apollo.queries.requirements.refetch();
this.$toast.show( this.$toast.show(
sprintf(__('Requirement %{reference} has been added'), { sprintf(__('Requirement %{reference} has been added'), {
reference: `REQ-${data.createRequirement.requirement.iid}`, reference: `REQ-${createReqMutation.requirement.iid}`,
}), }),
); );
this.showCreateForm = false; this.showRequirementCreateDrawer = false;
} else { } else {
throw new Error(`Error creating a requirement`); throw new Error(`Error creating a requirement ${res.message}`);
} }
}) })
.catch(e => { .catch(e => {
createFlash(__('Something went wrong while creating a requirement.')); createFlash({
Sentry.captureException(e); message: __('Something went wrong while creating a requirement.'),
parent: this.$el,
captureError: true,
});
throw new Error(`Error creating a requirement ${e.message}`);
}) })
.finally(() => { .finally(() => {
this.createRequirementRequestActive = false; this.createRequirementRequestActive = false;
}); });
}, },
handleRequirementEdit(enableRequirementEdit) {
this.enableRequirementEdit = enableRequirementEdit;
},
handleNewRequirementCancel() { handleNewRequirementCancel() {
this.showCreateForm = false; this.showRequirementCreateDrawer = false;
}, },
handleUpdateRequirementSave(params) { handleUpdateRequirementSave(requirement) {
this.createRequirementRequestActive = true; this.createRequirementRequestActive = true;
return this.updateRequirement({ return this.updateRequirement(requirement, {
...params,
errorFlashMessage: __('Something went wrong while updating a requirement.'), errorFlashMessage: __('Something went wrong while updating a requirement.'),
flashMessageContainer: this.$el,
}) })
.then(({ data }) => { .then(res => {
if (!data.updateRequirement.errors.length) { const updateReqMutation = res?.data?.updateRequirement || {};
this.showEditForm = false;
this.editedRequirement = null; if (updateReqMutation.errors?.length === 0) {
this.enableRequirementEdit = false;
this.editedRequirement = updateReqMutation.requirement;
this.$toast.show( this.$toast.show(
sprintf(__('Requirement %{reference} has been updated'), { sprintf(__('Requirement %{reference} has been updated'), {
reference: `REQ-${data.updateRequirement.requirement.iid}`, reference: `REQ-${this.editedRequirement.iid}`,
}), }),
); );
} else { } else {
throw new Error(`Error updating a requirement`); throw new Error(`Error updating a requirement ${res.message}`);
} }
}) })
.finally(() => { .finally(() => {
this.createRequirementRequestActive = false; this.createRequirementRequestActive = false;
}); });
}, },
handleRequirementStateChange(params) { handleRequirementStateChange(requirement) {
this.stateChangeRequestActiveFor = params.iid; this.stateChangeRequestActiveFor = requirement.iid;
return this.updateRequirement({ return this.updateRequirement(requirement, {
...params,
errorFlashMessage: errorFlashMessage:
params.state === FilterState.opened requirement.state === FilterState.opened
? __('Something went wrong while reopening a requirement.') ? __('Something went wrong while reopening a requirement.')
: __('Something went wrong while archiving a requirement.'), : __('Something went wrong while archiving a requirement.'),
}).then(({ data }) => { })
if (!data.updateRequirement.errors.length) { .then(res => {
const updateReqMutation = res?.data?.updateRequirement || {};
if (updateReqMutation.errors?.length === 0) {
this.$apollo.queries.requirementsCount.refetch(); this.$apollo.queries.requirementsCount.refetch();
this.stateChangeRequestActiveFor = 0; const reference = `REQ-${updateReqMutation.requirement.iid}`;
let toastMessage; let toastMessage;
if (params.state === FilterState.opened) { if (requirement.state === FilterState.opened) {
toastMessage = sprintf(__('Requirement %{reference} has been reopened'), { toastMessage = sprintf(__('Requirement %{reference} has been reopened'), {
reference: `REQ-${data.updateRequirement.requirement.iid}`, reference,
}); });
} else { } else {
toastMessage = sprintf(__('Requirement %{reference} has been archived'), { toastMessage = sprintf(__('Requirement %{reference} has been archived'), {
reference: `REQ-${data.updateRequirement.requirement.iid}`, reference,
}); });
} }
this.$toast.show(toastMessage); this.$toast.show(toastMessage);
} else { } else {
throw new Error(`Error archiving a requirement`); throw new Error(`Error archiving a requirement ${res.message}`);
} }
})
.finally(() => {
this.stateChangeRequestActiveFor = 0;
}); });
}, },
handleUpdateRequirementCancel() { handleUpdateRequirementDrawerClose() {
this.showEditForm = false; this.enableRequirementEdit = false;
this.showRequirementViewDrawer = false;
this.editedRequirement = null; this.editedRequirement = null;
}, },
handleFilterRequirements(filters = []) { handleFilterRequirements(filters = []) {
...@@ -532,10 +567,10 @@ export default { ...@@ -532,10 +567,10 @@ export default {
<requirements-tabs <requirements-tabs
:filter-by="filterBy" :filter-by="filterBy"
:requirements-count="requirementsCount" :requirements-count="requirementsCount"
:show-create-form="showCreateForm" :show-create-form="showRequirementCreateDrawer"
:can-create-requirement="canCreateRequirement" :can-create-requirement="canCreateRequirement"
@clickTab="handleTabClick" @click-tab="handleTabClick"
@clickNewRequirement="handleNewRequirementClick" @click-new-requirement="handleNewRequirementClick"
/> />
<filtered-search-bar <filtered-search-bar
:namespace="projectPath" :namespace="projectPath"
...@@ -550,17 +585,20 @@ export default { ...@@ -550,17 +585,20 @@ export default {
@onSort="handleSortRequirements" @onSort="handleSortRequirements"
/> />
<requirement-create-form <requirement-create-form
:drawer-open="showCreateForm" :drawer-open="showRequirementCreateDrawer"
:requirement-request-active="createRequirementRequestActive" :requirement-request-active="createRequirementRequestActive"
@save="handleNewRequirementSave" @save="handleNewRequirementSave"
@cancel="handleNewRequirementCancel" @drawer-close="handleNewRequirementCancel"
/> />
<requirement-edit-form <requirement-edit-form
:drawer-open="showEditForm" :drawer-open="showRequirementViewDrawer"
:requirement="editedRequirement" :requirement="editedRequirement"
:enable-requirement-edit="enableRequirementEdit"
:requirement-request-active="createRequirementRequestActive" :requirement-request-active="createRequirementRequestActive"
@save="handleUpdateRequirementSave" @save="handleUpdateRequirementSave"
@cancel="handleUpdateRequirementCancel" @enable-edit="handleRequirementEdit(true)"
@disable-edit="handleRequirementEdit(false)"
@drawer-close="handleUpdateRequirementDrawerClose"
/> />
<requirements-empty-state <requirements-empty-state
v-if="showEmptyState" v-if="showEmptyState"
...@@ -568,7 +606,7 @@ export default { ...@@ -568,7 +606,7 @@ export default {
:empty-state-path="emptyStatePath" :empty-state-path="emptyStatePath"
:requirements-count="requirementsCount" :requirements-count="requirementsCount"
:can-create-requirement="canCreateRequirement" :can-create-requirement="canCreateRequirement"
@clickNewRequirement="handleNewRequirementClick" @click-new-requirement="handleNewRequirementClick"
/> />
<requirements-loading <requirements-loading
v-show="requirementsListLoading" v-show="requirementsListLoading"
...@@ -585,7 +623,9 @@ export default { ...@@ -585,7 +623,9 @@ export default {
:key="requirement.iid" :key="requirement.iid"
:requirement="requirement" :requirement="requirement"
:state-change-request-active="stateChangeRequestActiveFor === requirement.iid" :state-change-request-active="stateChangeRequestActiveFor === requirement.iid"
@editClick="handleEditRequirementClick" :active="editedRequirement && editedRequirement.iid === requirement.iid"
@show-click="handleShowRequirementClick"
@edit-click="handleEditRequirementClick"
@archiveClick="handleRequirementStateChange" @archiveClick="handleRequirementStateChange"
@reopenClick="handleRequirementStateChange" @reopenClick="handleRequirementStateChange"
/> />
......
...@@ -50,7 +50,7 @@ export default { ...@@ -50,7 +50,7 @@ export default {
id="state-opened" id="state-opened"
data-state="opened" data-state="opened"
:title="__('Filter by requirements that are currently opened.')" :title="__('Filter by requirements that are currently opened.')"
@click="$emit('clickTab', { filterBy: $options.FilterState.opened })" @click="$emit('click-tab', { filterBy: $options.FilterState.opened })"
> >
{{ __('Open') }} {{ __('Open') }}
<gl-badge class="badge-pill">{{ requirementsCount.OPENED }}</gl-badge> <gl-badge class="badge-pill">{{ requirementsCount.OPENED }}</gl-badge>
...@@ -61,7 +61,7 @@ export default { ...@@ -61,7 +61,7 @@ export default {
id="state-archived" id="state-archived"
data-state="archived" data-state="archived"
:title="__('Filter by requirements that are currently archived.')" :title="__('Filter by requirements that are currently archived.')"
@click="$emit('clickTab', { filterBy: $options.FilterState.archived })" @click="$emit('click-tab', { filterBy: $options.FilterState.archived })"
> >
{{ __('Archived') }} {{ __('Archived') }}
<gl-badge class="badge-pill">{{ requirementsCount.ARCHIVED }}</gl-badge> <gl-badge class="badge-pill">{{ requirementsCount.ARCHIVED }}</gl-badge>
...@@ -72,7 +72,7 @@ export default { ...@@ -72,7 +72,7 @@ export default {
id="state-all" id="state-all"
data-state="all" data-state="all"
:title="__('Show all requirements.')" :title="__('Show all requirements.')"
@click="$emit('clickTab', { filterBy: $options.FilterState.all })" @click="$emit('click-tab', { filterBy: $options.FilterState.all })"
> >
{{ __('All') }} {{ __('All') }}
<gl-badge class="badge-pill">{{ requirementsCount.ALL }}</gl-badge> <gl-badge class="badge-pill">{{ requirementsCount.ALL }}</gl-badge>
...@@ -85,7 +85,7 @@ export default { ...@@ -85,7 +85,7 @@ export default {
variant="success" variant="success"
class="js-new-requirement qa-new-requirement-button" class="js-new-requirement qa-new-requirement-button"
:disabled="showCreateForm" :disabled="showCreateForm"
@click="$emit('clickNewRequirement')" @click="$emit('click-new-requirement')"
>{{ __('New requirement') }}</gl-button >{{ __('New requirement') }}</gl-button
> >
</div> </div>
......
import { __, sprintf } from '~/locale';
import { getTimeago } from '~/lib/utils/datetime_utility';
import { FilterState } from '../constants';
export default {
computed: {
reference() {
return `REQ-${this.requirement?.iid}`;
},
titleHtml() {
return this.requirement?.titleHtml;
},
descriptionHtml() {
return this.requirement?.descriptionHtml;
},
isArchived() {
return this.requirement?.state === FilterState.archived;
},
author() {
return this.requirement?.author;
},
createdAtFormatted() {
return sprintf(__('created %{timeAgo}'), {
timeAgo: getTimeago().format(this.requirement?.createdAt),
});
},
updatedAtFormatted() {
return sprintf(__('updated %{timeAgo}'), {
timeAgo: getTimeago().format(this.requirement?.updatedAt),
});
},
testReport() {
return this.requirement?.testReports.nodes[0];
},
canUpdate() {
return this.requirement?.userPermissions.updateRequirement;
},
canArchive() {
return this.requirement?.userPermissions.adminRequirement;
},
},
};
#import "~/graphql_shared/fragments/pageInfo.fragment.graphql"
#import "./requirement.fragment.graphql"
query projectRequirementsEE( query projectRequirementsEE(
$projectPath: ID! $projectPath: ID!
$state: RequirementState $state: RequirementState
...@@ -7,7 +10,7 @@ query projectRequirementsEE( ...@@ -7,7 +10,7 @@ query projectRequirementsEE(
$nextPageCursor: String = "" $nextPageCursor: String = ""
$authorUsernames: [String!] = [] $authorUsernames: [String!] = []
$search: String = "" $search: String = ""
$sortBy: Sort = created_desc $sortBy: Sort = CREATED_DESC
) { ) {
project(fullPath: $projectPath) { project(fullPath: $projectPath) {
requirements( requirements(
...@@ -21,36 +24,10 @@ query projectRequirementsEE( ...@@ -21,36 +24,10 @@ query projectRequirementsEE(
sort: $sortBy sort: $sortBy
) { ) {
nodes { nodes {
iid ...Requirement
title
createdAt
updatedAt
state
lastTestReportState
lastTestReportManuallyCreated
testReports(first: 1, sort: created_desc) {
nodes {
id
state
createdAt
}
}
userPermissions {
updateRequirement
adminRequirement
}
author {
name
username
avatarUrl
webUrl
}
} }
pageInfo { pageInfo {
hasPreviousPage ...PageInfo
hasNextPage
startCursor
endCursor
} }
} }
} }
......
#import "~/graphql_shared/fragments/author.fragment.graphql"
fragment Requirement on Requirement {
iid
title
titleHtml
description
descriptionHtml
createdAt
updatedAt
state
lastTestReportState
lastTestReportManuallyCreated
testReports(first: 1, sort: CREATED_DESC) {
nodes {
id
state
createdAt
}
}
userPermissions {
updateRequirement
adminRequirement
}
author {
...Author
}
}
#import "./requirement.fragment.graphql"
mutation updateRequirement($updateRequirementInput: UpdateRequirementInput!) { mutation updateRequirement($updateRequirementInput: UpdateRequirementInput!) {
updateRequirement(input: $updateRequirementInput) { updateRequirement(input: $updateRequirementInput) {
clientMutationId clientMutationId
errors errors
requirement { requirement {
iid ...Requirement
title
state
updatedAt
lastTestReportState
testReports(first: 1, sort: created_desc) {
nodes {
id
state
createdAt
}
}
} }
} }
} }
...@@ -38,6 +38,10 @@ export default () => { ...@@ -38,6 +38,10 @@ export default () => {
components: { components: {
RequirementsRoot, RequirementsRoot,
}, },
provide: {
descriptionPreviewPath: el.dataset.descriptionPreviewPath,
descriptionHelpPath: el.dataset.descriptionHelpPath,
},
data() { data() {
const { const {
filterBy, filterBy,
......
...@@ -20,6 +20,12 @@ ...@@ -20,6 +20,12 @@
overflow-y: auto !important; overflow-y: auto !important;
} }
} }
.requirement-form-drawer.zen-mode {
// We need to override `z-index` provided to GlDrawer
// in Zen mode to enable full-screen editing.
z-index: auto !important;
}
} }
.requirements-list-container { .requirements-list-container {
...@@ -73,7 +79,11 @@ ...@@ -73,7 +79,11 @@
} }
.gl-drawer { .gl-drawer {
width: 480px; // Both width & min-width
// are defined as per Pajamas
// See https://gitlab.com/gitlab-org/gitlab/-/merge_requests/44902#note_429056182
width: 28%;
min-width: 400px;
padding-left: $gl-padding; padding-left: $gl-padding;
padding-right: $gl-padding; padding-right: $gl-padding;
box-shadow: none; box-shadow: none;
......
...@@ -31,6 +31,8 @@ ...@@ -31,6 +31,8 @@
all: total_requirements, all: total_requirements,
requirements_web_url: project_requirements_management_requirements_path(@project), requirements_web_url: project_requirements_management_requirements_path(@project),
can_create_requirement: "#{can?(current_user, :create_requirement, @project)}", can_create_requirement: "#{can?(current_user, :create_requirement, @project)}",
description_preview_path: preview_markdown_path(@project),
description_help_path: help_page_path('user/markdown'),
empty_state_path: image_path('illustrations/empty-state/empty-requirements-lg.svg') } } empty_state_path: image_path('illustrations/empty-state/empty-requirements-lg.svg') } }
- if current_tab_count == 0 - if current_tab_count == 0
-# Show regular spinner only when there will be no -# Show regular spinner only when there will be no
......
---
title: Add support for providing requirement description.
merge_request: 44902
author:
type: added
...@@ -6,10 +6,10 @@ RSpec.describe 'Requirements list', :js do ...@@ -6,10 +6,10 @@ RSpec.describe 'Requirements list', :js do
let_it_be(:user) { create(:user) } let_it_be(:user) { create(:user) }
let_it_be(:user_guest) { create(:user) } let_it_be(:user_guest) { create(:user) }
let_it_be(:project) { create(:project, :repository) } let_it_be(:project) { create(:project, :repository) }
let_it_be(:requirement1) { create(:requirement, project: project, title: 'Some requirement-1', author: user, created_at: 5.days.ago, updated_at: 2.days.ago) } let_it_be(:requirement1) { create(:requirement, project: project, title: 'Some requirement-1', description: 'Sample description', author: user, created_at: 5.days.ago, updated_at: 2.days.ago) }
let_it_be(:requirement2) { create(:requirement, project: project, title: 'Some requirement-2', author: user, created_at: 6.days.ago, updated_at: 2.days.ago) } let_it_be(:requirement2) { create(:requirement, project: project, title: 'Some requirement-2', description: 'Sample description', author: user, created_at: 6.days.ago, updated_at: 2.days.ago) }
let_it_be(:requirement3) { create(:requirement, project: project, title: 'Some requirement-3', author: user, created_at: 7.days.ago, updated_at: 2.days.ago) } let_it_be(:requirement3) { create(:requirement, project: project, title: 'Some requirement-3', description: 'Sample description', author: user, created_at: 7.days.ago, updated_at: 2.days.ago) }
let_it_be(:requirement_archived) { create(:requirement, project: project, title: 'Some requirement-3', state: :archived, author: user, created_at: 8.days.ago, updated_at: 2.days.ago) } let_it_be(:requirement_archived) { create(:requirement, project: project, title: 'Some requirement-3', description: 'Sample description', state: :archived, author: user, created_at: 8.days.ago, updated_at: 2.days.ago) }
def create_requirement(title) def create_requirement(title)
page.within('.nav-controls') do page.within('.nav-controls') do
...@@ -131,34 +131,51 @@ RSpec.describe 'Requirements list', :js do ...@@ -131,34 +131,51 @@ RSpec.describe 'Requirements list', :js do
end end
end end
it 'shows title and description along with edit button in drawer' do
find('.requirements-list li.requirement', match: :first).click
page.within('.requirement-form-drawer') do
expect(page.find('.title-container')).to have_content(requirement1.title)
expect(page.find('.title-container')).to have_selector('button.btn-edit')
expect(page.find('.description-container')).to have_content(requirement1.description)
end
end
it 'shows edit form when edit button is clicked for a requirement' do it 'shows edit form when edit button is clicked for a requirement' do
page.within('.requirements-list li.requirement', match: :first) do
find('li.requirement-edit button[title="Edit"]').click
end
page.within('.requirement-form-drawer') do
expect(page.find('.gl-drawer-header span', match: :first)).to have_content("REQ-#{requirement1.iid}")
expect(page.find('textarea#requirementTitle')['value']).to have_content("#{requirement1.title}")
expect(page.find('textarea#requirementDescription')['value']).to have_content("#{requirement1.description}")
expect(page.find('input[type="checkbox"]')['checked']).to eq(requirement1.last_test_report_state)
expect(page.find('.js-requirement-save')).to have_content('Save changes')
end
end
it 'updates requirement using edit form' do
requirement_title = 'Foobar' requirement_title = 'Foobar'
requirement_description = 'Baz'
page.within('.requirements-list li.requirement', match: :first) do page.within('.requirements-list li.requirement', match: :first) do
find('li.requirement-edit button[title="Edit"]').click find('li.requirement-edit button[title="Edit"]').click
end end
page.within('.requirement-form') do page.within('.requirement-form-drawer') do
find('textarea#requirementTitle').native.send_keys requirement_title find('textarea#requirementTitle').native.send_keys requirement_title
find('button.js-requirement-save').click find('textarea#requirementDescription').native.send_keys requirement_description
find('input[type="checkbox"]').click
click_button 'Save changes'
wait_for_all_requests wait_for_all_requests
end end
page.within('.requirements-list li.requirement', match: :first) do page.within('.requirements-list li.requirement', match: :first) do
expect(page.find('.issue-title-text')).to have_content(requirement_title) expect(page.find('.issue-title-text')).to have_content(requirement_title)
end expect(page.find('.requirement-status-badge')).to have_content('satisfied')
end
it 'saves updated title for requirement using edit form' do
page.within('.requirements-list li.requirement', match: :first) do
find('li.requirement-edit button[title="Edit"]').click
end
page.within('.requirement-form') do
expect(page.find('span', match: :first)).to have_content("REQ-#{requirement1.iid}")
expect(page.find('textarea#requirementTitle')['value']).to have_content("#{requirement1.title}")
expect(page.find('.js-requirement-save')).to have_content('Save changes')
end end
end end
......
import $ from 'jquery';
import { shallowMount } from '@vue/test-utils'; import { shallowMount } from '@vue/test-utils';
import { GlDrawer, GlFormTextarea, GlFormCheckbox } from '@gitlab/ui';
import { getByText } from '@testing-library/dom';
import { GlDrawer, GlFormCheckbox, GlFormGroup, GlFormTextarea } from '@gitlab/ui';
import RequirementForm from 'ee/requirements/components/requirement_form.vue'; import RequirementForm from 'ee/requirements/components/requirement_form.vue';
import RequirementStatusBadge from 'ee/requirements/components/requirement_status_badge.vue';
import { TestReportStatus, MAX_TITLE_LENGTH } from 'ee/requirements/constants'; import { TestReportStatus, MAX_TITLE_LENGTH } from 'ee/requirements/constants';
import { mockRequirementsOpen } from '../mock_data'; import MarkdownField from '~/vue_shared/components/markdown/field.vue';
import ZenMode from '~/zen_mode';
import { mockRequirementsOpen, mockTestReport } from '../mock_data';
const createComponent = ({ const createComponent = ({
drawerOpen = true, drawerOpen = true,
...@@ -12,21 +19,30 @@ const createComponent = ({ ...@@ -12,21 +19,30 @@ const createComponent = ({
requirementRequestActive = false, requirementRequestActive = false,
} = {}) => } = {}) =>
shallowMount(RequirementForm, { shallowMount(RequirementForm, {
provide: {
descriptionPreviewPath: '/gitlab-org/gitlab-test/preview_markdown',
descriptionHelpPath: '/help/user/markdown',
},
propsData: { propsData: {
drawerOpen, drawerOpen,
requirement, requirement,
requirementRequestActive, requirementRequestActive,
}, },
stubs: {
GlDrawer,
MarkdownField,
},
}); });
const findGlFormTextArea = wrapper => wrapper.find(GlFormTextarea);
const findGlFormCheckbox = wrapper => wrapper.find(GlFormCheckbox);
describe('RequirementForm', () => { describe('RequirementForm', () => {
let renderGFMSpy;
let documentEventSpyOn;
let wrapper; let wrapper;
let wrapperWithRequirement; let wrapperWithRequirement;
beforeEach(() => { beforeEach(() => {
renderGFMSpy = jest.spyOn($.fn, 'renderGFM');
documentEventSpyOn = jest.spyOn($.prototype, 'on');
wrapper = createComponent(); wrapper = createComponent();
wrapperWithRequirement = createComponent({ wrapperWithRequirement = createComponent({
requirement: mockRequirementsOpen[0], requirement: mockRequirementsOpen[0],
...@@ -87,26 +103,28 @@ describe('RequirementForm', () => { ...@@ -87,26 +103,28 @@ describe('RequirementForm', () => {
}); });
}); });
}); });
describe('reference', () => {
it('returns string containing `requirement.iid` prefixed with `REQ-`', () => {
expect(wrapperWithRequirement.vm.reference).toBe(`REQ-${mockRequirementsOpen[0].iid}`);
});
});
}); });
describe('watchers', () => { describe('watchers', () => {
describe('requirement', () => { describe('requirement', () => {
describe('when requirement is not null', () => { describe('when requirement is not null', () => {
it('renders the value of `requirement.title` as title', async () => { it('renders the value of `requirement.title` as title and `requirement.description` as description', async () => {
wrapper.setProps({ wrapper.setProps({
requirement: mockRequirementsOpen[0], requirement: mockRequirementsOpen[0],
enableRequirementEdit: true,
}); });
await wrapper.vm.$nextTick(); await wrapper.vm.$nextTick();
expect(findGlFormTextArea(wrapper).attributes('value')).toBe( expect(
mockRequirementsOpen[0].title, wrapper
.find('[data-testid="title"]')
.find(GlFormTextarea)
.attributes('value'),
).toBe(mockRequirementsOpen[0].title);
expect(wrapper.find('[data-testid="description"] textarea').element.value).toBe(
mockRequirementsOpen[0].description,
); );
}); });
...@@ -118,26 +136,34 @@ describe('RequirementForm', () => { ...@@ -118,26 +136,34 @@ describe('RequirementForm', () => {
`renders the satisfied checkbox according to the value of \`requirement.satisfied\`=$satisfied`, `renders the satisfied checkbox according to the value of \`requirement.satisfied\`=$satisfied`,
async ({ requirement, satisfied }) => { async ({ requirement, satisfied }) => {
wrapper = createComponent(); wrapper = createComponent();
wrapper.setProps({ requirement }); wrapper.setProps({ requirement, enableRequirementEdit: true });
await wrapper.vm.$nextTick(); await wrapper.vm.$nextTick();
expect(findGlFormCheckbox(wrapper).vm.$attrs.checked).toBe(satisfied); expect(wrapper.find(GlFormCheckbox).vm.$attrs.checked).toBe(satisfied);
}, },
); );
}); });
describe('when requirement is null', () => { describe('when requirement is null', () => {
beforeEach(() => { beforeEach(() => {
wrapperWithRequirement.setProps({ wrapper.setProps({
requirement: null, requirement: null,
enableRequirementEdit: true,
}); });
}); });
it('renders empty string as title', async () => { it('renders empty string as title and description', async () => {
await wrapperWithRequirement.vm.$nextTick(); await wrapper.vm.$nextTick();
expect(findGlFormTextArea(wrapperWithRequirement).attributes('value')).toBe(''); expect(
wrapper
.find('[data-testid="title"]')
.find(GlFormTextarea)
.attributes('value'),
).toBe('');
expect(wrapper.find('[data-testid="description"] textarea').element.value).toBe('');
expect(wrapper.find(GlFormCheckbox).exists()).toBe(false);
}); });
}); });
}); });
...@@ -155,7 +181,38 @@ describe('RequirementForm', () => { ...@@ -155,7 +181,38 @@ describe('RequirementForm', () => {
await wrapper.vm.$nextTick(); await wrapper.vm.$nextTick();
expect(wrapper.vm.title).toBe(''); expect(wrapper.vm.title).toBe('');
expect(wrapper.vm.description).toBe('');
expect(wrapper.vm.satisfied).toBe(false);
});
});
});
describe('mounted', () => {
it('initializes `zenMode` prop on component', () => {
expect(wrapper.vm.zenMode instanceof ZenMode).toBe(true);
});
it('calls `renderGFM` on `$refs.gfmContainer`', () => {
expect(renderGFMSpy).toHaveBeenCalled();
});
it('binds events `zen_mode:enter` & `zen_mode:leave` events on document', () => {
expect(documentEventSpyOn).toHaveBeenCalledWith('zen_mode:enter', expect.any(Function));
expect(documentEventSpyOn).toHaveBeenCalledWith('zen_mode:leave', expect.any(Function));
});
}); });
describe('beforeDestroy', () => {
let documentEventSpyOff;
it('unbinds events `zen_mode:enter` & `zen_mode:leave` events on document', () => {
const wrapperTemp = createComponent();
documentEventSpyOff = jest.spyOn($.prototype, 'off');
wrapperTemp.destroy();
expect(documentEventSpyOff).toHaveBeenCalledWith('zen_mode:enter');
expect(documentEventSpyOff).toHaveBeenCalledWith('zen_mode:leave');
}); });
}); });
...@@ -188,33 +245,53 @@ describe('RequirementForm', () => { ...@@ -188,33 +245,53 @@ describe('RequirementForm', () => {
}); });
describe('handleSave', () => { describe('handleSave', () => {
it('emits `save` event on component with `title` as param when form is in create mode', () => { it('emits `save` event on component with object as param containing `title` & `description` when form is in create mode', () => {
const title = 'foo';
const description = '_bar_';
wrapper.setData({ wrapper.setData({
title: 'foo', title,
description,
}); });
wrapper.vm.handleSave(); wrapper.vm.handleSave();
return wrapper.vm.$nextTick(() => {
expect(wrapper.emitted('save')).toBeTruthy(); expect(wrapper.emitted('save')).toBeTruthy();
expect(wrapper.emitted('save')[0]).toEqual(['foo']); expect(wrapper.emitted('save')[0]).toEqual([
}); {
title,
description,
},
]);
}); });
it('emits `save` event on component with object as param containing `iid` & `title` & `lastTestReportState` when form is in update mode', () => { it('emits `save` event on component with object as param containing `iid`, `title`, `description` & `lastTestReportState` when form is in update mode', () => {
const { iid, title, description } = mockRequirementsOpen[0];
wrapperWithRequirement.vm.handleSave(); wrapperWithRequirement.vm.handleSave();
return wrapperWithRequirement.vm.$nextTick(() => {
expect(wrapperWithRequirement.emitted('save')).toBeTruthy(); expect(wrapperWithRequirement.emitted('save')).toBeTruthy();
expect(wrapperWithRequirement.emitted('save')[0]).toEqual([ expect(wrapperWithRequirement.emitted('save')[0]).toEqual([
{ {
iid: mockRequirementsOpen[0].iid, iid,
title: mockRequirementsOpen[0].title, title,
description,
lastTestReportState: wrapperWithRequirement.vm.newLastTestReportState(), lastTestReportState: wrapperWithRequirement.vm.newLastTestReportState(),
}, },
]); ]);
}); });
}); });
describe('handleCancel', () => {
it('emits `drawer-close` event when form create mode', () => {
wrapper.vm.handleCancel();
expect(wrapper.emitted('drawer-close')).toBeTruthy();
});
it('emits `disable-edit` event when form edit mode', () => {
wrapperWithRequirement.vm.handleCancel();
expect(wrapperWithRequirement.emitted('disable-edit')).toBeTruthy();
});
}); });
}); });
...@@ -223,57 +300,128 @@ describe('RequirementForm', () => { ...@@ -223,57 +300,128 @@ describe('RequirementForm', () => {
expect(wrapper.find(GlDrawer).exists()).toBe(true); expect(wrapper.find(GlDrawer).exists()).toBe(true);
}); });
it('renders element containing requirement reference when form is in edit mode', () => { describe('create requirement', () => {
expect(wrapperWithRequirement.find('span').text()).toBe(`REQ-${mockRequirementsOpen[0].iid}`); it('renders drawer header with string "New Requirement"', () => {
expect(getByText(wrapper.element, 'New Requirement')).not.toBeNull();
}); });
it('does not render gl-form-checkbox when form is in create mode', () => { it('renders title and description input fields', () => {
expect(findGlFormCheckbox(wrapper).exists()).toBe(false); expect(wrapper.find('[data-testid="title"]').exists()).toBe(true);
expect(wrapper.find('[data-testid="description"]').exists()).toBe(true);
}); });
it('renders gl-form-checkbox when form is in edit mode', () => { it('renders save button component', () => {
expect(findGlFormCheckbox(wrapperWithRequirement).exists()).toBe(true); const saveButton = wrapper.find('.js-requirement-save');
expect(saveButton.exists()).toBe(true);
expect(saveButton.text()).toBe('Create requirement');
}); });
it('renders gl-form-group component', () => { it('renders cancel button component', () => {
const glFormGroup = wrapper.find(GlFormGroup); const cancelButton = wrapper.find('.js-requirement-cancel');
expect(cancelButton.exists()).toBe(true);
expect(cancelButton.text()).toBe('Cancel');
});
});
expect(glFormGroup.exists()).toBe(true); describe('view requirement', () => {
expect(glFormGroup.attributes('label')).toBe('Title'); it('renders drawer header with `requirement.reference` and test report badge', () => {
expect(glFormGroup.attributes('label-for')).toBe('requirementTitle'); expect(
expect(glFormGroup.attributes('invalid-feedback')).toBe( getByText(wrapperWithRequirement.element, `REQ-${mockRequirementsOpen[0].iid}`),
`Requirement title cannot have more than ${MAX_TITLE_LENGTH} characters.`, ).not.toBeNull();
expect(wrapperWithRequirement.find(RequirementStatusBadge).exists()).toBe(true);
expect(wrapperWithRequirement.find(RequirementStatusBadge).props('testReport')).toBe(
mockTestReport,
); );
expect(glFormGroup.attributes('state')).toBe('true');
}); });
it('renders gl-form-textarea component', () => { it('renders requirement title', () => {
const glFormTextarea = findGlFormTextArea(wrapper); expect(
getByText(wrapperWithRequirement.element, mockRequirementsOpen[0].titleHtml),
).not.toBeNull();
});
it('renders edit button', () => {
const editButtonEl = wrapperWithRequirement.find('[data-testid="edit"]');
expect(editButtonEl.exists()).toBe(true);
expect(editButtonEl.props('icon')).toBe('pencil');
expect(editButtonEl.attributes('title')).toBe('Edit title and description');
});
it('renders requirement description', () => {
const descriptionEl = wrapperWithRequirement.find('[data-testid="descriptionContainer"]');
expect(descriptionEl.exists()).toBe(true);
expect(descriptionEl.text()).toBe('fortitudinis fomentis dolor mitigari solet.');
});
describe('edit', () => {
beforeEach(async () => {
wrapperWithRequirement.setProps({
enableRequirementEdit: true,
});
expect(glFormTextarea.exists()).toBe(true); await wrapperWithRequirement.vm.$nextTick();
expect(glFormTextarea.attributes('id')).toBe('requirementTitle');
expect(glFormTextarea.attributes('placeholder')).toBe('Describe the requirement here');
expect(glFormTextarea.attributes('max-rows')).toBe('25');
}); });
it('renders gl-form-textarea component populated with `requirement.title` when `requirement` prop is defined', () => { it('renders flash error container', () => {
expect(findGlFormTextArea(wrapperWithRequirement).attributes('value')).toBe( expect(wrapperWithRequirement.find('[data-testid="form-error-container"]').exists()).toBe(
mockRequirementsOpen[0].title, true,
); );
}); });
it('renders save button component', () => { it('renders title input field', () => {
const saveButton = wrapper.find('.js-requirement-save'); const titleInputEl = wrapperWithRequirement.find('[data-testid="title"]');
const titleTextarea = titleInputEl.find(GlFormTextarea);
expect(saveButton.exists()).toBe(true); expect(titleInputEl.exists()).toBe(true);
expect(saveButton.text()).toBe('Create requirement'); expect(titleInputEl.attributes()).toMatchObject({
label: 'Title',
state: 'true',
'label-for': 'requirementTitle',
'invalid-feedback': `Requirement title cannot have more than ${MAX_TITLE_LENGTH} characters.`,
}); });
it('renders cancel button component', () => { expect(titleTextarea.exists()).toBe(true);
const cancelButton = wrapper.find('.js-requirement-cancel'); expect(titleTextarea.attributes()).toMatchObject({
id: 'requirementTitle',
placeholder: 'Requirement title',
value: mockRequirementsOpen[0].title,
'max-rows': '25',
});
});
expect(cancelButton.exists()).toBe(true); it('renders description input field', () => {
expect(cancelButton.text()).toBe('Cancel'); const descriptionInputEl = wrapperWithRequirement.find('[data-testid="description"]');
const markdownEl = descriptionInputEl.find(MarkdownField);
const descriptionTextarea = markdownEl.find('textarea');
expect(descriptionInputEl.exists()).toBe(true);
expect(descriptionInputEl.find('label').text()).toBe('Description');
expect(markdownEl.exists()).toBe(true);
expect(markdownEl.props()).toMatchObject({
markdownPreviewPath: '/gitlab-org/gitlab-test/preview_markdown',
markdownDocsPath: '/help/user/markdown',
enableAutocomplete: false,
textareaValue: mockRequirementsOpen[0].description,
});
expect(descriptionTextarea.exists()).toBe(true);
expect(descriptionTextarea.attributes()).toMatchObject({
id: 'requirementDescription',
placeholder: 'Describe the requirement here',
'aria-label': 'Description',
});
});
it('renders satisfied checkbox field', () => {
expect(wrapperWithRequirement.find(GlFormCheckbox).exists()).toBe(true);
expect(wrapperWithRequirement.find(GlFormCheckbox).text()).toBe('Satisfied');
});
});
}); });
}); });
}); });
...@@ -32,64 +32,6 @@ describe('RequirementItem', () => { ...@@ -32,64 +32,6 @@ describe('RequirementItem', () => {
wrapperArchived.destroy(); wrapperArchived.destroy();
}); });
describe('computed', () => {
describe('reference', () => {
it('returns string containing `requirement.iid` prefixed with "REQ-"', () => {
expect(wrapper.vm.reference).toBe(`REQ-${requirement1.iid}`);
});
});
describe('canUpdate', () => {
it('returns value of `requirement.userPermissions.updateRequirement`', () => {
expect(wrapper.vm.canUpdate).toBe(requirement1.userPermissions.updateRequirement);
});
});
describe('canArchive', () => {
it('returns value of `requirement.userPermissions.updateRequirement`', () => {
expect(wrapper.vm.canArchive).toBe(requirement1.userPermissions.adminRequirement);
});
});
describe('createdAt', () => {
it('returns timeago-style string representing `requirement.createdAt`', () => {
// We don't have to be accurate here as it is already covered in rspecs
expect(wrapper.vm.createdAt).toContain('created');
expect(wrapper.vm.createdAt).toContain('ago');
});
});
describe('updatedAt', () => {
it('returns timeago-style string representing `requirement.updatedAt`', () => {
// We don't have to be accurate here as it is already covered in rspecs
expect(wrapper.vm.updatedAt).toContain('updated');
expect(wrapper.vm.updatedAt).toContain('ago');
});
});
describe('isArchived', () => {
it('returns `true` when current requirement is archived', () => {
expect(wrapperArchived.vm.isArchived).toBe(true);
});
it('returns `false` when current requirement is archived', () => {
expect(wrapper.vm.isArchived).toBe(false);
});
});
describe('author', () => {
it('returns value of `requirement.author`', () => {
expect(wrapper.vm.author).toBe(requirement1.author);
});
});
describe('testReport', () => {
it('returns testReport object from reports array within `requirement`', () => {
expect(wrapper.vm.testReport).toBe(mockTestReport);
});
});
});
describe('methods', () => { describe('methods', () => {
describe('handleArchiveClick', () => { describe('handleArchiveClick', () => {
it('emits `archiveClick` event on component with object containing `requirement.iid` & `state` as "ARCHIVED" as param', () => { it('emits `archiveClick` event on component with object containing `requirement.iid` & `state` as "ARCHIVED" as param', () => {
...@@ -139,6 +81,13 @@ describe('RequirementItem', () => { ...@@ -139,6 +81,13 @@ describe('RequirementItem', () => {
}); });
}); });
it('emits `show-click` event with requirement as param', () => {
wrapper.trigger('click');
expect(wrapper.emitted('show-click')).toBeTruthy();
expect(wrapper.emitted('show-click')[0]).toEqual([requirement1]);
});
it('renders element containing requirement reference', () => { it('renders element containing requirement reference', () => {
expect(wrapper.find('.issuable-reference').text()).toBe(`REQ-${requirement1.iid}`); expect(wrapper.find('.issuable-reference').text()).toBe(`REQ-${requirement1.iid}`);
}); });
...@@ -186,6 +135,11 @@ describe('RequirementItem', () => { ...@@ -186,6 +135,11 @@ describe('RequirementItem', () => {
expect(editButtonEl.exists()).toBe(true); expect(editButtonEl.exists()).toBe(true);
expect(editButtonEl.attributes('title')).toBe('Edit'); expect(editButtonEl.attributes('title')).toBe('Edit');
editButtonEl.vm.$emit('click');
expect(wrapper.emitted('edit-click')).toBeTruthy();
expect(wrapper.emitted('edit-click')[0]).toEqual([wrapper.vm.requirement]);
}); });
it('does not render element containing requirement `Edit` button when `requirement.userPermissions.updateRequirement` is false', () => { it('does not render element containing requirement `Edit` button when `requirement.userPermissions.updateRequirement` is false', () => {
......
...@@ -12,10 +12,9 @@ import createRequirement from 'ee/requirements/queries/createRequirement.mutatio ...@@ -12,10 +12,9 @@ import createRequirement from 'ee/requirements/queries/createRequirement.mutatio
import updateRequirement from 'ee/requirements/queries/updateRequirement.mutation.graphql'; import updateRequirement from 'ee/requirements/queries/updateRequirement.mutation.graphql';
import { TEST_HOST } from 'helpers/test_constants'; import { TEST_HOST } from 'helpers/test_constants';
import * as Sentry from '~/sentry/wrapper';
import AuthorToken from '~/vue_shared/components/filtered_search_bar/tokens/author_token.vue'; import AuthorToken from '~/vue_shared/components/filtered_search_bar/tokens/author_token.vue';
import FilteredSearchBarRoot from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue'; import FilteredSearchBarRoot from '~/vue_shared/components/filtered_search_bar/filtered_search_bar_root.vue';
import { deprecatedCreateFlash as createFlash } from '~/flash'; import createFlash from '~/flash';
import { import {
FilterState, FilterState,
...@@ -140,9 +139,9 @@ describe('RequirementsRoot', () => { ...@@ -140,9 +139,9 @@ describe('RequirementsRoot', () => {
}); });
describe('showEmptyState', () => { describe('showEmptyState', () => {
it('returns `false` when `showCreateForm` is true', () => { it('returns `false` when `showRequirementCreateDrawer` is true', () => {
wrapper.setData({ wrapper.setData({
showCreateForm: true, showRequirementCreateDrawer: true,
}); });
return wrapper.vm.$nextTick(() => { return wrapper.vm.$nextTick(() => {
...@@ -329,6 +328,28 @@ describe('RequirementsRoot', () => { ...@@ -329,6 +328,28 @@ describe('RequirementsRoot', () => {
); );
}); });
it('calls `$apollo.mutate` with variables containing `description` when it is included in object param', () => {
jest.spyOn(wrapper.vm.$apollo, 'mutate').mockResolvedValue(mockUpdateMutationResult);
wrapper.vm.updateRequirement({
iid: '1',
description: '_foo_',
});
expect(wrapper.vm.$apollo.mutate).toHaveBeenCalledWith(
expect.objectContaining({
mutation: updateRequirement,
variables: {
updateRequirementInput: {
projectPath: 'gitlab-org/gitlab-shell',
iid: '1',
description: '_foo_',
},
},
}),
);
});
it('calls `$apollo.mutate` with variables containing `state` when it is included in object param', () => { it('calls `$apollo.mutate` with variables containing `state` when it is included in object param', () => {
jest.spyOn(wrapper.vm.$apollo, 'mutate').mockResolvedValue(mockUpdateMutationResult); jest.spyOn(wrapper.vm.$apollo, 'mutate').mockResolvedValue(mockUpdateMutationResult);
...@@ -396,35 +417,49 @@ describe('RequirementsRoot', () => { ...@@ -396,35 +417,49 @@ describe('RequirementsRoot', () => {
}); });
}); });
it('calls `createFlash` with provided `errorFlashMessage` param and `Sentry.captureException` when request fails', () => { it('calls `createFlash` with provided `errorFlashMessage` param when request fails', () => {
jest.spyOn(wrapper.vm.$apollo, 'mutate').mockRejectedValue(new Error()); jest.spyOn(wrapper.vm.$apollo, 'mutate').mockRejectedValue(new Error({}));
jest.spyOn(Sentry, 'captureException').mockImplementation();
return wrapper.vm return wrapper.vm
.updateRequirement({ .updateRequirement(
{
iid: '1', iid: '1',
},
{
errorFlashMessage: 'Something went wrong', errorFlashMessage: 'Something went wrong',
}) },
.then(() => { )
expect(createFlash).toHaveBeenCalledWith('Something went wrong'); .catch(() => {
expect(Sentry.captureException).toHaveBeenCalledWith(expect.any(Object)); expect(createFlash).toHaveBeenCalledWith({
message: 'Something went wrong',
captureError: true,
});
}); });
}); });
}); });
describe('handleNewRequirementClick', () => { describe('handleNewRequirementClick', () => {
it('sets `showCreateForm` prop to `true`', () => { it('sets `showRequirementCreateDrawer` prop to `true`', () => {
wrapper.vm.handleNewRequirementClick(); wrapper.vm.handleNewRequirementClick();
expect(wrapper.vm.showCreateForm).toBe(true); expect(wrapper.vm.showRequirementCreateDrawer).toBe(true);
});
});
describe('handleShowRequirementClick', () => {
it('sets `showRequirementViewDrawer` prop to `true`', () => {
wrapper.vm.handleShowRequirementClick(mockRequirementsOpen[0]);
expect(wrapper.vm.showRequirementViewDrawer).toBe(true);
expect(wrapper.vm.editedRequirement).toBe(mockRequirementsOpen[0]);
}); });
}); });
describe('handleEditRequirementClick', () => { describe('handleEditRequirementClick', () => {
it('sets `showEditForm` prop to `true` and `editedRequirement` to value of passed param', () => { it('sets `showRequirementViewDrawer` prop to `true` and `editedRequirement` to value of passed param', () => {
wrapper.vm.handleEditRequirementClick(mockRequirementsOpen[0]); wrapper.vm.handleEditRequirementClick(mockRequirementsOpen[0]);
expect(wrapper.vm.showEditForm).toBe(true); expect(wrapper.vm.showRequirementViewDrawer).toBe(true);
expect(wrapper.vm.editedRequirement).toBe(mockRequirementsOpen[0]); expect(wrapper.vm.editedRequirement).toBe(mockRequirementsOpen[0]);
}); });
}); });
...@@ -446,7 +481,10 @@ describe('RequirementsRoot', () => { ...@@ -446,7 +481,10 @@ describe('RequirementsRoot', () => {
.spyOn(wrapper.vm.$apollo, 'mutate') .spyOn(wrapper.vm.$apollo, 'mutate')
.mockReturnValue(Promise.resolve(mockMutationResult)); .mockReturnValue(Promise.resolve(mockMutationResult));
wrapper.vm.handleNewRequirementSave('foo'); wrapper.vm.handleNewRequirementSave({
title: 'foo',
description: '_bar_',
});
expect(wrapper.vm.createRequirementRequestActive).toBe(true); expect(wrapper.vm.createRequirementRequestActive).toBe(true);
}); });
...@@ -456,7 +494,10 @@ describe('RequirementsRoot', () => { ...@@ -456,7 +494,10 @@ describe('RequirementsRoot', () => {
.spyOn(wrapper.vm.$apollo, 'mutate') .spyOn(wrapper.vm.$apollo, 'mutate')
.mockReturnValue(Promise.resolve(mockMutationResult)); .mockReturnValue(Promise.resolve(mockMutationResult));
wrapper.vm.handleNewRequirementSave('foo'); wrapper.vm.handleNewRequirementSave({
title: 'foo',
description: '_bar_',
});
expect(wrapper.vm.$apollo.mutate).toHaveBeenCalledWith( expect(wrapper.vm.$apollo.mutate).toHaveBeenCalledWith(
expect.objectContaining({ expect.objectContaining({
...@@ -465,13 +506,14 @@ describe('RequirementsRoot', () => { ...@@ -465,13 +506,14 @@ describe('RequirementsRoot', () => {
createRequirementInput: { createRequirementInput: {
projectPath: 'gitlab-org/gitlab-shell', projectPath: 'gitlab-org/gitlab-shell',
title: 'foo', title: 'foo',
description: '_bar_',
}, },
}, },
}), }),
); );
}); });
it('sets `showCreateForm` and `createRequirementRequestActive` props to `false` and refetches requirements count and list when request is successful', () => { it('sets `showRequirementCreateDrawer` and `createRequirementRequestActive` props to `false` and refetches requirements count and list when request is successful', () => {
jest jest
.spyOn(wrapper.vm.$apollo, 'mutate') .spyOn(wrapper.vm.$apollo, 'mutate')
.mockReturnValue(Promise.resolve(mockMutationResult)); .mockReturnValue(Promise.resolve(mockMutationResult));
...@@ -482,10 +524,15 @@ describe('RequirementsRoot', () => { ...@@ -482,10 +524,15 @@ describe('RequirementsRoot', () => {
.spyOn(wrapper.vm.$apollo.queries.requirements, 'refetch') .spyOn(wrapper.vm.$apollo.queries.requirements, 'refetch')
.mockImplementation(jest.fn()); .mockImplementation(jest.fn());
return wrapper.vm.handleNewRequirementSave('foo').then(() => { return wrapper.vm
.handleNewRequirementSave({
title: 'foo',
description: '_bar_',
})
.then(() => {
expect(wrapper.vm.$apollo.queries.requirementsCount.refetch).toHaveBeenCalled(); expect(wrapper.vm.$apollo.queries.requirementsCount.refetch).toHaveBeenCalled();
expect(wrapper.vm.$apollo.queries.requirements.refetch).toHaveBeenCalled(); expect(wrapper.vm.$apollo.queries.requirements.refetch).toHaveBeenCalled();
expect(wrapper.vm.showCreateForm).toBe(false); expect(wrapper.vm.showRequirementCreateDrawer).toBe(false);
expect(wrapper.vm.createRequirementRequestActive).toBe(false); expect(wrapper.vm.createRequirementRequestActive).toBe(false);
}); });
}); });
...@@ -493,7 +540,12 @@ describe('RequirementsRoot', () => { ...@@ -493,7 +540,12 @@ describe('RequirementsRoot', () => {
it('calls `$toast.show` with string "Requirement added successfully" when request is successful', () => { it('calls `$toast.show` with string "Requirement added successfully" when request is successful', () => {
jest.spyOn(wrapper.vm.$apollo, 'mutate').mockResolvedValue(mockMutationResult); jest.spyOn(wrapper.vm.$apollo, 'mutate').mockResolvedValue(mockMutationResult);
return wrapper.vm.handleNewRequirementSave('foo').then(() => { return wrapper.vm
.handleNewRequirementSave({
title: 'foo',
description: '_bar_',
})
.then(() => {
expect(wrapper.vm.$toast.show).toHaveBeenCalledWith('Requirement REQ-1 has been added'); expect(wrapper.vm.$toast.show).toHaveBeenCalledWith('Requirement REQ-1 has been added');
}); });
}); });
...@@ -501,10 +553,17 @@ describe('RequirementsRoot', () => { ...@@ -501,10 +553,17 @@ describe('RequirementsRoot', () => {
it('sets `createRequirementRequestActive` prop to `false` and calls `createFlash` when `$apollo.mutate` request fails', () => { it('sets `createRequirementRequestActive` prop to `false` and calls `createFlash` when `$apollo.mutate` request fails', () => {
jest.spyOn(wrapper.vm.$apollo, 'mutate').mockReturnValue(Promise.reject(new Error())); jest.spyOn(wrapper.vm.$apollo, 'mutate').mockReturnValue(Promise.reject(new Error()));
return wrapper.vm.handleNewRequirementSave('foo').then(() => { return wrapper.vm
expect(createFlash).toHaveBeenCalledWith( .handleNewRequirementSave({
'Something went wrong while creating a requirement.', title: 'foo',
); description: '_bar_',
})
.catch(() => {
expect(createFlash).toHaveBeenCalledWith({
message: 'Something went wrong while creating a requirement.',
captureError: true,
parent: expect.any(Object),
});
expect(wrapper.vm.createRequirementRequestActive).toBe(false); expect(wrapper.vm.createRequirementRequestActive).toBe(false);
}); });
}); });
...@@ -533,12 +592,14 @@ describe('RequirementsRoot', () => { ...@@ -533,12 +592,14 @@ describe('RequirementsRoot', () => {
expect.objectContaining({ expect.objectContaining({
iid: '1', iid: '1',
title: 'foo', title: 'foo',
}),
expect.objectContaining({
errorFlashMessage: 'Something went wrong while updating a requirement.', errorFlashMessage: 'Something went wrong while updating a requirement.',
}), }),
); );
}); });
it('sets `showEditForm` to `true`, `editedRequirement` to `null` and `createRequirementRequestActive` prop to `false` when request is successful', () => { it('sets `showRequirementViewDrawer` to `true`, `editedRequirement` to `null` and `createRequirementRequestActive` prop to `false` when request is successful', () => {
jest.spyOn(wrapper.vm, 'updateRequirement').mockResolvedValue(mockUpdateMutationResult); jest.spyOn(wrapper.vm, 'updateRequirement').mockResolvedValue(mockUpdateMutationResult);
return wrapper.vm return wrapper.vm
...@@ -547,8 +608,10 @@ describe('RequirementsRoot', () => { ...@@ -547,8 +608,10 @@ describe('RequirementsRoot', () => {
title: 'foo', title: 'foo',
}) })
.then(() => { .then(() => {
expect(wrapper.vm.showEditForm).toBe(false); expect(wrapper.vm.enableRequirementEdit).toBe(false);
expect(wrapper.vm.editedRequirement).toBe(null); expect(wrapper.vm.editedRequirement).toEqual(
mockUpdateMutationResult.data.updateRequirement.requirement,
);
expect(wrapper.vm.createRequirementRequestActive).toBe(false); expect(wrapper.vm.createRequirementRequestActive).toBe(false);
}); });
}); });
...@@ -582,14 +645,14 @@ describe('RequirementsRoot', () => { ...@@ -582,14 +645,14 @@ describe('RequirementsRoot', () => {
}); });
describe('handleNewRequirementCancel', () => { describe('handleNewRequirementCancel', () => {
it('sets `showCreateForm` prop to `false`', () => { it('sets `showRequirementCreateDrawer` prop to `false`', () => {
wrapper.setData({ wrapper.setData({
showCreateForm: true, showRequirementCreateDrawer: true,
}); });
wrapper.vm.handleNewRequirementCancel(); wrapper.vm.handleNewRequirementCancel();
expect(wrapper.vm.showCreateForm).toBe(false); expect(wrapper.vm.showRequirementCreateDrawer).toBe(false);
}); });
}); });
...@@ -617,6 +680,8 @@ describe('RequirementsRoot', () => { ...@@ -617,6 +680,8 @@ describe('RequirementsRoot', () => {
expect.objectContaining({ expect.objectContaining({
iid: '1', iid: '1',
state: FilterState.opened, state: FilterState.opened,
}),
expect.objectContaining({
errorFlashMessage: 'Something went wrong while reopening a requirement.', errorFlashMessage: 'Something went wrong while reopening a requirement.',
}), }),
); );
...@@ -634,6 +699,8 @@ describe('RequirementsRoot', () => { ...@@ -634,6 +699,8 @@ describe('RequirementsRoot', () => {
expect.objectContaining({ expect.objectContaining({
iid: '1', iid: '1',
state: FilterState.archived, state: FilterState.archived,
}),
expect.objectContaining({
errorFlashMessage: 'Something went wrong while archiving a requirement.', errorFlashMessage: 'Something went wrong while archiving a requirement.',
}), }),
); );
...@@ -693,11 +760,12 @@ describe('RequirementsRoot', () => { ...@@ -693,11 +760,12 @@ describe('RequirementsRoot', () => {
}); });
}); });
describe('handleUpdateRequirementCancel', () => { describe('handleUpdateRequirementDrawerClose', () => {
it('sets `showEditForm` prop to `false` and `editedRequirement` to `null`', () => { it('sets `enableRequirementEdit` & `showRequirementViewDrawer` to false and `editedRequirement` to `null`', () => {
wrapper.vm.handleUpdateRequirementCancel(); wrapper.vm.handleUpdateRequirementDrawerClose();
expect(wrapper.vm.showEditForm).toBe(false); expect(wrapper.vm.enableRequirementEdit).toBe(false);
expect(wrapper.vm.showRequirementViewDrawer).toBe(false);
expect(wrapper.vm.editedRequirement).toBe(null); expect(wrapper.vm.editedRequirement).toBe(null);
}); });
}); });
...@@ -847,9 +915,9 @@ describe('RequirementsRoot', () => { ...@@ -847,9 +915,9 @@ describe('RequirementsRoot', () => {
expect(wrapper.find('requirement-edit-form-stub').exists()).toBe(true); expect(wrapper.find('requirement-edit-form-stub').exists()).toBe(true);
}); });
it('does not render requirement-empty-state component when `showCreateForm` prop is `true`', () => { it('does not render requirement-empty-state component when `showRequirementCreateDrawer` prop is `true`', () => {
wrapper.setData({ wrapper.setData({
showCreateForm: true, showRequirementCreateDrawer: true,
}); });
return wrapper.vm.$nextTick(() => { return wrapper.vm.$nextTick(() => {
......
import { shallowMount } from '@vue/test-utils';
import RequirementItem from 'ee/requirements/components/requirement_item.vue';
import { FilterState } from 'ee/requirements/constants';
import { mockAuthor, mockTestReport, requirement1 as mockRequirement } from '../mock_data';
const createComponent = (requirement = mockRequirement) =>
shallowMount(RequirementItem, {
propsData: {
requirement,
},
});
describe('RequirementMeta Mixin', () => {
let wrapper;
beforeEach(() => {
wrapper = createComponent();
});
afterEach(() => {
wrapper.destroy();
});
describe('computed', () => {
describe('reference', () => {
it('returns string containing `requirement.iid` prefixed with `REQ-`', () => {
expect(wrapper.vm.reference).toBe(`REQ-${mockRequirement.iid}`);
});
});
describe('titleHtml', () => {
it('returns value of `requirement.titleHtml`', () => {
expect(wrapper.vm.titleHtml).toBe(mockRequirement.titleHtml);
});
});
describe('descriptionHtml', () => {
it('returns value of `requirement.descriptionHtml`', () => {
expect(wrapper.vm.descriptionHtml).toBe(mockRequirement.descriptionHtml);
});
});
describe('isArchived', () => {
it('returns true when `requirement.state` is "ARCHIVED"', async () => {
wrapper.setProps({
requirement: {
...mockRequirement,
state: FilterState.archived,
},
});
await wrapper.vm.$nextTick();
expect(wrapper.vm.isArchived).toBe(true);
});
it('returns false when `requirement.state` is "OPENED"', () => {
expect(wrapper.vm.isArchived).toBe(false);
});
});
describe('author', () => {
it('returns value of `requirement.author`', () => {
expect(wrapper.vm.author).toBe(mockAuthor);
});
});
describe('createdAtFormatted', () => {
it('returns timeago-style string representing `requirement.createdAtFormatted`', () => {
// We don't have to be accurate here as it is already covered in rspecs
expect(wrapper.vm.createdAtFormatted).toContain('created');
expect(wrapper.vm.createdAtFormatted).toContain('ago');
});
});
describe('updatedAtFormatted', () => {
it('returns timeago-style string representing `requirement.updatedAtFormatted`', () => {
// We don't have to be accurate here as it is already covered in rspecs
expect(wrapper.vm.updatedAtFormatted).toContain('updated');
expect(wrapper.vm.updatedAtFormatted).toContain('ago');
});
});
describe('testReport', () => {
it('returns testReport object from reports array within `requirement`', () => {
expect(wrapper.vm.testReport).toBe(mockTestReport);
});
});
describe('canUpdate', () => {
it('returns value of `requirement.userPermissions.updateRequirement`', () => {
expect(wrapper.vm.canUpdate).toBe(mockRequirement.userPermissions.updateRequirement);
});
});
describe('canArchive', () => {
it('returns value of `requirement.userPermissions.updateRequirement`', () => {
expect(wrapper.vm.canArchive).toBe(mockRequirement.userPermissions.adminRequirement);
});
});
});
});
...@@ -34,6 +34,10 @@ export const mockTestReportMissing = { ...@@ -34,6 +34,10 @@ export const mockTestReportMissing = {
export const requirement1 = { export const requirement1 = {
iid: '1', iid: '1',
title: 'Virtutis, magnitudinis animi, patientiae, fortitudinis fomentis dolor mitigari solet.', title: 'Virtutis, magnitudinis animi, patientiae, fortitudinis fomentis dolor mitigari solet.',
titleHtml:
'Virtutis, magnitudinis animi, patientiae, fortitudinis fomentis dolor mitigari solet.',
description: 'fortitudinis _fomentis_ dolor mitigari solet.',
descriptionHtml: 'fortitudinis <i>fomentis</i> dolor mitigari solet.',
createdAt: '2020-03-19T08:09:09Z', createdAt: '2020-03-19T08:09:09Z',
updatedAt: '2020-03-20T08:09:09Z', updatedAt: '2020-03-20T08:09:09Z',
state: 'OPENED', state: 'OPENED',
...@@ -50,6 +54,10 @@ export const requirement1 = { ...@@ -50,6 +54,10 @@ export const requirement1 = {
export const requirement2 = { export const requirement2 = {
iid: '2', iid: '2',
title: 'Est autem officium, quod ita factum est, ut eius facti probabilis ratio reddi possit.', title: 'Est autem officium, quod ita factum est, ut eius facti probabilis ratio reddi possit.',
titleHtml:
'Est autem officium, quod ita factum est, ut eius facti probabilis ratio reddi possit.',
description: 'ut eius facti _probabilis_ ratio reddi possit.',
descriptionHtml: 'ut eius facti <i>probabilis</i> ratio reddi possit.',
createdAt: '2020-03-19T08:08:14Z', createdAt: '2020-03-19T08:08:14Z',
updatedAt: '2020-03-20T08:08:14Z', updatedAt: '2020-03-20T08:08:14Z',
state: 'OPENED', state: 'OPENED',
...@@ -66,6 +74,9 @@ export const requirement2 = { ...@@ -66,6 +74,9 @@ export const requirement2 = {
export const requirement3 = { export const requirement3 = {
iid: '3', iid: '3',
title: 'Non modo carum sibi quemque, verum etiam vehementer carum esse', title: 'Non modo carum sibi quemque, verum etiam vehementer carum esse',
titleHtml: 'Non modo carum sibi quemque, verum etiam vehementer carum esse',
description: 'verum etiam _vehementer_ carum esse.',
descriptionHtml: 'verum etiam <i>vehementer</i> carum esse.',
createdAt: '2020-03-19T08:08:25Z', createdAt: '2020-03-19T08:08:25Z',
updatedAt: '2020-03-20T08:08:25Z', updatedAt: '2020-03-20T08:08:25Z',
state: 'OPENED', state: 'OPENED',
...@@ -82,6 +93,9 @@ export const requirement3 = { ...@@ -82,6 +93,9 @@ export const requirement3 = {
export const requirementArchived = { export const requirementArchived = {
iid: '23', iid: '23',
title: 'Cuius quidem, quoniam Stoicus fuit', title: 'Cuius quidem, quoniam Stoicus fuit',
titleHtml: 'Cuius quidem, quoniam Stoicus fuit',
description: 'quoniam _Stoicus_ fuit.',
descriptionHtml: 'quoniam <i>Stoicus</i> fuit.',
createdAt: '2020-03-31T13:31:40Z', createdAt: '2020-03-31T13:31:40Z',
updatedAt: '2020-03-31T13:31:40Z', updatedAt: '2020-03-31T13:31:40Z',
state: 'ARCHIVED', state: 'ARCHIVED',
......
...@@ -22312,6 +22312,9 @@ msgstr "" ...@@ -22312,6 +22312,9 @@ msgstr ""
msgid "Requirement %{reference} has been updated" msgid "Requirement %{reference} has been updated"
msgstr "" msgstr ""
msgid "Requirement title"
msgstr ""
msgid "Requirement title cannot have more than %{limit} characters." msgid "Requirement title cannot have more than %{limit} characters."
msgstr "" msgstr ""
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment