diff --git a/ui/src/components/EnhancedPagination/PageChangeButton.tsx b/ui/src/components/EnhancedPagination/PageChangeButton.tsx
new file mode 100644
index 00000000..3a691edf
--- /dev/null
+++ b/ui/src/components/EnhancedPagination/PageChangeButton.tsx
@@ -0,0 +1,56 @@
+import { Button, Icon } from '@chakra-ui/react';
+import { FiChevronLeft, FiChevronRight, FiChevronsLeft, FiChevronsRight } from 'react-icons/fi';
+
+export enum PAGE_CHANGE_BUTTON_TYPE {
+ PREVIOUS = 'previous',
+ NEXT = 'next',
+ FIRST = 'first',
+ LAST = 'last',
+}
+
+type PageButtonProps = {
+ type: PAGE_CHANGE_BUTTON_TYPE;
+ onClick: () => void;
+ isEnabled?: boolean;
+};
+
+const PageChangeButton = ({ type, onClick, isEnabled = false }: PageButtonProps) => {
+ const iconMap = {
+ [PAGE_CHANGE_BUTTON_TYPE.PREVIOUS]: { icon: FiChevronLeft, value: 'previous' },
+ [PAGE_CHANGE_BUTTON_TYPE.FIRST]: { icon: FiChevronsLeft, value: 'first' },
+ [PAGE_CHANGE_BUTTON_TYPE.LAST]: { icon: FiChevronsRight, value: 'last' },
+ [PAGE_CHANGE_BUTTON_TYPE.NEXT]: { icon: FiChevronRight, value: 'next' },
+ };
+
+ const icon = iconMap[type] || iconMap[PAGE_CHANGE_BUTTON_TYPE.NEXT];
+
+ return (
+
+ );
+};
+
+export default PageChangeButton;
diff --git a/ui/src/components/EnhancedPagination/PageNumberItem.tsx b/ui/src/components/EnhancedPagination/PageNumberItem.tsx
new file mode 100644
index 00000000..c8ed9e58
--- /dev/null
+++ b/ui/src/components/EnhancedPagination/PageNumberItem.tsx
@@ -0,0 +1,42 @@
+import { Box, Text } from '@chakra-ui/react';
+
+const PageNumberItem = ({
+ isActive = false,
+ onClick,
+ value,
+ isEllipsis = false,
+}: {
+ isActive?: boolean;
+ onClick?: () => void;
+ value?: number;
+ isEllipsis?: boolean;
+}) => (
+ {} : onClick}
+ _hover={{ backgroundColor: 'gray.400', cursor: 'pointer' }}
+ _disabled={{
+ _hover: { cursor: 'not-allowed' },
+ backgroundColor: 'gray.400',
+ }}
+ data-testid={isEllipsis ? 'ellipsis' : `page-number-${value}`}
+ >
+
+ {isEllipsis ? '...' : value}
+
+
+);
+
+export default PageNumberItem;
diff --git a/ui/src/components/EnhancedPagination/Pagination.tsx b/ui/src/components/EnhancedPagination/Pagination.tsx
new file mode 100644
index 00000000..2aee2d87
--- /dev/null
+++ b/ui/src/components/EnhancedPagination/Pagination.tsx
@@ -0,0 +1,101 @@
+import { LinksType } from '@/services/common';
+import { Stack, Box } from '@chakra-ui/react';
+import PageNumberItem from './PageNumberItem';
+import PageChangeButton, { PAGE_CHANGE_BUTTON_TYPE } from './PageChangeButton';
+
+const getPage = (url: string): number => {
+ const parser = new URLSearchParams(new URL(url).search);
+ return parseInt(parser.get('page') || '0');
+};
+
+export type PaginationProps = {
+ links: LinksType;
+ currentPage: number;
+ handlePageChange: (pageNumber: number) => void;
+};
+
+const Pagination = ({ links, currentPage, handlePageChange }: PaginationProps) => {
+ const firstPage = getPage(links.first);
+ const lastPage = getPage(links.last);
+
+ const renderPageNumbers = () => {
+ if (currentPage === firstPage) {
+ return (
+ <>
+ handlePageChange(currentPage)}
+ />
+ {lastPage > firstPage + 1 && }
+ {lastPage !== firstPage && (
+ handlePageChange(lastPage)} />
+ )}
+ >
+ );
+ } else if (currentPage === lastPage) {
+ return (
+ <>
+ handlePageChange(firstPage)} />
+ {lastPage > firstPage + 1 && }
+ handlePageChange(currentPage)}
+ />
+ >
+ );
+ } else {
+ return (
+ <>
+ handlePageChange(firstPage)} />
+ {currentPage > firstPage + 1 && }
+ handlePageChange(currentPage)}
+ />
+ {currentPage < lastPage - 1 && }
+ handlePageChange(lastPage)} />
+ >
+ );
+ }
+ };
+
+ return (
+
+
+ handlePageChange(firstPage)}
+ />
+ handlePageChange(currentPage - 1)}
+ />
+ {renderPageNumbers()}
+ handlePageChange(currentPage + 1)}
+ />
+ handlePageChange(lastPage)}
+ />
+
+
+ );
+};
+
+export default Pagination;
diff --git a/ui/src/components/EnhancedPagination/__tests__/Pagination.test.tsx b/ui/src/components/EnhancedPagination/__tests__/Pagination.test.tsx
new file mode 100644
index 00000000..a41a037d
--- /dev/null
+++ b/ui/src/components/EnhancedPagination/__tests__/Pagination.test.tsx
@@ -0,0 +1,86 @@
+import { fireEvent, render, screen } from '@testing-library/react';
+import { expect } from '@jest/globals';
+import '@testing-library/jest-dom';
+
+import Pagination, { PaginationProps } from '../Pagination';
+
+describe('Pagination', () => {
+ const mockHandlePageChange = jest.fn();
+ const defaultProps: PaginationProps = {
+ links: {
+ self: 'http://localhost:3000/api/v1/connectors?',
+ first: 'http://localhost:3000/api/v1/connectors?page=1&per_page=10',
+ prev: null,
+ next: 'http://localhost:3000/api/v1/connectors?page=2&per_page=10',
+ last: 'http://localhost:3000/api/v1/connectors?page=10&per_page=10',
+ },
+ currentPage: 5,
+ handlePageChange: mockHandlePageChange,
+ };
+
+ beforeEach(() => {
+ mockHandlePageChange.mockClear();
+ });
+
+ it('should render correct page numbers for middle page', () => {
+ render();
+ expect(screen.getByTestId('page-number-1')).toBeTruthy();
+ expect(screen.getByTestId('page-number-5')).toBeTruthy();
+ expect(screen.getByTestId('page-number-10')).toBeTruthy();
+ });
+
+ it('should render correct page numbers for first page', () => {
+ render();
+ expect(screen.getByTestId('page-number-1')).toBeTruthy();
+ expect(screen.getByTestId('page-number-10')).toBeTruthy();
+ expect(screen.queryByTestId('page-number-5')).not.toBeTruthy();
+ });
+
+ it('should render correct page numbers for last page', () => {
+ render();
+ expect(screen.getByTestId('page-number-1')).toBeTruthy();
+ expect(screen.getByTestId('page-number-10')).toBeTruthy();
+ });
+
+ it('should call handlePageChange with correct page number when a page number is clicked', () => {
+ render();
+ const pageButton = screen.getByTestId('page-number-1');
+ fireEvent.click(pageButton);
+ expect(mockHandlePageChange).toHaveBeenCalledWith(1);
+ });
+
+ it('should disable first and previous buttons on first page', () => {
+ render();
+
+ expect(screen.getByTestId('page-change-first')).toHaveProperty('disabled', true);
+ expect(screen.getByTestId('page-change-previous')).toHaveProperty('disabled', true);
+ });
+
+ it('should disable next and last buttons on last page', () => {
+ render();
+
+ expect(screen.getByTestId('page-change-next')).toHaveProperty('disabled', true);
+ expect(screen.getByTestId('page-change-last')).toHaveProperty('disabled', true);
+ });
+
+ it('should call handlePageChange with correct page number when navigation buttons are clicked', () => {
+ render();
+ fireEvent.click(screen.getByTestId('page-change-first'));
+ expect(mockHandlePageChange).toHaveBeenCalledWith(1);
+
+ fireEvent.click(screen.getByTestId('page-change-previous'));
+ expect(mockHandlePageChange).toHaveBeenCalledWith(4);
+
+ fireEvent.click(screen.getByTestId('page-change-next'));
+ expect(mockHandlePageChange).toHaveBeenCalledWith(6);
+
+ fireEvent.click(screen.getByTestId('page-change-last'));
+ expect(mockHandlePageChange).toHaveBeenCalledWith(10);
+ });
+
+ it('should render ellipsis correctly', () => {
+ render();
+ const ellipses = screen.getAllByText('...');
+ expect(ellipses).toHaveLength(2);
+ });
+});
diff --git a/ui/src/components/EnhancedPagination/index.ts b/ui/src/components/EnhancedPagination/index.ts
new file mode 100644
index 00000000..eaef04eb
--- /dev/null
+++ b/ui/src/components/EnhancedPagination/index.ts
@@ -0,0 +1 @@
+export { default } from './Pagination';
diff --git a/ui/src/components/ModelTable/ModelTable.tsx b/ui/src/components/ModelTable/ModelTable.tsx
index a8984ff5..bc3920d1 100644
--- a/ui/src/components/ModelTable/ModelTable.tsx
+++ b/ui/src/components/ModelTable/ModelTable.tsx
@@ -1,10 +1,11 @@
import GenerateTable from '@/components/Table/Table';
-import { getAllModels, APIData } from '@/services/models';
+import { getAllModels, GetAllModelsResponse } from '@/services/models';
import { addIconDataToArray, ConvertToTableData } from '@/utils';
import NoModels from '@/views/Models/NoModels';
import Loader from '@/components/Loader';
import useQueryWrapper from '@/hooks/useQueryWrapper';
import { useStore } from '@/stores';
+import { ApiResponse } from '@/services/common';
type ModelTableProps = {
handleOnRowClick: (args: unknown) => void;
@@ -13,7 +14,7 @@ type ModelTableProps = {
const ModelTable = ({ handleOnRowClick }: ModelTableProps): JSX.Element => {
const activeWorkspaceId = useStore((state) => state.workspaceId);
- const { data } = useQueryWrapper(
+ const { data } = useQueryWrapper, Error>(
['models', activeWorkspaceId],
() => getAllModels({ type: 'data' }),
{
diff --git a/ui/src/services/common.ts b/ui/src/services/common.ts
index 68465e0d..b646148b 100644
--- a/ui/src/services/common.ts
+++ b/ui/src/services/common.ts
@@ -2,11 +2,11 @@ export * from '@/services/axios';
export type APIRequestMethod = 'get' | 'post' | 'put' | 'delete';
-type LinksType = {
+export type LinksType = {
first: string;
last: string;
- next: string;
- prev: string;
+ next: string | null;
+ prev: string | null;
self: string;
};
diff --git a/ui/src/views/Activate/Syncs/SyncRecords/SyncRecords.tsx b/ui/src/views/Activate/Syncs/SyncRecords/SyncRecords.tsx
index 5a228751..f87a9a97 100644
--- a/ui/src/views/Activate/Syncs/SyncRecords/SyncRecords.tsx
+++ b/ui/src/views/Activate/Syncs/SyncRecords/SyncRecords.tsx
@@ -7,7 +7,6 @@ import { getSyncRecords } from '@/services/syncs';
import Loader from '@/components/Loader';
import ContentContainer from '@/components/ContentContainer';
-import Pagination from '@/components/Pagination';
import SyncRunEmptyImage from '@/assets/images/empty-state-illustration.svg';
import { SyncRecordsTopBar } from './SyncRecordsTopBar';
@@ -21,6 +20,7 @@ import { SyncRecordResponse } from '@/views/Activate/Syncs/types';
import DataTable from '@/components/DataTable';
import { SyncRecordsColumns, useDynamicSyncColumns } from './SyncRecordsColumns';
+import Pagination from '@/components/EnhancedPagination';
const SyncRecords = (): JSX.Element => {
const [searchParams, setSearchParams] = useSearchParams();
@@ -81,18 +81,6 @@ const SyncRecords = (): JSX.Element => {
}
}, [isFilteredSyncRecordsError, toast]);
- const handleNextPage = () => {
- if (filteredSyncRunRecords?.links?.next) {
- setCurrentPage((prevPage) => prevPage + 1);
- }
- };
-
- const handlePrevPage = () => {
- if (filteredSyncRunRecords?.links?.prev) {
- setCurrentPage((prevPage) => Math.max(prevPage - 1, 1));
- }
- };
-
const handleStatusTabChange = (status: SyncRecordStatus) => {
setCurrentPage(1);
setCurrentStatusTab(status);
@@ -143,14 +131,16 @@ const SyncRecords = (): JSX.Element => {
-
-
+
+ {filteredSyncRunRecords.links ? (
+
+ ) : (
+ <>Pagination unavailable.>
+ )}
)}