Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat: Add Edit/Delete dropdown #3592

Merged
merged 19 commits into from
Oct 17, 2024
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
70 changes: 68 additions & 2 deletions packages/client/src/components/GrantNotes.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,11 +10,11 @@ import { id } from '@/helpers/testHelpers';
const CURRENT_USER_ID = id();

const getMockNotes = (count, hasMoreCursor = null, userId = id()) => ({
notes: Array.from(Array(count), () => ({
notes: Array.from(Array(count), (el, i) => ({
id: id(),
createdAt: new Date().toISOString(),
isRevised: true,
text: 'Text',
text: `Text ${i}`,
grant: {},
user: {
id: userId,
Expand Down Expand Up @@ -43,11 +43,18 @@ const mockStore = {
'grants/getNotesForGrant': vi.fn(),
'grants/getNotesForCurrentUser': vi.fn(),
'grants/saveNoteForGrant': vi.fn(),
'grants/deleteGrantNoteForUser': vi.fn(),
},
};

const store = createStore(mockStore);

const stubTextArea = {
template: '<textarea />',
props: { modelValue: String },
methods: { focus: vi.fn() },
};

beforeEach(() => {
vitest.clearAllMocks();
});
Expand Down Expand Up @@ -137,6 +144,65 @@ describe('GrantNotes component', () => {
expect(userNoteCmp.exists()).equal(true);
});

it('Correctly toggles edit mode for note input', async () => {
const userNote = getMockUserNote(1);
const allNotes = getMockNotes(2);

mockStore.actions['grants/getNotesForCurrentUser'].mockResolvedValue(userNote);
mockStore.actions['grants/getNotesForGrant'].mockResolvedValue(allNotes);

const wrapper = mount(GrantNotes, {
global: {
plugins: [store],
stubs: {
'b-form-textarea': stubTextArea,
},
},
});

await flushPromises();

const editBtn = wrapper.findComponent('[data-test-edit-note-btn]');
editBtn.trigger('click');

await flushPromises();

const form = wrapper.find('[data-test-edit-form]');
expect(form.exists()).toBe(true);
expect(wrapper.findComponent(stubTextArea).props('modelValue')).toBe(userNote.notes[0].text);
});

it('Correctly deletes a user note', async () => {
const userNote = getMockUserNote(1);
const allNotes = getMockNotes(2);

mockStore.actions['grants/getNotesForCurrentUser'].mockResolvedValue(userNote);
mockStore.actions['grants/getNotesForGrant'].mockResolvedValue(allNotes);
mockStore.actions['grants/deleteGrantNoteForUser'].mockResolvedValue();

const wrapper = mount(GrantNotes, {
global: {
plugins: [store],
stubs: {
'b-form-textarea': stubTextArea,
},
},
});

await flushPromises();

mockStore.actions['grants/getNotesForCurrentUser'].mockResolvedValue(getMockUserNote(0));

const deleteBtn = wrapper.findComponent('[data-test-delete-note-btn]');

deleteBtn.trigger('click');

await flushPromises();

expect(mockStore.actions['grants/deleteGrantNoteForUser']).toHaveBeenCalled();
expect(wrapper.findComponent(stubTextArea).props('modelValue')).toBe('');
});

it('Correctly retrieves more notes using "Show More"', async () => {
const userNote = getMockUserNote(1);

Expand Down
158 changes: 100 additions & 58 deletions packages/client/src/components/GrantNotes.vue
Original file line number Diff line number Diff line change
Expand Up @@ -3,51 +3,54 @@
<!-- Note Edit -->
<div
v-if="editingNote"
class="d-flex note-edit-container"
data-test-edit-form
class="note-edit-container"
>
<UserAvatar
:user-name="loggedInUser.name"
size="2.5rem"
:color="loggedInUser.avatar_color"
/>
<b-form-group class="ml-2 mb-2 flex-grow-1 position-relative">
<b-form-textarea
ref="noteTextarea"
v-model="noteText"
class="note-textarea"
placeholder="Leave a note with tips or barriers to applying..."
rows="2"
max-rows="8"
:formatter="formatter"
:disabled="submittingNote"
data-test-note-input
@keydown="handleKeyDown"
<div class="d-flex">
<UserAvatar
:user-name="loggedInUser.name"
size="2.5rem"
:color="loggedInUser.avatar_color"
/>
<b-button
ref="submitNoteBtn"
variant="link"
class="note-send-btn position-absolute px-2"
:disabled="noteSendBtnDisabled"
data-test-submit-btn
@click="submitNote"
>
<b-icon
class="p-0"
icon="send"
<b-form-group class="ml-2 mb-2 flex-grow-1 position-relative">
<b-form-textarea
ref="noteTextarea"
v-model="noteText"
class="note-textarea"
placeholder="Leave a note with tips or barriers to applying..."
rows="2"
max-rows="8"
:formatter="formatter"
:disabled="savingNote"
data-test-note-input
@keydown="handleKeyDown"
/>
</b-button>
<b-button
ref="submitNoteBtn"
variant="link"
class="note-send-btn position-absolute px-2"
:disabled="noteSendBtnDisabled"
data-test-submit-btn
@click="submitNote"
>
<b-icon
class="p-0"
icon="send"
/>
</b-button>

<template #description>
<div class="d-flex">
<small class="pl-2">
e.g. need co-applicants, we applied last year
</small>
<div :class="charCountClass">
{{ filteredNoteText.length }} / 300
<template #description>
<div class="d-flex">
<small class="pl-2">
e.g. need co-applicants, we applied last year
</small>
<div :class="charCountClass">
{{ filteredNoteText.length }} / 300
</div>
</div>
</div>
</template>
</b-form-group>
</template>
</b-form-group>
</div>
</div>

<!-- Current User's Note -->
Expand All @@ -58,17 +61,38 @@
:note="userNote"
>
<template #actions>
<b-button
class="note-edit-btn p-0"
<b-dropdown
right
variant="link"
@click="toggleEditNote"
toggle-class="p-0"
no-caret
:disabled="savingNote"
>
<b-icon
icon="pencil-square"
class="mr-1"
/>
<span>EDIT</span>
</b-button>
<template #button-content>
<span class="note-edit-btn">
<b-icon
icon="pencil-square"
class="mr-1"
/>
<span class="note-edit-btn-text">EDIT</span>
</span>
</template>
<b-dropdown-item-button
title="Edit note"
data-test-edit-note-btn
@click="toggleEditNote"
>
Edit
</b-dropdown-item-button>
<b-dropdown-item-button
title="Delete note"
variant="danger"
data-test-delete-note-btn
@click="deleteUserNote"
>
Delete
</b-dropdown-item-button>
</b-dropdown>
</template>
</GrantNote>

Expand Down Expand Up @@ -121,7 +145,7 @@ export default {
otherNotes: [],
userNote: null,
noteText: '',
submittingNote: false,
savingNote: false,
editingNote: false,
notesNextCursor: null,
};
Expand All @@ -141,7 +165,7 @@ export default {
return this.filteredNoteText.length === 0;
},
noteSendBtnDisabled() {
return this.emptyNoteText || this.submittingNote;
return this.emptyNoteText || this.savingNote;
},
charCountClass() {
const errColor = this.filteredNoteText.length === 300 ? 'text-error' : '';
Expand All @@ -163,13 +187,13 @@ export default {
getNotesForGrant: 'grants/getNotesForGrant',
getNotesForCurrentUser: 'grants/getNotesForCurrentUser',
saveNoteForGrant: 'grants/saveNoteForGrant',
deleteGrantNoteForUser: 'grants/deleteGrantNoteForUser',
}),
formatter(value) {
return value.substring(0, 300);
},
async toggleEditNote() {
this.editingNote = true;
this.noteText = this.userNote.text;

await nextTick();
this.$refs.noteTextarea.focus();
Expand All @@ -180,7 +204,7 @@ export default {
}
},
async submitNote() {
this.submittingNote = true;
this.savingNote = true;

try {
await this.saveNoteForGrant({ grantId: this.currentGrant.grant_id, text: this.filteredNoteText });
Expand All @@ -190,13 +214,16 @@ export default {
// Error already logged
}

this.submittingNote = false;
this.savingNote = false;
},
async fetchUsersNote() {
const result = await this.getNotesForCurrentUser({ grantId: this.currentGrant.grant_id });

setUserNote(result) {
this.userNote = result && result.notes.length ? result.notes[0] : null;
this.editingNote = !this.userNote;
this.noteText = this.userNote ? this.userNote.text : '';
},
async fetchUsersNote() {
const result = await this.getNotesForCurrentUser({ grantId: this.currentGrant.grant_id });
this.setUserNote(result);
},
async fetchNextNotes() {
const query = {
Expand All @@ -216,6 +243,17 @@ export default {
this.notesNextCursor = result.pagination.next;
}
},
async deleteUserNote() {
this.savingNote = true;
try {
await this.deleteGrantNoteForUser({ grantId: this.currentGrant.grant_id });
this.$emit('noteSaved');
this.setUserNote(null);
} catch (e) {
// Error already logged
}
this.savingNote = false;
},
},
};
</script>
Expand Down Expand Up @@ -250,10 +288,14 @@ textarea.note-textarea::placeholder {
}

.note-edit-btn {
font-size: 0.875rem;
vertical-align: middle;
color: $raw-gray-500
}

.note-edit-btn-text {
font-size: 0.75rem;
}

.note-send-btn {
bottom: 1.5625rem;
right: 0;
Expand Down
11 changes: 11 additions & 0 deletions packages/client/src/store/modules/grants.js
Original file line number Diff line number Diff line change
Expand Up @@ -282,6 +282,17 @@ export default {
throw e;
}
},
async deleteGrantNoteForUser({ rootGetters, commit }, { grantId }) {
try {
const userId = rootGetters['users/loggedInUser']?.id;
await fetchApi.deleteRequest(`/api/organizations/${rootGetters['users/selectedAgencyId']}/grants/${grantId}/notes/user/${userId}`);
} catch (e) {
const text = `Error deleting grant note for user: ${e.message}`;
commit('alerts/addAlert', { text, level: 'err' }, { root: true });
datadogRum.addError(e, { grantId, text });
throw e;
}
},
async followGrantForCurrentUser({ rootGetters, commit }, { grantId }) {
try {
await fetchApi.put(`/api/organizations/${rootGetters['users/selectedAgencyId']}/grants/${grantId}/follow`);
Expand Down
28 changes: 28 additions & 0 deletions packages/server/__tests__/api/grants.test.js
Original file line number Diff line number Diff line change
Expand Up @@ -990,6 +990,34 @@ HHS-2021-IHS-TPI-0001,Community Health Aide Program: Tribal Planning &`;
});
});
});
context('DELETE /:grantId/notes/user/:userId', () => {
const GRANT_ID = '335255';

let staffGrantNote;

beforeEach(async () => {
[staffGrantNote] = await knex('grant_notes')
.insert({ grant_id: GRANT_ID, user_id: staffUser.id }, 'id');

await knex('grant_notes_revisions')
.insert({ grant_note_id: staffGrantNote.id, text: 'Test note 1.' });

await knex('grant_notes_revisions')
.insert({ grant_note_id: staffGrantNote.id, text: 'Test note 2.' });
});

it('Deletes all notes for user', async () => {
const revisions = await knex('grant_notes_revisions').where('grant_note_id', staffGrantNote.id);
expect(revisions).to.have.length(2);

const resp = await fetchApi(`/grants/${GRANT_ID}/notes/user/${staffUser.id}`, agencies.own, {
...fetchOptions.staff,
method: 'delete',
});

expect(resp.statusText).to.equal('OK');
});
});
context('GET /:grantId/followers', () => {
const GRANT_ID = 335255;

Expand Down
Loading
Loading