PromucFlow_constructor/app/client/src/components/propertyControls/PrimaryColumnsControl.tsx

336 lines
9.1 KiB
TypeScript
Raw Normal View History

import React, { useCallback, useState } from "react";
import BaseControl, { ControlProps } from "./BaseControl";
import {
StyledInputGroup,
StyledDragIcon,
StyledEditIcon,
StyledDeleteIcon,
StyledVisibleIcon,
StyledHiddenIcon,
StyledPropertyPaneButton,
} from "./StyledControls";
import styled from "constants/DefaultTheme";
import { DroppableComponent } from "components/ads/DraggableListComponent";
import { ColumnProperties } from "components/designSystems/appsmith/TableComponent/Constants";
import EmptyDataState from "components/utils/EmptyDataState";
import { getNextEntityName } from "utils/AppsmithUtils";
import {
getDefaultColumnProperties,
getTableStyles,
} from "components/designSystems/appsmith/TableComponent/TableUtilities";
import { debounce } from "lodash";
import { Size, Category } from "components/ads/Button";
import { reorderColumns } from "components/designSystems/appsmith/TableComponent/TableHelpers";
const ItemWrapper = styled.div`
display: flex;
justify-content: flex-start;
align-items: center;
`;
const TabsWrapper = styled.div`
width: 100%;
display: flex;
flex-direction: column;
`;
const StyledOptionControlInputGroup = styled(StyledInputGroup)`
margin-right: 2px;
margin-bottom: 2px;
width: 100%;
padding-left: 30px;
&&& {
input {
padding-left: 24px;
border: none;
color: ${(props) => props.theme.colors.textOnDarkBG};
background: ${(props) => props.theme.colors.paneInputBG};
&:focus {
border: none;
color: ${(props) => props.theme.colors.textOnDarkBG};
background: ${(props) => props.theme.colors.paneInputBG};
}
}
}
`;
const AddColumnButton = styled(StyledPropertyPaneButton)`
width: 100%;
display: flex;
justify-content: center;
&&&& {
margin-top: 12px;
margin-bottom: 8px;
}
`;
type RenderComponentProps = {
index: number;
item: {
label: string;
isDerived?: boolean;
isVisible?: boolean;
};
updateOption: (index: number, value: string) => void;
onEdit?: (index: number) => void;
deleteOption: (index: number) => void;
toggleVisibility?: (index: number) => void;
};
const getOriginalColumn = (
columns: Record<string, ColumnProperties>,
index: number,
columnOrder?: string[],
): ColumnProperties | undefined => {
const reorderedColumns = reorderColumns(columns, columnOrder || []);
const column: ColumnProperties | undefined = Object.values(
reorderedColumns,
).find((column: ColumnProperties) => column.index === index);
return column;
};
function ColumnControlComponent(props: RenderComponentProps) {
const [value, setValue] = useState(props.item.label);
const {
updateOption,
onEdit,
item,
deleteOption,
toggleVisibility,
index,
} = props;
const [visibility, setVisibility] = useState(item.isVisible);
const debouncedUpdate = debounce(updateOption, 1000);
const onChange = useCallback(
(index: number, value: string) => {
setValue(value);
debouncedUpdate(index, value);
},
[updateOption],
);
return (
<ItemWrapper>
<StyledDragIcon height={20} width={20} />
<StyledOptionControlInputGroup
dataType="text"
placeholder="Column Title"
onChange={(value: string) => {
onChange(index, value);
}}
defaultValue={value}
/>
<StyledEditIcon
className="t--edit-column-btn"
height={20}
width={20}
onClick={() => {
onEdit && onEdit(index);
}}
/>
{!!item.isDerived ? (
<StyledDeleteIcon
className="t--delete-column-btn"
height={20}
width={20}
onClick={() => {
deleteOption && deleteOption(index);
}}
/>
) : visibility ? (
<StyledVisibleIcon
className="t--show-column-btn"
height={20}
width={20}
onClick={() => {
setVisibility(!visibility);
toggleVisibility && toggleVisibility(index);
}}
/>
) : (
<StyledHiddenIcon
className="t--show-column-btn"
height={20}
width={20}
onClick={() => {
setVisibility(!visibility);
toggleVisibility && toggleVisibility(index);
}}
/>
)}
</ItemWrapper>
);
}
class PrimaryColumnsControl extends BaseControl<ControlProps> {
render() {
// Get columns from widget properties
const columns: Record<string, ColumnProperties> =
this.props.propertyValue || {};
// If there are no columns, show empty state
if (Object.keys(columns).length === 0) {
return <EmptyDataState />;
}
// Get an empty array of length of columns
let columnOrder: string[] = new Array(Object.keys(columns).length);
if (this.props.widgetProperties.columnOrder) {
columnOrder = this.props.widgetProperties.columnOrder;
} else {
columnOrder = Object.keys(columns);
}
const reorderedColumns = reorderColumns(columns, columnOrder);
const draggableComponentColumns = Object.values(reorderedColumns).map(
(column: ColumnProperties) => {
return {
label: column.label,
id: column.id,
isVisible: column.isVisible,
isDerived: column.isDerived,
index: column.index,
};
},
);
return (
<TabsWrapper>
<DroppableComponent
items={draggableComponentColumns}
itemHeight={45}
renderComponent={ColumnControlComponent}
updateOption={this.updateOption}
updateItems={this.updateItems}
deleteOption={this.deleteOption}
toggleVisibility={this.toggleVisibility}
onEdit={this.onEdit}
/>
<AddColumnButton
className="t--add-column-btn"
icon="plus"
tag="button"
type="button"
text="Add a new column"
onClick={this.addNewColumn}
size={Size.medium}
category={Category.tertiary}
/>
</TabsWrapper>
);
}
addNewColumn = () => {
const columns: Record<string, ColumnProperties> =
this.props.propertyValue || {};
const columnIds = Object.keys(columns);
const newColumnName = getNextEntityName("customColumn", columnIds);
const nextIndex = columnIds.length;
const columnProps: ColumnProperties = getDefaultColumnProperties(
newColumnName,
nextIndex,
this.props.widgetProperties.widgetName,
true,
);
const tableStyles = getTableStyles(this.props.widgetProperties);
const column = {
...columnProps,
buttonStyle: "rgb(3, 179, 101)",
buttonLabelColor: "#FFFFFF",
...tableStyles,
};
this.updateProperty(`${this.props.propertyName}.${column.id}`, column);
};
onEdit = (index: number) => {
const columns: Record<string, ColumnProperties> =
this.props.propertyValue || [];
const originalColumn = getOriginalColumn(
columns,
index,
this.props.widgetProperties.columnOrder,
);
this.props.openNextPanel({
...originalColumn,
widgetId: this.props.widgetProperties.widgetId,
});
};
//Used to reorder columns
updateItems = (items: Array<Record<string, unknown>>) => {
this.updateProperty(
"columnOrder",
items.map(({ id }) => id),
);
};
toggleVisibility = (index: number) => {
const columns: Record<string, ColumnProperties> =
this.props.propertyValue || {};
const originalColumn = getOriginalColumn(
columns,
index,
this.props.widgetProperties.columnOrder,
);
if (originalColumn) {
this.updateProperty(
`${this.props.propertyName}.${originalColumn.id}.isVisible`,
!originalColumn.isVisible,
);
}
};
deleteOption = (index: number) => {
const columns: Record<string, ColumnProperties> =
this.props.propertyValue || {};
const derivedColumns = this.props.widgetProperties.derivedColumns || {};
const columnOrder = this.props.widgetProperties.columnOrder || [];
const originalColumn = getOriginalColumn(columns, index, columnOrder);
if (originalColumn) {
const propertiesToDelete = [
`${this.props.propertyName}.${originalColumn.id}`,
];
if (derivedColumns[originalColumn.id])
propertiesToDelete.push(`derivedColumns.${originalColumn.id}`);
const columnOrderIndex = columnOrder.findIndex(
(column: string) => column === originalColumn.id,
);
if (columnOrderIndex > -1)
propertiesToDelete.push(`columnOrder[${columnOrderIndex}]`);
this.deleteProperties(propertiesToDelete);
}
};
updateOption = (index: number, updatedLabel: string) => {
const columns: Record<string, ColumnProperties> =
this.props.propertyValue || {};
const originalColumn = getOriginalColumn(
columns,
index,
this.props.widgetProperties.columnOrder,
);
if (originalColumn) {
this.updateProperty(
`${this.props.propertyName}.${originalColumn.id}.label`,
updatedLabel,
);
}
};
static getControlType() {
return "PRIMARY_COLUMNS";
}
}
export default PrimaryColumnsControl;