Skip to content

Commit

Permalink
UIREQMED-42: Add Decline Action for Mediated request with status of N…
Browse files Browse the repository at this point in the history
…ew - Awaiting confirmation
  • Loading branch information
Dmitriy-Litvinenko committed Nov 8, 2024
1 parent eff3995 commit d1f7c1e
Show file tree
Hide file tree
Showing 9 changed files with 209 additions and 4 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@
* Persist filters when changing between actions. Refs UIREQMED-44.
* Implement functionality for editing mediated request. Refs UIREQMED-22.
* *BREAKING* Migrate to new `mod-circulation-bff` endpoints. Refs UIREQMED-39.
* Add Decline Action for Mediated request with status of New - Awaiting confirmation. Refs UIREQMED-42.

## [1.1.0](https://github.com/folio-org/ui-requests-mediated/tree/v1.1.0) (2024-10-30)
* Update github actions. Refs UIREQMED-14.
Expand Down
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -171,7 +171,7 @@
],
"subPermissions": [
"ui-requests-mediated.view",
"requests-mediated.mediated-request.item.delete"
"requests-mediated.decline-mediated-request.execute"
],
"visible": true
},
Expand Down
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
import {
useState,
} from 'react';
import PropTypes from 'prop-types';
import {
useHistory,
Expand All @@ -14,6 +17,8 @@ import {
import {
IfPermission,
TitleManager,
useOkapiKy,
useStripes,

Check warning on line 21 in src/components/MediatedRequestsActivities/components/MediatedRequestsDetail/MediatedRequestsDetail.js

View workflow job for this annotation

GitHub Actions / build-npm

'useStripes' is defined but never used. Allowed unused vars must match /React/u

Check warning on line 21 in src/components/MediatedRequestsActivities/components/MediatedRequestsDetail/MediatedRequestsDetail.js

View workflow job for this annotation

GitHub Actions / build-npm

'useStripes' is defined but never used. Allowed unused vars must match /React/u
} from '@folio/stripes/core';
import {
Button,
Expand All @@ -31,6 +36,7 @@ import TitleInformation from '../TitleInformation';
import MediatedRequestInformation from '../MediatedRequestInformation';
import ItemDetail from '../ItemDetail';
import UserDetail from '../UserDetail';
import DeclineModal from './components/DeclineModal';
import {
useMediatedRequestById,
useUserById,
Expand Down Expand Up @@ -64,22 +70,40 @@ const MediatedRequestsDetail = ({
stripes,
patronGroups,
setRequest,
update,
}) => {
const history = useHistory();
const location = useLocation();
const ky = useOkapiKy();
const mediatedRequestIdFromPathname = location.pathname.substring(location.pathname.lastIndexOf('/') + 1);

const { formatMessage } = useIntl();
const {
mediatedRequest,
isFetching,
shouldUpdate,
setShouldUpdate,
} = useMediatedRequestById(mediatedRequestIdFromPathname);
const { userData } = useUserById(mediatedRequest?.requesterId, isFetching);
const { servicePoints } = useServicePoints();
const mediatedRequestsActivitiesUrl = getMediatedRequestsActivitiesUrl();

setRequest(mediatedRequest);

const [declineModalOpen, setDeclineModalOpen] = useState(false);
const onOpenDeclineModal = () => setDeclineModalOpen(true);
const onConfirmDeclineModal = () => {
ky.post(`requests-mediated/mediated-requests/${mediatedRequestIdFromPathname}/decline`)
.then(() => {
update();
setShouldUpdate(shouldUpdate + 1);
setDeclineModalOpen(false);
})
.catch(() => {
setDeclineModalOpen(false);
});
};
const onCloseDeclineModal = () => setDeclineModalOpen(false);
const isActionMenuVisible = () => (
get(mediatedRequest, MEDIATED_REQUESTS_RECORD_FIELD_PATH[MEDIATED_REQUESTS_RECORD_FIELD_NAME.STATUS], DEFAULT_VIEW_VALUE) === MEDIATED_REQUEST_STATUS.NEW_AWAITING_CONFIRMATION
? stripes.hasPerm('ui-requests-mediated.requests-mediated.view-confirm.execute') || stripes.hasPerm('ui-requests-mediated.requests-mediated.view-create-edit.execute') || stripes.hasPerm('ui-requests-mediated.requests-mediated.view-decline.execute')
Expand Down Expand Up @@ -114,7 +138,10 @@ const MediatedRequestsDetail = ({
<Button
buttonStyle="dropdownItem"
marginBottom0
onClick={onToggle}
onClick={() => {
onOpenDeclineModal();
onToggle();
}}
>
<Icon icon={ICONS.TIMES_CIRCLE}>
<FormattedMessage id="ui-requests-mediated.mediatedRequestDetails.actionMenu.decline" />
Expand Down Expand Up @@ -218,6 +245,12 @@ const MediatedRequestsDetail = ({
hideAssignButton
/>
</AccordionStatus>
<DeclineModal
open={declineModalOpen}
title={get(mediatedRequest, MEDIATED_REQUESTS_RECORD_FIELD_PATH[MEDIATED_REQUESTS_RECORD_FIELD_NAME.TITLE], DEFAULT_VIEW_VALUE)}
onConfirm={onConfirmDeclineModal}
onClose={onCloseDeclineModal}
/>
</>
}
</Pane>
Expand All @@ -227,6 +260,7 @@ const MediatedRequestsDetail = ({
MediatedRequestsDetail.propTypes = {
stripes: PropTypes.object.isRequired,
setRequest: PropTypes.func.isRequired,
update: PropTypes.func.isRequired,
patronGroups: PropTypes.arrayOf(PropTypes.shape({
id: PropTypes.string,
group: PropTypes.string,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,57 @@
import PropTypes from 'prop-types';
import { FormattedMessage } from 'react-intl';
import {
Button,
Modal,
ModalFooter,
} from '@folio/stripes/components';

const DeclineModal = ({
open,
title,
onConfirm,
onClose,
}) => {
const footer = (
<ModalFooter>
<Button
data-testid="confirmButton"
buttonStyle="primary"
onClick={onConfirm}
>
<FormattedMessage id="ui-requests-mediated.declineModal.confirm" />
</Button>
<Button
data-testid="backButton"
onClick={onClose}
>
<FormattedMessage id="ui-requests-mediated.declineModal.back" />
</Button>
</ModalFooter>
);

return (
<Modal
dismissible
label={<FormattedMessage id="ui-requests-mediated.declineModal.title" />}
open={open}
size="small"
footer={footer}
onClose={onClose}
>
<FormattedMessage
id="ui-requests-mediated.declineModal.message"
values={{ title }}
/>
</Modal>
);
};

DeclineModal.propTypes = {
open: PropTypes.bool.isRequired,
title: PropTypes.string.isRequired,
onConfirm: PropTypes.func.isRequired,
onClose: PropTypes.func.isRequired,
};

export default DeclineModal;
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
import {
render,
screen,
fireEvent,
} from '@folio/jest-config-stripes/testing-library/react';

import DeclineModal from './DeclineModal';

const testIds = {
confirmButton: 'confirmButton',
backButton: 'backButton',
};
const messageIds = {
title: 'ui-requests-mediated.declineModal.title',
message: 'ui-requests-mediated.declineModal.message',
confirmButton: 'ui-requests-mediated.declineModal.confirm',
backButton: 'ui-requests-mediated.declineModal.back',
};
const defaultProps = {
open: true,
title: 'Title',
onConfirm: jest.fn(),
onClose: jest.fn(),
};

describe('DeclineModal', () => {
afterEach(() => {
jest.clearAllMocks();
});

beforeEach(() => {
render(
<DeclineModal {...defaultProps} />
);
});

it('should render title', () => {
expect(screen.getByText(messageIds.title)).toBeVisible();
});

it('should render message', () => {
expect(screen.getByText(messageIds.message)).toBeVisible();
});

describe('Confirm button', () => {
it('should render confirm button', () => {
expect(screen.getByTestId(testIds.confirmButton)).toBeVisible();
});

it('should render confirm button text', () => {
expect(screen.getByText(messageIds.confirmButton)).toBeVisible();
});

it('should call onConfirm', () => {
fireEvent.click(screen.getByTestId(testIds.confirmButton));

expect(defaultProps.onConfirm).toHaveBeenCalled();
});
});

describe('Back button', () => {
it('should render back button', () => {
expect(screen.getByTestId(testIds.backButton)).toBeVisible();
});

it('should render back button text', () => {
expect(screen.getByText(messageIds.backButton)).toBeVisible();
});

it('should call onClose', () => {
fireEvent.click(screen.getByTestId(testIds.backButton));

expect(defaultProps.onClose).toHaveBeenCalled();
});
});
});
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export { default } from './DeclineModal';
10 changes: 9 additions & 1 deletion src/hooks/useMediatedRequestById/useMediatedRequestById.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
import {
useState,
} from 'react';

import { useQuery } from 'react-query';

import {
Expand All @@ -6,20 +10,24 @@ import {
} from '@folio/stripes/core';

const useMediatedRequestById = (mediatedRequestId) => {
const [shouldUpdate, setShouldUpdate] = useState(0);

const ky = useOkapiKy();
const [namespace] = useNamespace({ key: 'mediatedRequest' });
const {
data,
isFetching,
} = useQuery(
[namespace, mediatedRequestId],
[namespace, mediatedRequestId, shouldUpdate],
() => ky.get(`requests-mediated/mediated-requests/${mediatedRequestId}`).json(),
{ enabled: Boolean(mediatedRequestId) },
);

return {
isFetching,
mediatedRequest: data,
shouldUpdate,
setShouldUpdate,
};
};

Expand Down
25 changes: 24 additions & 1 deletion src/routes/MediatedRequestsActivitiesContainer.js
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,8 @@ class MediatedRequestsActivitiesContainer extends React.Component {
query: buildQuery,
},
},
// shouldRefresh: () => true,
resourceShouldRefresh: true,
throwErrors: false,
},
reportRecords: {
Expand Down Expand Up @@ -161,6 +163,21 @@ class MediatedRequestsActivitiesContainer extends React.Component {
}
};

update = () => {
const {
location,
history,
} = this.props;
const {
resources: {
query,
},
} = this.source;
const url = buildUrl(location, query);

history.push(url);
};

render() {
if (this.source) {
this.source.update(this.props, MEDIATED_REQUESTS_RECORDS_NAME);
Expand All @@ -176,7 +193,13 @@ class MediatedRequestsActivitiesContainer extends React.Component {
mutator={this.props.mutator}
settings={this.props.settings}
>
{this.props.children}
{
React.Children.map(
this.props.children, child => React.cloneElement(child, {
update: this.update,
})
)
}
</MediatedRequestsActivities>
);
}
Expand Down
5 changes: 5 additions & 0 deletions translations/ui-requests-mediated/en.json
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,11 @@
"confirmItem.errorModal.sendItemInTransit.message": "No \"Send item in transit\" transaction could be found for this item",
"confirmItem.errorModal.close": "Close",

"declineModal.title": "Confirm Mediated request decline",
"declineModal.message": "<b>{title}</b> will be <b>declined</b>",
"declineModal.confirm": "Confirm",
"declineModal.back": "Back",

"confirmItemList.columnName.arrivalDate": "Arrival date",
"confirmItemList.columnName.inTransitDate": "In transit date",
"confirmItemList.columnName.title": "Title",
Expand Down

0 comments on commit d1f7c1e

Please sign in to comment.