diff --git a/.changeset/kind-pens-build.md b/.changeset/kind-pens-build.md
new file mode 100644
index 00000000..113fddca
--- /dev/null
+++ b/.changeset/kind-pens-build.md
@@ -0,0 +1,19 @@
+---
+"@gitcoin/ui": patch
+---
+
+### Features
+
+- Created new program picker modal component for improved program selection
+- Refactored carousel component for better performance and maintainability
+
+### Bug Fixes
+
+- Improved disabled field styling
+- Fixed conditional padding in review application page checker
+- Updated program list filter placeholder text
+
+### Chores
+
+- Enhanced IndexDB implementation
+- Added mock service worker file for testing purposes
diff --git a/packages/ui/.gitignore b/packages/ui/.gitignore
index 65fa6bba..6291b7a6 100644
--- a/packages/ui/.gitignore
+++ b/packages/ui/.gitignore
@@ -24,5 +24,4 @@ dist-ssr
*.sw?
.env
-storybook-static
-mockServiceWorker.js
\ No newline at end of file
+storybook-static
\ No newline at end of file
diff --git a/packages/ui/public/mockServiceWorker.js b/packages/ui/public/mockServiceWorker.js
new file mode 100644
index 00000000..c9c662a2
--- /dev/null
+++ b/packages/ui/public/mockServiceWorker.js
@@ -0,0 +1,295 @@
+/* eslint-disable */
+/* tslint:disable */
+
+/**
+ * Mock Service Worker.
+ * @see https://github.com/mswjs/msw
+ * - Please do NOT modify this file.
+ * - Please do NOT serve this file on production.
+ */
+
+const PACKAGE_VERSION = '2.6.6'
+const INTEGRITY_CHECKSUM = 'ca7800994cc8bfb5eb961e037c877074'
+const IS_MOCKED_RESPONSE = Symbol('isMockedResponse')
+const activeClientIds = new Set()
+
+self.addEventListener('install', function () {
+ self.skipWaiting()
+})
+
+self.addEventListener('activate', function (event) {
+ event.waitUntil(self.clients.claim())
+})
+
+self.addEventListener('message', async function (event) {
+ const clientId = event.source.id
+
+ if (!clientId || !self.clients) {
+ return
+ }
+
+ const client = await self.clients.get(clientId)
+
+ if (!client) {
+ return
+ }
+
+ const allClients = await self.clients.matchAll({
+ type: 'window',
+ })
+
+ switch (event.data) {
+ case 'KEEPALIVE_REQUEST': {
+ sendToClient(client, {
+ type: 'KEEPALIVE_RESPONSE',
+ })
+ break
+ }
+
+ case 'INTEGRITY_CHECK_REQUEST': {
+ sendToClient(client, {
+ type: 'INTEGRITY_CHECK_RESPONSE',
+ payload: {
+ packageVersion: PACKAGE_VERSION,
+ checksum: INTEGRITY_CHECKSUM,
+ },
+ })
+ break
+ }
+
+ case 'MOCK_ACTIVATE': {
+ activeClientIds.add(clientId)
+
+ sendToClient(client, {
+ type: 'MOCKING_ENABLED',
+ payload: {
+ client: {
+ id: client.id,
+ frameType: client.frameType,
+ },
+ },
+ })
+ break
+ }
+
+ case 'MOCK_DEACTIVATE': {
+ activeClientIds.delete(clientId)
+ break
+ }
+
+ case 'CLIENT_CLOSED': {
+ activeClientIds.delete(clientId)
+
+ const remainingClients = allClients.filter((client) => {
+ return client.id !== clientId
+ })
+
+ // Unregister itself when there are no more clients
+ if (remainingClients.length === 0) {
+ self.registration.unregister()
+ }
+
+ break
+ }
+ }
+})
+
+self.addEventListener('fetch', function (event) {
+ const { request } = event
+
+ // Bypass navigation requests.
+ if (request.mode === 'navigate') {
+ return
+ }
+
+ // Opening the DevTools triggers the "only-if-cached" request
+ // that cannot be handled by the worker. Bypass such requests.
+ if (request.cache === 'only-if-cached' && request.mode !== 'same-origin') {
+ return
+ }
+
+ // Bypass all requests when there are no active clients.
+ // Prevents the self-unregistered worked from handling requests
+ // after it's been deleted (still remains active until the next reload).
+ if (activeClientIds.size === 0) {
+ return
+ }
+
+ // Generate unique request ID.
+ const requestId = crypto.randomUUID()
+ event.respondWith(handleRequest(event, requestId))
+})
+
+async function handleRequest(event, requestId) {
+ const client = await resolveMainClient(event)
+ const response = await getResponse(event, client, requestId)
+
+ // Send back the response clone for the "response:*" life-cycle events.
+ // Ensure MSW is active and ready to handle the message, otherwise
+ // this message will pend indefinitely.
+ if (client && activeClientIds.has(client.id)) {
+ ;(async function () {
+ const responseClone = response.clone()
+
+ sendToClient(
+ client,
+ {
+ type: 'RESPONSE',
+ payload: {
+ requestId,
+ isMockedResponse: IS_MOCKED_RESPONSE in response,
+ type: responseClone.type,
+ status: responseClone.status,
+ statusText: responseClone.statusText,
+ body: responseClone.body,
+ headers: Object.fromEntries(responseClone.headers.entries()),
+ },
+ },
+ [responseClone.body],
+ )
+ })()
+ }
+
+ return response
+}
+
+// Resolve the main client for the given event.
+// Client that issues a request doesn't necessarily equal the client
+// that registered the worker. It's with the latter the worker should
+// communicate with during the response resolving phase.
+async function resolveMainClient(event) {
+ const client = await self.clients.get(event.clientId)
+
+ if (activeClientIds.has(event.clientId)) {
+ return client
+ }
+
+ if (client?.frameType === 'top-level') {
+ return client
+ }
+
+ const allClients = await self.clients.matchAll({
+ type: 'window',
+ })
+
+ return allClients
+ .filter((client) => {
+ // Get only those clients that are currently visible.
+ return client.visibilityState === 'visible'
+ })
+ .find((client) => {
+ // Find the client ID that's recorded in the
+ // set of clients that have registered the worker.
+ return activeClientIds.has(client.id)
+ })
+}
+
+async function getResponse(event, client, requestId) {
+ const { request } = event
+
+ // Clone the request because it might've been already used
+ // (i.e. its body has been read and sent to the client).
+ const requestClone = request.clone()
+
+ function passthrough() {
+ // Cast the request headers to a new Headers instance
+ // so the headers can be manipulated with.
+ const headers = new Headers(requestClone.headers)
+
+ // Remove the "accept" header value that marked this request as passthrough.
+ // This prevents request alteration and also keeps it compliant with the
+ // user-defined CORS policies.
+ headers.delete('accept', 'msw/passthrough')
+
+ return fetch(requestClone, { headers })
+ }
+
+ // Bypass mocking when the client is not active.
+ if (!client) {
+ return passthrough()
+ }
+
+ // Bypass initial page load requests (i.e. static assets).
+ // The absence of the immediate/parent client in the map of the active clients
+ // means that MSW hasn't dispatched the "MOCK_ACTIVATE" event yet
+ // and is not ready to handle requests.
+ if (!activeClientIds.has(client.id)) {
+ return passthrough()
+ }
+
+ // Notify the client that a request has been intercepted.
+ const requestBuffer = await request.arrayBuffer()
+ const clientMessage = await sendToClient(
+ client,
+ {
+ type: 'REQUEST',
+ payload: {
+ id: requestId,
+ url: request.url,
+ mode: request.mode,
+ method: request.method,
+ headers: Object.fromEntries(request.headers.entries()),
+ cache: request.cache,
+ credentials: request.credentials,
+ destination: request.destination,
+ integrity: request.integrity,
+ redirect: request.redirect,
+ referrer: request.referrer,
+ referrerPolicy: request.referrerPolicy,
+ body: requestBuffer,
+ keepalive: request.keepalive,
+ },
+ },
+ [requestBuffer],
+ )
+
+ switch (clientMessage.type) {
+ case 'MOCK_RESPONSE': {
+ return respondWithMock(clientMessage.data)
+ }
+
+ case 'PASSTHROUGH': {
+ return passthrough()
+ }
+ }
+
+ return passthrough()
+}
+
+function sendToClient(client, message, transferrables = []) {
+ return new Promise((resolve, reject) => {
+ const channel = new MessageChannel()
+
+ channel.port1.onmessage = (event) => {
+ if (event.data && event.data.error) {
+ return reject(event.data.error)
+ }
+
+ resolve(event.data)
+ }
+
+ client.postMessage(
+ message,
+ [channel.port2].concat(transferrables.filter(Boolean)),
+ )
+ })
+}
+
+async function respondWithMock(response) {
+ // Setting response status code to 0 is a no-op.
+ // However, when responding with a "Response.error()", the produced Response
+ // instance will have status code set to 0. Since it's not possible to create
+ // a Response instance with status code 0, handle that use-case separately.
+ if (response.status === 0) {
+ return Response.error()
+ }
+
+ const mockedResponse = new Response(response.body, response)
+
+ Reflect.defineProperty(mockedResponse, IS_MOCKED_RESPONSE, {
+ value: true,
+ enumerable: true,
+ })
+
+ return mockedResponse
+}
\ No newline at end of file
diff --git a/packages/ui/src/components/Form/FormControllers/DisabledProgramInputFormController/DisabledProgramInputFormController.tsx b/packages/ui/src/components/Form/FormControllers/DisabledProgramInputFormController/DisabledProgramInputFormController.tsx
index b3d5249e..a94c7370 100644
--- a/packages/ui/src/components/Form/FormControllers/DisabledProgramInputFormController/DisabledProgramInputFormController.tsx
+++ b/packages/ui/src/components/Form/FormControllers/DisabledProgramInputFormController/DisabledProgramInputFormController.tsx
@@ -1,6 +1,5 @@
"use client";
-import React from "react";
import { Controller, useFormContext } from "react-hook-form";
import { getChainInfo } from "@/lib";
@@ -32,7 +31,7 @@ export const renderDisabledProgramInput = (value: {
const { chainId, programName } = value;
const chainInfo = getChainInfo(chainId);
return (
-
+
diff --git a/packages/ui/src/components/GenericProgressForm/GenericProgressForm.stories.tsx b/packages/ui/src/components/GenericProgressForm/GenericProgressForm.stories.tsx
index 2afd9c43..e7ff81f3 100644
--- a/packages/ui/src/components/GenericProgressForm/GenericProgressForm.stories.tsx
+++ b/packages/ui/src/components/GenericProgressForm/GenericProgressForm.stories.tsx
@@ -22,5 +22,6 @@ export const Default: Story = {
onSubmit: async (values: any) => onSubmit(values),
dbName: "formDB",
storeName: "formDrafts",
+ stepsPersistKey: "roundSetup",
},
};
diff --git a/packages/ui/src/components/GenericProgressForm/GenericProgressForm.tsx b/packages/ui/src/components/GenericProgressForm/GenericProgressForm.tsx
index f4b660ea..469967e6 100644
--- a/packages/ui/src/components/GenericProgressForm/GenericProgressForm.tsx
+++ b/packages/ui/src/components/GenericProgressForm/GenericProgressForm.tsx
@@ -18,6 +18,7 @@ export interface GenericProgressFormProps {
onSubmit: (values: any) => Promise
;
dbName: string;
storeName: string;
+ stepsPersistKey: string;
}
export const GenericProgressForm = ({
@@ -26,9 +27,10 @@ export const GenericProgressForm = ({
onSubmit,
dbName,
storeName,
+ stepsPersistKey,
}: GenericProgressFormProps) => {
- const { currentStep, updateStep } = useFormProgress(name);
- const { getValues } = useIndexedDB({ dbName, storeName });
+ const { currentStep, updateStep } = useFormProgress(stepsPersistKey);
+ const { getValues, isReady } = useIndexedDB({ dbName, storeName });
const formRef = useRef<{ isFormValid: () => Promise }>(null);
const handleNextStep = () => {
@@ -59,6 +61,8 @@ export const GenericProgressForm = ({
const currentStepProps = steps[currentStep];
const progressValue = (currentStep / steps.length) * 100;
+ if (!isReady) return null;
+
return (
diff --git a/packages/ui/src/features/checker/pages/ReviewApplicationsPage/ReviewApplicationsPage.tsx b/packages/ui/src/features/checker/pages/ReviewApplicationsPage/ReviewApplicationsPage.tsx
index c77151d6..f5b47839 100644
--- a/packages/ui/src/features/checker/pages/ReviewApplicationsPage/ReviewApplicationsPage.tsx
+++ b/packages/ui/src/features/checker/pages/ReviewApplicationsPage/ReviewApplicationsPage.tsx
@@ -2,6 +2,7 @@
import { useMemo } from "react";
+import { cn } from "@/lib";
import { Button } from "@/primitives/Button";
import { Icon, IconType } from "@/primitives/Icon";
import { StatCardProps } from "@/primitives/StatCard";
@@ -16,7 +17,7 @@ import {
useCheckerContext,
useCheckerDispatchContext,
} from "~checker/store";
-import { getManagerUrl, getRoundLinkOnManager } from "~checker/utils";
+import { getRoundLinkOnManager } from "~checker/utils";
import { PoolSummary } from "~pool";
export const ReviewApplicationsPage = ({ isStandalone }: { isStandalone: boolean }) => {
@@ -81,7 +82,7 @@ export const ReviewApplicationsPage = ({ isStandalone }: { isStandalone: boolean
donationsEndTime={poolData?.donationsEndTime}
/>
)}
-
+
{isStandalone && (
+ />
);
},
);
CarouselPrevious.displayName = "CarouselPrevious";
-const CarouselNext = React.forwardRef
>(
- ({ className, variant = "outlined-primary", size = "icon", ...props }, ref) => {
- const { orientation, scrollNext, canScrollNext } = useCarousel();
+const CarouselNext = React.forwardRef(
+ ({ className, size = "lg", carouselIconType = "default", ...props }, ref) => {
+ const { scrollNext, canScrollNext } = useCarousel();
+ const { icon } = carouselVariants({ size });
+ const Icon = CarouselIcons[carouselIconType].right;
return (
}
{...props}
- >
-
- Next slide
-
+ />
);
},
);
diff --git a/packages/ui/src/primitives/Carousel/index.ts b/packages/ui/src/primitives/Carousel/index.ts
new file mode 100644
index 00000000..2b91cb69
--- /dev/null
+++ b/packages/ui/src/primitives/Carousel/index.ts
@@ -0,0 +1 @@
+export * from "./Carousel";
\ No newline at end of file
diff --git a/packages/ui/src/primitives/index.ts b/packages/ui/src/primitives/index.ts
index 82baa6fa..c787a08c 100644
--- a/packages/ui/src/primitives/index.ts
+++ b/packages/ui/src/primitives/index.ts
@@ -23,3 +23,4 @@ export * from "./Toast";
export * from "./VerticalTabs";
export * from "./Checkbox";
export * from "./Switch";
+export * from "./Carousel";