Skip to content

Commit

Permalink
Fixed Unable to Open User Home & Added Complete test coverage for Home (
Browse files Browse the repository at this point in the history
PalisadoesFoundation#1690)

* Create AddPostModal & Shift SVG constants

* Fix Bug in Rendering of Posts

* Added Complete test coverage for Home and StartPostModal

* Remove unecessary tests
  • Loading branch information
GlenDsza authored Mar 14, 2024
1 parent 42a225e commit cdaad16
Show file tree
Hide file tree
Showing 11 changed files with 772 additions and 625 deletions.
14 changes: 14 additions & 0 deletions src/assets/svgs/article.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
14 changes: 14 additions & 0 deletions src/assets/svgs/media.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
14 changes: 14 additions & 0 deletions src/assets/svgs/userEvent.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
41 changes: 4 additions & 37 deletions src/components/UserPortal/PostCard/PostCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,40 +18,7 @@ import SendIcon from '@mui/icons-material/Send';
import { errorHandler } from 'utils/errorHandler';
import CommentCard from '../CommentCard/CommentCard';
import useLocalStorage from 'utils/useLocalstorage';

interface InterfacePostCardProps {
id: string;
creator: {
firstName: string;
lastName: string;
email: string;
id: string;
};
image: string;
video: string;
text: string;
title: string;
likeCount: number;
commentCount: number;
comments: {
creator: {
_id: string;
firstName: string;
lastName: string;
email: string;
};
likeCount: number;
likedBy: {
id: string;
}[];
text: string;
}[];
likedBy: {
firstName: string;
lastName: string;
id: string;
}[];
}
import type { InterfacePostCard } from 'utils/interfaces';

interface InterfaceCommentCardProps {
id: string;
Expand All @@ -68,7 +35,7 @@ interface InterfaceCommentCardProps {
text: string;
}

export default function postCard(props: InterfacePostCardProps): JSX.Element {
export default function postCard(props: InterfacePostCard): JSX.Element {
const { t } = useTranslation('translation', {
keyPrefix: 'postCard',
});
Expand Down Expand Up @@ -173,7 +140,7 @@ export default function postCard(props: InterfacePostCardProps): JSX.Element {
};

return (
<>
<div data-testid="postCardContainer">
<Card className="my-3">
<Card.Header>
<div className={`${styles.cardHeader}`}>
Expand Down Expand Up @@ -289,6 +256,6 @@ export default function postCard(props: InterfacePostCardProps): JSX.Element {
</InputGroup>
</Modal.Body>
</Modal>
</>
</div>
);
}
50 changes: 50 additions & 0 deletions src/components/UserPortal/StartPostModal/StartPostModal.module.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,50 @@
.userImage {
display: flex;
width: 50px;
height: 50px;
align-items: center;
justify-content: center;
overflow: hidden;
border-radius: 50%;
position: relative;
border: 2px solid #31bb6b;
}

.userImage img {
position: absolute;
top: 0;
left: 0;
width: 100%;
scale: 1.5;
}

.previewImage {
overflow: hidden;
display: flex;
justify-content: center;
align-items: center;
margin-bottom: 1rem;
}

.previewImage img {
border-radius: 8px;
}

.icons {
width: 25px;
}

.icons svg {
stroke: #000;
}

.icons.dark {
cursor: pointer;
border: none;
outline: none;
background-color: transparent;
}

.icons.dark svg {
stroke: #000;
}
217 changes: 217 additions & 0 deletions src/components/UserPortal/StartPostModal/StartPostModal.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,217 @@
import React from 'react';
import { MockedProvider } from '@apollo/react-testing';
import type { RenderResult } from '@testing-library/react';
import { act, fireEvent, render, screen } from '@testing-library/react';
import { I18nextProvider } from 'react-i18next';
import userEvent from '@testing-library/user-event';
import { CREATE_POST_MUTATION } from 'GraphQl/Mutations/mutations';
import { Provider } from 'react-redux';
import { BrowserRouter } from 'react-router-dom';
import { toast } from 'react-toastify';
import { store } from 'state/store';
import { StaticMockLink } from 'utils/StaticMockLink';
import i18nForTest from 'utils/i18nForTest';
import StartPostModal from './StartPostModal';

jest.mock('react-toastify', () => ({
toast: {
error: jest.fn(),
info: jest.fn(),
success: jest.fn(),
},
}));

const MOCKS = [
{
request: {
query: CREATE_POST_MUTATION,
variables: {
title: 'Dummy Post',
text: 'This is dummy text',
organizationId: '123',
},
result: {
data: {
createPost: {
_id: '453',
},
},
},
},
},
];

const link = new StaticMockLink(MOCKS, true);

afterEach(() => {
localStorage.clear();
});

async function wait(ms = 100): Promise<void> {
await act(() => {
return new Promise((resolve) => {
setTimeout(resolve, ms);
});
});
}

const renderStartPostModal = (
visibility: boolean,
image: string | null,
): RenderResult => {
const cardProps = {
show: visibility,
onHide: jest.fn(),
fetchPosts: jest.fn(),
userData: {
user: {
__typename: 'User',
image: image,
firstName: 'Glen',
lastName: 'dsza',
email: 'glen@dsza.com',
appLanguageCode: 'en',
userType: 'USER',
pluginCreationAllowed: true,
adminApproved: true,
createdAt: '2023-02-18T09:22:27.969Z',
adminFor: [],
createdOrganizations: [],
joinedOrganizations: [],
createdEvents: [],
registeredEvents: [],
eventAdmin: [],
membershipRequests: [],
organizationsBlockedBy: [],
},
},
organizationId: '123',
};

return render(
<MockedProvider addTypename={false} link={link}>
<BrowserRouter>
<Provider store={store}>
<I18nextProvider i18n={i18nForTest}>
<StartPostModal {...cardProps} />
</I18nextProvider>
</Provider>
</BrowserRouter>
</MockedProvider>,
);
};

describe('Testing StartPostModal Component: User Portal', () => {
afterAll(() => {
jest.clearAllMocks();
});

test('Check if StartPostModal renders properly', async () => {
renderStartPostModal(true, null);

const modal = await screen.findByTestId('startPostModal');
expect(modal).toBeInTheDocument();
});

test('triggers file input when the upload icon is clicked', async () => {
const clickSpy = jest.spyOn(HTMLInputElement.prototype, 'click');
renderStartPostModal(true, null);

await wait();
const postImageInput = screen.getByTestId('postImageInput');
expect(postImageInput).toHaveAttribute('type', 'file');
expect(postImageInput).toHaveStyle({ display: 'inline-block' });

const iconButton = screen.getByTestId('addMediaBtn');
fireEvent.click(iconButton);

expect(clickSpy).toHaveBeenCalled();
clickSpy.mockRestore();
});

test('On invalid post submission with empty body Error toast should be shown', async () => {
const toastSpy = jest.spyOn(toast, 'error');
renderStartPostModal(true, null);
await wait();

userEvent.click(screen.getByTestId('createPostBtn'));
expect(toastSpy).toBeCalledWith("Can't create a post with an empty body.");
});

test('On valid post submission Info toast should be shown', async () => {
renderStartPostModal(true, null);
await wait();

const randomPostInput = 'test post input';
userEvent.type(screen.getByTestId('postInput'), randomPostInput);
expect(screen.queryByText(randomPostInput)).toBeInTheDocument();

userEvent.click(screen.getByTestId('createPostBtn'));

expect(toast.error).not.toBeCalledWith();
expect(toast.info).toBeCalledWith('Processing your post. Please wait.');
});

test('If user image is null then default image should be shown', async () => {
renderStartPostModal(true, null);
await wait();

const userImage = screen.getByTestId('userImage');
expect(userImage).toHaveAttribute(
'src',
'/src/assets/images/defaultImg.png',
);
});

test('If user image is not null then user image should be shown', async () => {
renderStartPostModal(true, 'image.png');
await wait();

const userImage = screen.getByTestId('userImage');
expect(userImage).toHaveAttribute('src', 'image.png');
});

test('should update post image state when a file is selected', async () => {
renderStartPostModal(true, null);
await wait();

const file = new File(['(⌐□_□)'], 'chad.png', { type: 'image/png' });
const input = screen.getByTestId('postImageInput') as HTMLInputElement;
await act(async () => {
fireEvent.change(input, { target: { files: [file] } });
});

expect(input.files?.[0]).toEqual(file);
const previewImage = await screen.findByAltText('Post Image Preview');
expect(previewImage).toBeInTheDocument();
});

test('should not update post image state when no file is selected', async () => {
renderStartPostModal(true, null);
await wait();

const input = screen.getByTestId('postImageInput') as HTMLInputElement;
await act(async () => {
fireEvent.change(input, { target: { files: null } });
});

const previewImage = screen.queryByAltText('Post Image Preview');
expect(previewImage).not.toBeInTheDocument();
});

test('triggers file input when fileInputRef exists', async () => {
const clickSpy = jest.spyOn(HTMLInputElement.prototype, 'click');
const refMock = { current: { click: clickSpy } };

renderStartPostModal(true, null);
await wait();

jest.spyOn(React, 'useRef').mockReturnValueOnce(refMock);

const iconButton = screen.getByTestId('addMediaBtn');
fireEvent.click(iconButton);

expect(clickSpy).toHaveBeenCalled();
clickSpy.mockRestore();
});
});
Loading

0 comments on commit cdaad16

Please sign in to comment.