PromucFlow_constructor/app/client/src/widgets/TableWidget/component/index.tsx

329 lines
10 KiB
TypeScript
Raw Normal View History

import React, { useEffect, useMemo } from "react";
import Table from "./Table";
import {
ColumnTypes,
CompactMode,
ReactTableColumnProps,
ReactTableFilter,
} from "./Constants";
import { Row } from "react-table";
import { EventType } from "constants/AppsmithActionConstants/ActionConstants";
import { isEqual } from "lodash";
2020-06-03 10:50:10 +00:00
export interface ColumnMenuOptionProps {
content: string | JSX.Element;
closeOnClick?: boolean;
isSelected?: boolean;
editColumnName?: boolean;
2020-06-03 10:50:10 +00:00
columnAccessor?: string;
id?: string;
category?: boolean;
options?: ColumnMenuSubOptionProps[];
onClick?: (columnIndex: number, isSelected: boolean) => void;
2020-06-03 10:50:10 +00:00
}
export interface ColumnMenuSubOptionProps {
content: string | JSX.Element;
isSelected?: boolean;
closeOnClick?: boolean;
onClick?: (columnIndex: number) => void;
id?: string;
category?: boolean;
isHeader?: boolean;
2020-06-03 10:50:10 +00:00
}
interface ReactTableComponentProps {
widgetId: string;
widgetName: string;
searchKey: string;
2020-06-03 10:50:10 +00:00
isDisabled?: boolean;
isVisible?: boolean;
isLoading: boolean;
editMode: boolean;
2020-06-03 10:50:10 +00:00
width: number;
height: number;
pageSize: number;
totalRecordsCount?: number;
tableData: Array<Record<string, unknown>>;
2020-06-03 10:50:10 +00:00
disableDrag: (disable: boolean) => void;
onRowClick: (rowData: Record<string, unknown>, rowIndex: number) => void;
onCommandClick: (dynamicTrigger: string, onComplete: () => void) => void;
selectAllRow: (pageData: Row<Record<string, unknown>>[]) => void;
unSelectAllRow: (pageData: Row<Record<string, unknown>>[]) => void;
updatePageNo: (pageNo: number, event?: EventType) => void;
sortTableColumn: (column: string, asc: boolean) => void;
nextPageClick: () => void;
prevPageClick: () => void;
2020-06-03 10:50:10 +00:00
pageNo: number;
serverSidePaginationEnabled: boolean;
selectedRowIndex: number;
selectedRowIndices: number[];
multiRowSelection?: boolean;
2020-06-03 10:50:10 +00:00
hiddenColumns?: string[];
triggerRowSelection: boolean;
2020-06-03 10:50:10 +00:00
columnSizeMap?: { [key: string]: number };
handleResizeColumn: (columnSizeMap: { [key: string]: number }) => void;
handleReorderColumn: (columnOrder: string[]) => void;
searchTableData: (searchKey: any) => void;
filters?: ReactTableFilter[];
applyFilter: (filters: ReactTableFilter[]) => void;
columns: ReactTableColumnProps[];
compactMode?: CompactMode;
isVisibleSearch?: boolean;
isVisibleFilters?: boolean;
isVisibleDownload?: boolean;
isVisiblePagination?: boolean;
delimiter: string;
2020-06-03 10:50:10 +00:00
}
function ReactTableComponent(props: ReactTableComponentProps) {
2021-05-03 18:42:21 +00:00
const {
applyFilter,
columns,
columnSizeMap,
compactMode,
delimiter,
2021-05-03 18:42:21 +00:00
disableDrag,
editMode,
filters,
handleReorderColumn,
handleResizeColumn,
height,
isLoading,
isVisibleDownload,
isVisibleFilters,
isVisiblePagination,
isVisibleSearch,
multiRowSelection,
2021-05-03 18:42:21 +00:00
nextPageClick,
onRowClick,
pageNo,
pageSize,
prevPageClick,
searchKey,
searchTableData,
selectAllRow,
2021-05-03 18:42:21 +00:00
selectedRowIndex,
selectedRowIndices,
serverSidePaginationEnabled,
sortTableColumn: _sortTableColumn,
tableData,
totalRecordsCount,
2021-05-03 18:42:21 +00:00
triggerRowSelection,
unSelectAllRow,
2021-05-03 18:42:21 +00:00
updatePageNo,
widgetId,
widgetName,
width,
} = props;
const { columnOrder, hiddenColumns } = useMemo(() => {
const order: string[] = [];
const hidden: string[] = [];
2021-05-03 18:42:21 +00:00
columns.forEach((item) => {
if (item.isHidden) {
hidden.push(item.accessor);
} else {
order.push(item.accessor);
}
});
return { columnOrder: order, hiddenColumns: hidden };
2021-05-03 18:42:21 +00:00
}, [columns]);
useEffect(() => {
let dragged = -1;
const headers = Array.prototype.slice.call(
2021-05-03 18:42:21 +00:00
document.querySelectorAll(`#table${widgetId} .draggable-header`),
);
headers.forEach((header, i) => {
header.setAttribute("draggable", true);
header.ondragstart = (e: React.DragEvent<HTMLDivElement>) => {
header.style =
"background: #efefef; border-radius: 4px; z-index: 100; width: 100%; text-overflow: none; overflow: none;";
e.stopPropagation();
dragged = i;
};
header.ondrag = (e: React.DragEvent<HTMLDivElement>) => {
e.stopPropagation();
};
header.ondragend = (e: React.DragEvent<HTMLDivElement>) => {
header.style = "";
e.stopPropagation();
setTimeout(() => (dragged = -1), 1000);
};
// the dropped header
header.ondragover = (e: React.DragEvent<HTMLDivElement>) => {
if (i !== dragged && dragged !== -1) {
if (dragged > i) {
header.parentElement.className = "th header-reorder highlight-left";
} else if (dragged < i) {
header.parentElement.className =
"th header-reorder highlight-right";
}
2020-06-03 10:50:10 +00:00
}
e.preventDefault();
};
header.ondragenter = (e: React.DragEvent<HTMLDivElement>) => {
if (i !== dragged && dragged !== -1) {
if (dragged > i) {
header.parentElement.className = "th header-reorder highlight-left";
} else if (dragged < i) {
header.parentElement.className =
"th header-reorder highlight-right";
}
2020-06-03 10:50:10 +00:00
}
e.preventDefault();
};
2020-06-03 10:50:10 +00:00
header.ondragleave = (e: React.DragEvent<HTMLDivElement>) => {
header.parentElement.className = "th header-reorder";
e.preventDefault();
};
header.ondrop = (e: React.DragEvent<HTMLDivElement>) => {
header.style = "";
header.parentElement.className = "th header-reorder";
if (i !== dragged && dragged !== -1) {
e.preventDefault();
const newColumnOrder = [...columnOrder];
// The dragged column
const movedColumnName = newColumnOrder.splice(dragged, 1);
// If the dragged column exists
if (movedColumnName && movedColumnName.length === 1) {
newColumnOrder.splice(i, 0, movedColumnName[0]);
}
2021-05-03 18:42:21 +00:00
handleReorderColumn([...newColumnOrder, ...hiddenColumns]);
} else {
dragged = -1;
}
};
});
});
const sortTableColumn = (columnIndex: number, asc: boolean) => {
if (columnIndex === -1) {
2021-05-03 18:42:21 +00:00
_sortTableColumn("", asc);
} else {
2021-05-03 18:42:21 +00:00
const column = columns[columnIndex];
const columnType = column.metaProperties?.type || ColumnTypes.TEXT;
if (
columnType !== ColumnTypes.IMAGE &&
columnType !== ColumnTypes.VIDEO
) {
2021-05-03 18:42:21 +00:00
_sortTableColumn(column.accessor, asc);
}
}
};
const selectTableRow = (row: {
original: Record<string, unknown>;
index: number;
}) => {
2021-05-03 18:42:21 +00:00
onRowClick(row.original, row.index);
2020-06-03 10:50:10 +00:00
};
const toggleAllRowSelect = (
isSelect: boolean,
pageData: Row<Record<string, unknown>>[],
) => {
if (isSelect) {
selectAllRow(pageData);
} else {
unSelectAllRow(pageData);
}
};
return (
<Table
2021-05-03 18:42:21 +00:00
applyFilter={applyFilter}
columnSizeMap={columnSizeMap}
columns={columns}
compactMode={compactMode}
data={tableData}
delimiter={delimiter}
disableDrag={() => {
2021-05-03 18:42:21 +00:00
disableDrag(true);
}}
2021-05-03 18:42:21 +00:00
editMode={editMode}
enableDrag={() => {
2021-05-03 18:42:21 +00:00
disableDrag(false);
}}
2021-05-03 18:42:21 +00:00
filters={filters}
handleResizeColumn={handleResizeColumn}
height={height}
isLoading={isLoading}
isVisibleDownload={isVisibleDownload}
isVisibleFilters={isVisibleFilters}
isVisiblePagination={isVisiblePagination}
isVisibleSearch={isVisibleSearch}
multiRowSelection={multiRowSelection}
nextPageClick={nextPageClick}
2021-05-03 18:42:21 +00:00
pageNo={pageNo - 1}
pageSize={pageSize || 1}
prevPageClick={prevPageClick}
2021-05-03 18:42:21 +00:00
searchKey={searchKey}
searchTableData={searchTableData}
selectTableRow={selectTableRow}
2021-05-03 18:42:21 +00:00
selectedRowIndex={selectedRowIndex}
selectedRowIndices={selectedRowIndices}
serverSidePaginationEnabled={serverSidePaginationEnabled}
sortTableColumn={sortTableColumn}
toggleAllRowSelect={toggleAllRowSelect}
totalRecordsCount={totalRecordsCount}
2021-05-03 18:42:21 +00:00
triggerRowSelection={triggerRowSelection}
updatePageNo={updatePageNo}
widgetId={widgetId}
widgetName={widgetName}
width={width}
/>
);
}
2020-06-03 10:50:10 +00:00
2021-05-04 06:51:08 +00:00
export default React.memo(ReactTableComponent, (prev, next) => {
return (
prev.applyFilter === next.applyFilter &&
prev.compactMode === next.compactMode &&
prev.delimiter === next.delimiter &&
2021-05-04 06:51:08 +00:00
prev.disableDrag === next.disableDrag &&
prev.editMode === next.editMode &&
prev.filters === next.filters &&
prev.handleReorderColumn === next.handleReorderColumn &&
prev.handleResizeColumn === next.handleResizeColumn &&
prev.height === next.height &&
prev.isLoading === next.isLoading &&
prev.isVisibleDownload === next.isVisibleDownload &&
prev.isVisibleFilters === next.isVisibleFilters &&
prev.isVisiblePagination === next.isVisiblePagination &&
prev.isVisibleSearch === next.isVisibleSearch &&
2021-05-04 06:51:08 +00:00
prev.nextPageClick === next.nextPageClick &&
prev.onRowClick === next.onRowClick &&
prev.pageNo === next.pageNo &&
prev.pageSize === next.pageSize &&
prev.prevPageClick === next.prevPageClick &&
prev.searchKey === next.searchKey &&
prev.searchTableData === next.searchTableData &&
prev.selectedRowIndex === next.selectedRowIndex &&
prev.selectedRowIndices === next.selectedRowIndices &&
prev.serverSidePaginationEnabled === next.serverSidePaginationEnabled &&
prev.sortTableColumn === next.sortTableColumn &&
prev.totalRecordsCount === next.totalRecordsCount &&
2021-05-04 06:51:08 +00:00
prev.triggerRowSelection === next.triggerRowSelection &&
prev.updatePageNo === next.updatePageNo &&
prev.widgetId === next.widgetId &&
prev.widgetName === next.widgetName &&
prev.width === next.width &&
isEqual(prev.columnSizeMap, next.columnSizeMap) &&
isEqual(prev.tableData, next.tableData) &&
// Using JSON stringify becuase isEqual doesnt work with functions,
// and we are not changing the columns manually.
JSON.stringify(prev.columns) === JSON.stringify(next.columns)
2021-05-04 06:51:08 +00:00
);
});