/*
* Copyright 2021-2024 Avaiga Private Limited
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
* an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*/
import React from "react";
import { render } from "@testing-library/react";
import "@testing-library/jest-dom";
import userEvent from "@testing-library/user-event";
import AutoLoadingTable from "./AutoLoadingTable";
import { TaipyContext } from "../../context/taipyContext";
import { TaipyState, INITIAL_STATE } from "../../context/taipyReducers";
import { TableValueType } from "./tableUtils";
jest.mock(
"react-virtualized-auto-sizer",
() =>
({ children }: any) =>
children({ height: 600, width: 600 })
);
const valueKey = "Infinite-Entity-asc";
const tableValue = {
[valueKey]: {
data: [
{
Entity: "Austria",
Day_str: "2020-04-01T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 856,
},
{
Entity: "Austria",
Day_str: "2020-04-02T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 823,
},
{
Entity: "Austria",
Day_str: "2020-04-03T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 829,
},
{
Entity: "Austria",
Day_str: "2020-04-04T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 826,
},
{
Entity: "Austria",
Day_str: "2020-04-05T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 712,
},
{
Entity: "Austria",
Day_str: "2020-04-06T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 824,
},
{
Entity: "Austria",
Day_str: "2020-04-07T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 857,
},
{
Entity: "Austria",
Day_str: "2020-04-08T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 829,
},
{
Entity: "Austria",
Day_str: "2020-04-09T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 820,
},
{
Entity: "Austria",
Day_str: "2020-04-10T00:00:00.000000Z",
Code: "AUT",
"Daily hospital occupancy": 771,
},
],
rowcount: 14477,
start: 0,
},
};
const tableColumns = JSON.stringify({ Entity: { dfid: "Entity" } });
const tableWidthColumns = JSON.stringify({ Entity: { dfid: "Entity", width: "100px" }, Country: {dfid: "Country"} });
describe("AutoLoadingTable Component", () => {
it("renders", async () => {
const { getByText } = render();
const elt = getByText("Entity");
expect(elt.tagName).toBe("DIV");
});
it("displays the right info for class", async () => {
const { getByText } = render(
);
const elt = getByText("Entity").closest("table");
expect(elt?.parentElement?.parentElement?.parentElement).toHaveClass("taipy-table", "taipy-table-autoloading");
});
it("is disabled", async () => {
const { getByText } = render();
const elt = getByText("Entity");
expect(elt.parentElement).toHaveClass("Mui-disabled");
});
it("is enabled by default", async () => {
const { getByText } = render();
const elt = getByText("Entity");
expect(elt.parentElement).not.toHaveClass("Mui-disabled");
});
it("is enabled by active", async () => {
const { getByText, getAllByTestId } = render();
const elt = getByText("Entity");
expect(elt.parentElement).not.toHaveClass("Mui-disabled");
expect(getAllByTestId("ArrowDownwardIcon").length).toBeGreaterThan(0);
});
it("hides sort icons when not active", async () => {
const { queryByTestId } = render();
expect(queryByTestId("ArrowDownwardIcon")).toBeNull();
});
it("hides sort icons when not sortable", async () => {
const { queryByTestId } = render();
expect(queryByTestId("ArrowDownwardIcon")).toBeNull();
});
it("set width if requested", async () => {
const { getByText } = render();
const header = getByText("Entity").closest("tr");
expect(header?.firstChild).toHaveStyle({"min-width": "100px"});
expect(header?.lastChild).toHaveStyle({"width": "100%"});
});
// keep getting undefined Error from jest, it seems to be linked to the setTimeout that makes the code run after the end of the test :-(
// https://github.com/facebook/jest/issues/12262
// Looks like the right way to handle this is to use jest fakeTimers and runAllTimers ...
xit("dispatch a well formed message on sort", async () => {
jest.useFakeTimers();
const dispatch = jest.fn();
const state: TaipyState = INITIAL_STATE;
const { getByText } = render(
);
const elt = getByText("Entity");
await userEvent.click(elt);
jest.runAllTimers();
expect(dispatch).toHaveBeenCalledWith({
name: "",
payload: {
columns: ["Entity"],
end: 99,
id: undefined,
infinite: true,
orderby: "Entity",
pagekey: "Infinite-Entity-Entity-asc",
handlenan: false,
sort: "asc",
start: 0,
aggregates: [],
applies: undefined,
cellClassNames: undefined,
tooltips: undefined,
filters: []
},
type: "REQUEST_DATA_UPDATE",
});
jest.useRealTimers();
});
it("dispatch a well formed message at first render", async () => {
const dispatch = jest.fn();
const state: TaipyState = INITIAL_STATE;
render(
);
expect(dispatch).toHaveBeenCalledWith({
name: "",
payload: { id: "table", names: ["varname"], refresh: false },
type: "REQUEST_UPDATE",
});
});
it("displays the received data", async () => {
const dispatch = jest.fn();
const state: TaipyState = { ...INITIAL_STATE, data: { table: undefined } };
const { getAllByText, rerender } = render(
);
const newState = { ...state, data: { ...state.data, table: tableValue } };
rerender(
);
const elements = getAllByText("Austria");
expect(elements.length).toBeGreaterThan(1);
expect(elements[0].tagName).toBe("SPAN");
});
it("selects the rows", async () => {
const dispatch = jest.fn();
const state: TaipyState = INITIAL_STATE;
const selected = [2, 4, 6];
const { getAllByText, rerender } = render(
);
rerender(
);
const elements = getAllByText("Austria");
elements.forEach((elt: HTMLElement, idx: number) =>
selected.indexOf(idx) == -1
? expect(elt.parentElement?.parentElement?.parentElement?.parentElement).not.toHaveClass("Mui-selected")
: expect(elt.parentElement?.parentElement?.parentElement?.parentElement).toHaveClass("Mui-selected")
);
expect(document.querySelectorAll(".Mui-selected")).toHaveLength(selected.length);
});
});