Skip to content

Commit

Permalink
Merge pull request #139 from Real-Dev-Squad/develop
Browse files Browse the repository at this point in the history
Dev to Main Sync
  • Loading branch information
iamitprakash authored Aug 29, 2024
2 parents d56f64a + c2472d8 commit efb8f3d
Show file tree
Hide file tree
Showing 56 changed files with 1,429 additions and 924 deletions.
32 changes: 0 additions & 32 deletions __tests__/components/App/InputSection.test.tsx

This file was deleted.

141 changes: 116 additions & 25 deletions __tests__/components/App/OutputSection.test.tsx
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
import '@testing-library/jest-dom/extend-expect';

import { fireEvent, render, screen } from '@testing-library/react';

import OutputSection from '@/components/App/OutputSection';
Expand All @@ -6,21 +8,65 @@ describe('OutputSection component', () => {
const shortUrl = 'https://rds.li/123456';
const originalUrl = 'https://status.realdevsquad.com/task/details/josuets45sds';

const mockHandleCopyUrl = jest.fn();
const mockHandleCreateNew = jest.fn();

beforeAll(() => {
Object.assign(navigator, {
clipboard: {
writeText: jest.fn().mockResolvedValue(() => Promise.resolve()),
},
});
});

beforeEach(() => {
HTMLCanvasElement.prototype.getContext = jest.fn().mockReturnValue({
fillRect: jest.fn(),
clearRect: jest.fn(),
getImageData: jest.fn(),
putImageData: jest.fn(),
createImageData: jest.fn(),
setTransform: jest.fn(),
drawImage: jest.fn(),
save: jest.fn(),
fillText: jest.fn(),
restore: jest.fn(),
beginPath: jest.fn(),
moveTo: jest.fn(),
lineTo: jest.fn(),
closePath: jest.fn(),
stroke: jest.fn(),
translate: jest.fn(),
scale: jest.fn(),
rotate: jest.fn(),
arc: jest.fn(),
fill: jest.fn(),
measureText: jest.fn().mockReturnValue({ width: 0 }),
transform: jest.fn(),
rect: jest.fn(),
clip: jest.fn(),
isPointInPath: jest.fn(),
isPointInStroke: jest.fn(),
canvas: document.createElement('canvas'),
});

HTMLCanvasElement.prototype.toDataURL = jest.fn().mockReturnValue('');

jest.clearAllMocks();
});

it('renders OutputSection component correctly', () => {
render(
<OutputSection
shortUrl={shortUrl}
originalUrl={originalUrl}
isLoaded={true}
handleCopyUrl={mockHandleCopyUrl}
handleCreateNew={mockHandleCreateNew}
/>
);

expect(screen.getByTestId('copy-button')).toBeInTheDocument();
expect(screen.getByTestId('share-button')).toBeInTheDocument();
expect(screen.getByTestId('output-heading')).toHaveTextContent('Your shortened URL is ready!');
});

it('calls handleCopyUrl function on button click', () => {
Expand All @@ -29,14 +75,14 @@ describe('OutputSection component', () => {
shortUrl={shortUrl}
originalUrl={originalUrl}
isLoaded={true}
handleCopyUrl={mockHandleCopyUrl}
handleCreateNew={mockHandleCopyUrl}
handleCreateNew={mockHandleCreateNew}
/>
);

const copyButton = screen.getByTestId('copy-button');
fireEvent.click(copyButton);
expect(mockHandleCopyUrl).toHaveBeenCalled();

expect(navigator.clipboard.writeText).toHaveBeenCalledWith(shortUrl);
});

it('opens a new tab when share button is clicked', () => {
Expand All @@ -45,8 +91,7 @@ describe('OutputSection component', () => {
shortUrl={shortUrl}
originalUrl={originalUrl}
isLoaded={true}
handleCopyUrl={mockHandleCopyUrl}
handleCreateNew={mockHandleCopyUrl}
handleCreateNew={mockHandleCreateNew}
/>
);

Expand All @@ -55,52 +100,98 @@ describe('OutputSection component', () => {
expect(shareButton).toHaveAttribute('target', '_blank');
});

it('renders create new button when window width is less than 768px', () => {
it('renders social media share links', () => {
render(
<OutputSection
shortUrl={shortUrl}
originalUrl={originalUrl}
isLoaded={true}
handleCopyUrl={mockHandleCopyUrl}
handleCreateNew={mockHandleCreateNew}
/>
);

Object.defineProperty(window, 'innerWidth', { writable: true, configurable: true, value: 500 });
fireEvent(window, new Event('resize'));

const createNewButton = screen.getByText('Create New');
expect(createNewButton).toBeInTheDocument();
expect(screen.getByTestId('twitter-share')).toBeInTheDocument();
expect(screen.getByTestId('discord-share')).toBeInTheDocument();
expect(screen.getByTestId('linkedin-share')).toBeInTheDocument();
expect(screen.getByTestId('whatsapp-share')).toBeInTheDocument();
});
it('renders "Create New" button and calls the onClick handler when clicked', () => {

it('renders shimmer when isLoaded is false', () => {
render(
<OutputSection
originalUrl={originalUrl}
shortUrl={shortUrl}
isLoaded={false}
handleCreateNew={mockHandleCreateNew}
/>
);

const shimmer = screen.getByTestId('output-section-shimmer');
expect(shimmer).toBeInTheDocument();
});

it('updates the button text on download click', async () => {
render(
<OutputSection
shortUrl="https://example.com/short-url"
originalUrl="https://example.com/original-url"
isLoaded={true}
handleCopyUrl={mockHandleCopyUrl}
handleCreateNew={mockHandleCreateNew}
/>
);
const downloadButton = screen.getByTestId('download-button');
fireEvent.click(downloadButton);

const createNewButton = screen.getByText('Create New');
expect(createNewButton).toBeInTheDocument();
fireEvent.click(createNewButton);
expect(mockHandleCreateNew).toHaveBeenCalled();
const updatedText = await screen.findByText('Downloaded');
expect(updatedText).toBeInTheDocument();
});

it('renders shimmer when isLoaded is false', () => {
it('triggers the download process correctly', () => {
render(
<OutputSection
shortUrl={shortUrl}
originalUrl={originalUrl}
isLoaded={true}
handleCreateNew={mockHandleCreateNew}
/>
);

const downloadButton = screen.getByTestId('download-button');
fireEvent.click(downloadButton);
expect(HTMLCanvasElement.prototype.toDataURL).toHaveBeenCalled();
});

it('does nothing when canvas is not found during download', () => {
document.getElementById = jest.fn().mockReturnValue(null);

render(
<OutputSection
shortUrl={shortUrl}
isLoaded={false}
handleCopyUrl={mockHandleCopyUrl}
originalUrl={originalUrl}
isLoaded={true}
handleCreateNew={mockHandleCreateNew}
/>
);

const shimmer = screen.getByTestId('output-section-shimmer');
expect(shimmer).toBeInTheDocument();
const downloadButton = screen.getByTestId('download-button');
fireEvent.click(downloadButton);

expect(HTMLCanvasElement.prototype.toDataURL).not.toHaveBeenCalled();
});

it('does nothing when shortUrl is empty during copy', () => {
render(
<OutputSection
shortUrl=""
originalUrl={originalUrl}
isLoaded={true}
handleCreateNew={mockHandleCreateNew}
/>
);

const copyButton = screen.getByTestId('copy-button');
fireEvent.click(copyButton);

expect(navigator.clipboard.writeText).not.toHaveBeenCalled();
});
});
134 changes: 134 additions & 0 deletions __tests__/components/App/SortenUrlForm.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,134 @@
import { fireEvent, render, screen } from '@testing-library/react';

import ShortenUrlForm, { HomeText } from '@/components/App/ShortenUrlForm';

describe('UrlForm component', () => {
const testUrl = 'https://stackoverflow.com/questions/26944762/when-to-use-chore-as-type-of-commit-message';

it('renders UrlForm component correctly', () => {
const mockSetUrl = jest.fn();
const mockOnSubmit = jest.fn();
const mockClearError = jest.fn();
render(
<ShortenUrlForm
url={testUrl}
setUrl={mockSetUrl}
onSubmit={mockOnSubmit}
error={null}
clearError={mockClearError}
loading={false}
/>
);
expect(screen.getByPlaceholderText('Enter the URL')).toBeInTheDocument();
expect(screen.getByTestId('shorten-button')).toBeInTheDocument();
});

it('calls setUrl function on input change and clearError', () => {
const mockSetUrl = jest.fn();
const mockOnSubmit = jest.fn();
const mockClearError = jest.fn();
render(
<ShortenUrlForm
url={testUrl}
setUrl={mockSetUrl}
onSubmit={mockOnSubmit}
error={null}
clearError={mockClearError}
loading={false}
/>
);
const inputElement = screen.getByPlaceholderText('Enter the URL');
fireEvent.change(inputElement, { target: { value: 'https://realdevsquad.com' } });
expect(mockSetUrl).toHaveBeenCalledWith('https://realdevsquad.com');
expect(mockClearError).toHaveBeenCalled();
});

it('calls onSubmit function on button click', () => {
const mockSetUrl = jest.fn();
const mockOnSubmit = jest.fn();
const mockClearError = jest.fn();
render(
<ShortenUrlForm
url={testUrl}
setUrl={mockSetUrl}
onSubmit={mockOnSubmit}
error={null}
clearError={mockClearError}
loading={false}
/>
);
const generateButton = screen.getByTestId('shorten-button');
fireEvent.click(generateButton);
expect(mockOnSubmit).toHaveBeenCalledWith(testUrl);
});

it('does not call onSubmit function if URL is empty', () => {
const mockSetUrl = jest.fn();
const mockOnSubmit = jest.fn();
const mockClearError = jest.fn();

render(
<ShortenUrlForm
url=""
setUrl={mockSetUrl}
onSubmit={mockOnSubmit}
error={null}
clearError={mockClearError}
loading={false}
/>
);

const generateButton = screen.getByTestId('shorten-button');
fireEvent.click(generateButton);

expect(mockOnSubmit).not.toHaveBeenCalled();
});

it('renders error message if error prop is passed', () => {
const errorMessage = 'Enter a valid URL';
render(
<ShortenUrlForm
url={testUrl}
setUrl={jest.fn()}
onSubmit={jest.fn()}
error={errorMessage}
clearError={jest.fn()}
loading={false}
/>
);
const errorElement = screen.getByText(errorMessage);
expect(errorElement).toBeInTheDocument();
expect(errorElement).toHaveTextContent(errorMessage);
});

it('renders the main heading correctly', () => {
render(<HomeText />);
const mainHeading = screen.getByText('Shorten Your URL');
expect(mainHeading).toBeInTheDocument();
expect(mainHeading).toHaveClass(
'text-3xl md:text-6xl xl:text-7xl sm:text-5xl text-center text-white font-semibold pb-2 lg:pb-4'
);
});

it('renders the subheading correctly', () => {
render(<HomeText />);
const subHeading = screen.getByText('Perfect Links Every Time');
expect(subHeading).toBeInTheDocument();
expect(subHeading).toHaveClass(
'text-2xl sm:text-3xl md:text-4xl xl:text-5xl text-center text-white font-semibold'
);
});

it('renders the paragraph text correctly', () => {
render(<HomeText />);
const paragraph = screen.getByText(/Ready to shorten your URL\? Enter your/i);
expect(paragraph).toBeInTheDocument();
expect(paragraph).toHaveClass('xl:text-xl text-base text-white mt-4 text-center');
});

it('renders the paragraph text with a line break for small screens', () => {
render(<HomeText />);
const paragraph = screen.getByText(/Ready to shorten your URL\? Enter your/i);
expect(paragraph.innerHTML).toContain('<br class="sm:hidden">');
});
});
Loading

0 comments on commit efb8f3d

Please sign in to comment.