|
import { render, screen, within } from "@testing-library/react";
|
|
import { afterEach, describe, expect, it, test, vi } from "vitest";
|
|
import userEvent from "@testing-library/user-event";
|
|
import { formatTimeDelta } from "#/utils/format-time-delta";
|
|
import { ConversationCard } from "#/components/features/conversation-panel/conversation-card";
|
|
import { clickOnEditButton } from "./utils";
|
|
|
|
describe("ConversationCard", () => {
|
|
const onClick = vi.fn();
|
|
const onDelete = vi.fn();
|
|
const onChangeTitle = vi.fn();
|
|
const onDownloadWorkspace = vi.fn();
|
|
|
|
afterEach(() => {
|
|
vi.clearAllMocks();
|
|
});
|
|
|
|
it("should render the conversation card", () => {
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
onChangeTitle={onChangeTitle}
|
|
isActive
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
const expectedDate = `${formatTimeDelta(new Date("2021-10-01T12:00:00Z"))} ago`;
|
|
|
|
const card = screen.getByTestId("conversation-card");
|
|
const title = within(card).getByTestId("conversation-card-title");
|
|
|
|
expect(title).toHaveValue("Conversation 1");
|
|
within(card).getByText(expectedDate);
|
|
});
|
|
|
|
it("should render the selectedRepository if available", () => {
|
|
const { rerender } = render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
onChangeTitle={onChangeTitle}
|
|
isActive
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
expect(
|
|
screen.queryByTestId("conversation-card-selected-repository"),
|
|
).not.toBeInTheDocument();
|
|
|
|
rerender(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
onChangeTitle={onChangeTitle}
|
|
isActive
|
|
title="Conversation 1"
|
|
selectedRepository="org/selectedRepository"
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
screen.getByTestId("conversation-card-selected-repository");
|
|
});
|
|
|
|
it("should toggle a context menu when clicking the ellipsis button", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
onChangeTitle={onChangeTitle}
|
|
isActive
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
expect(screen.queryByTestId("context-menu")).not.toBeInTheDocument();
|
|
|
|
const ellipsisButton = screen.getByTestId("ellipsis-button");
|
|
await user.click(ellipsisButton);
|
|
|
|
screen.getByTestId("context-menu");
|
|
|
|
await user.click(ellipsisButton);
|
|
|
|
expect(screen.queryByTestId("context-menu")).not.toBeInTheDocument();
|
|
});
|
|
|
|
it("should call onDelete when the delete button is clicked", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
const ellipsisButton = screen.getByTestId("ellipsis-button");
|
|
await user.click(ellipsisButton);
|
|
|
|
const menu = screen.getByTestId("context-menu");
|
|
const deleteButton = within(menu).getByTestId("delete-button");
|
|
|
|
await user.click(deleteButton);
|
|
|
|
expect(onDelete).toHaveBeenCalled();
|
|
});
|
|
|
|
test("clicking the selectedRepository should not trigger the onClick handler", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository="org/selectedRepository"
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
const selectedRepository = screen.getByTestId(
|
|
"conversation-card-selected-repository",
|
|
);
|
|
await user.click(selectedRepository);
|
|
|
|
expect(onClick).not.toHaveBeenCalled();
|
|
});
|
|
|
|
test("conversation title should call onChangeTitle when changed and blurred", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
onChangeTitle={onChangeTitle}
|
|
/>,
|
|
);
|
|
|
|
const title = screen.getByTestId("conversation-card-title");
|
|
expect(title).toBeDisabled();
|
|
|
|
await clickOnEditButton(user);
|
|
|
|
expect(title).toBeEnabled();
|
|
expect(screen.queryByTestId("context-menu")).not.toBeInTheDocument();
|
|
|
|
expect(document.activeElement).toBe(title);
|
|
|
|
await user.clear(title);
|
|
await user.type(title, "New Conversation Name ");
|
|
await user.tab();
|
|
|
|
expect(onChangeTitle).toHaveBeenCalledWith("New Conversation Name");
|
|
expect(title).toHaveValue("New Conversation Name");
|
|
expect(title).toBeDisabled();
|
|
});
|
|
|
|
it("should reset title and not call onChangeTitle when the title is empty", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
await clickOnEditButton(user);
|
|
|
|
const title = screen.getByTestId("conversation-card-title");
|
|
|
|
await user.clear(title);
|
|
await user.tab();
|
|
|
|
expect(onChangeTitle).not.toHaveBeenCalled();
|
|
expect(title).toHaveValue("Conversation 1");
|
|
});
|
|
|
|
test("clicking the title should not trigger the onClick handler", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
const title = screen.getByTestId("conversation-card-title");
|
|
await user.click(title);
|
|
|
|
expect(onClick).not.toHaveBeenCalled();
|
|
});
|
|
|
|
test("clicking the delete button should not trigger the onClick handler", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
const ellipsisButton = screen.getByTestId("ellipsis-button");
|
|
await user.click(ellipsisButton);
|
|
|
|
const menu = screen.getByTestId("context-menu");
|
|
const deleteButton = within(menu).getByTestId("delete-button");
|
|
|
|
await user.click(deleteButton);
|
|
|
|
expect(onClick).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("should call onDownloadWorkspace when the download button is clicked", async () => {
|
|
const user = userEvent.setup();
|
|
render(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
onDelete={onDelete}
|
|
onChangeTitle={onChangeTitle}
|
|
onDownloadWorkspace={onDownloadWorkspace}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
const ellipsisButton = screen.getByTestId("ellipsis-button");
|
|
await user.click(ellipsisButton);
|
|
|
|
const menu = screen.getByTestId("context-menu");
|
|
const downloadButton = within(menu).getByTestId("download-button");
|
|
|
|
await user.click(downloadButton);
|
|
|
|
expect(onDownloadWorkspace).toHaveBeenCalled();
|
|
});
|
|
|
|
it("should not display the edit or delete options if the handler is not provided", async () => {
|
|
const user = userEvent.setup();
|
|
const { rerender } = render(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
const ellipsisButton = screen.getByTestId("ellipsis-button");
|
|
await user.click(ellipsisButton);
|
|
|
|
expect(screen.queryByTestId("edit-button")).toBeInTheDocument();
|
|
expect(screen.queryByTestId("delete-button")).not.toBeInTheDocument();
|
|
|
|
|
|
await user.click(ellipsisButton);
|
|
|
|
rerender(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
onDelete={onDelete}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
await user.click(ellipsisButton);
|
|
|
|
expect(screen.queryByTestId("edit-button")).not.toBeInTheDocument();
|
|
expect(screen.queryByTestId("delete-button")).toBeInTheDocument();
|
|
});
|
|
|
|
it("should not render the ellipsis button if there are no actions", () => {
|
|
const { rerender } = render(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
onDelete={onDelete}
|
|
onChangeTitle={onChangeTitle}
|
|
onDownloadWorkspace={onDownloadWorkspace}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
expect(screen.getByTestId("ellipsis-button")).toBeInTheDocument();
|
|
|
|
rerender(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
onDelete={onDelete}
|
|
onDownloadWorkspace={onDownloadWorkspace}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
expect(screen.getByTestId("ellipsis-button")).toBeInTheDocument();
|
|
|
|
rerender(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
onDownloadWorkspace={onDownloadWorkspace}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
expect(screen.queryByTestId("ellipsis-button")).toBeInTheDocument();
|
|
|
|
rerender(
|
|
<ConversationCard
|
|
onClick={onClick}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
expect(screen.queryByTestId("ellipsis-button")).not.toBeInTheDocument();
|
|
});
|
|
|
|
describe("state indicator", () => {
|
|
it("should render the 'STOPPED' indicator by default", () => {
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
/>,
|
|
);
|
|
|
|
screen.getByTestId("STOPPED-indicator");
|
|
});
|
|
|
|
it("should render the other indicators when provided", () => {
|
|
render(
|
|
<ConversationCard
|
|
onDelete={onDelete}
|
|
isActive
|
|
onChangeTitle={onChangeTitle}
|
|
title="Conversation 1"
|
|
selectedRepository={null}
|
|
lastUpdatedAt="2021-10-01T12:00:00Z"
|
|
status="RUNNING"
|
|
/>,
|
|
);
|
|
|
|
expect(screen.queryByTestId("STOPPED-indicator")).not.toBeInTheDocument();
|
|
screen.getByTestId("RUNNING-indicator");
|
|
});
|
|
});
|
|
});
|
|
|