PromucFlow_constructor/app/client/src/widgets/BaseWidget.tsx
Anand Srinivasan 186a97c699
fix: trigger fields linting issue (#11875)
* initial fix

* fix warnings

* log evaluation order length

* log time

* logs

* re-order

* rename

* onclick vs release logs

* unit test for loading properties

* clean up

* ends with .tableData

* add comment
2022-03-31 17:07:18 +05:30

544 lines
15 KiB
TypeScript

/**
* Widget are responsible for accepting the abstraction layer inputs, interpretting them into rederable props and
* spawing components based on those props
* Widgets are also responsible for dispatching actions and updating the state tree
*/
import {
CONTAINER_GRID_PADDING,
CSSUnit,
CSSUnits,
PositionType,
PositionTypes,
RenderMode,
RenderModes,
WidgetType,
} from "constants/WidgetConstants";
import React, { Component, ReactNode } from "react";
import { get, memoize } from "lodash";
import DraggableComponent from "components/editorComponents/DraggableComponent";
import SnipeableComponent from "components/editorComponents/SnipeableComponent";
import ResizableComponent from "components/editorComponents/ResizableComponent";
import { ExecuteTriggerPayload } from "constants/AppsmithActionConstants/ActionConstants";
import PositionedContainer from "components/designSystems/appsmith/PositionedContainer";
import WidgetNameComponent from "components/editorComponents/WidgetNameComponent";
import shallowequal from "shallowequal";
import { EditorContext } from "components/editorComponents/EditorContextProvider";
import ErrorBoundary from "components/editorComponents/ErrorBoundry";
import { DerivedPropertiesMap } from "utils/WidgetFactory";
import {
DataTreeEvaluationProps,
EVAL_ERROR_PATH,
EvaluationError,
PropertyEvaluationErrorType,
WidgetDynamicPathListProps,
} from "utils/DynamicBindingUtils";
import { PropertyPaneConfig } from "constants/PropertyControlConstants";
import { BatchPropertyUpdatePayload } from "actions/controlActions";
import OverlayCommentsWrapper from "comments/inlineComments/OverlayCommentsWrapper";
import PreventInteractionsOverlay from "components/editorComponents/PreventInteractionsOverlay";
import AppsmithConsole from "utils/AppsmithConsole";
import { ENTITY_TYPE } from "entities/AppsmithConsole";
import PreviewModeComponent from "components/editorComponents/PreviewModeComponent";
/***
* BaseWidget
*
* The abstract class which is extended/implemented by all widgets.
* Widgets must adhere to the abstractions provided by BaseWidget.
*
* Do not:
* 1) Use the context directly in the widgets
* 2) Update or access the dsl in the widgets
* 3) Call actions in widgets or connect the widgets to the entity reducers
*
*/
abstract class BaseWidget<
T extends WidgetProps,
K extends WidgetState
> extends Component<T, K> {
static contextType = EditorContext;
static getPropertyPaneConfig(): PropertyPaneConfig[] {
return [];
}
static getDerivedPropertiesMap(): DerivedPropertiesMap {
return {};
}
static getDefaultPropertiesMap(): Record<string, any> {
return {};
}
// TODO Find a way to enforce this, (dont let it be set)
static getMetaPropertiesMap(): Record<string, any> {
return {};
}
/**
* getLoadingProperties returns a list of regexp's used to specify bindingPaths,
* which can set the isLoading prop of the widget.
* When:
* 1. the path is bound to an action (API/Query)
* 2. the action is currently in-progress
*
* if undefined, all paths can set the isLoading state
* if empty array, no paths can set the isLoading state
*/
static getLoadingProperties(): Array<RegExp> | undefined {
return;
}
/**
* Widget abstraction to register the widget type
* ```javascript
* getWidgetType() {
* return "MY_AWESOME_WIDGET",
* }
* ```
*/
/**
* Widgets can execute actions using this `executeAction` method.
* Triggers may be specific to the widget
*/
executeAction(actionPayload: ExecuteTriggerPayload): void {
const { executeAction } = this.context;
executeAction &&
executeAction({
...actionPayload,
source: {
id: this.props.widgetId,
name: this.props.widgetName,
},
});
actionPayload.triggerPropertyName &&
AppsmithConsole.info({
text: `${actionPayload.triggerPropertyName} triggered`,
source: {
type: ENTITY_TYPE.WIDGET,
id: this.props.widgetId,
name: this.props.widgetName,
},
});
}
disableDrag(disable: boolean) {
const { disableDrag } = this.context;
disableDrag && disable !== undefined && disableDrag(disable);
}
updateWidget(
operationName: string,
widgetId: string,
widgetProperties: any,
): void {
const { updateWidget } = this.context;
updateWidget && updateWidget(operationName, widgetId, widgetProperties);
}
deleteWidgetProperty(propertyPaths: string[]): void {
const { deleteWidgetProperty } = this.context;
const { widgetId } = this.props;
if (deleteWidgetProperty && widgetId) {
deleteWidgetProperty(widgetId, propertyPaths);
}
}
batchUpdateWidgetProperty(
updates: BatchPropertyUpdatePayload,
shouldReplay = true,
): void {
const { batchUpdateWidgetProperty } = this.context;
const { widgetId } = this.props;
if (batchUpdateWidgetProperty && widgetId) {
batchUpdateWidgetProperty(widgetId, updates, shouldReplay);
}
}
updateWidgetProperty(propertyName: string, propertyValue: any): void {
this.batchUpdateWidgetProperty({
modify: { [propertyName]: propertyValue },
});
}
resetChildrenMetaProperty(widgetId: string) {
const { resetChildrenMetaProperty } = this.context;
resetChildrenMetaProperty(widgetId);
}
/* eslint-disable @typescript-eslint/no-empty-function */
/* eslint-disable @typescript-eslint/no-unused-vars */
componentDidUpdate(prevProps: T) {}
componentDidMount(): void {}
/* eslint-enable @typescript-eslint/no-empty-function */
getComponentDimensions = () => {
return this.calculateWidgetBounds(
this.props.rightColumn,
this.props.leftColumn,
this.props.topRow,
this.props.bottomRow,
this.props.parentColumnSpace,
this.props.parentRowSpace,
);
};
calculateWidgetBounds(
rightColumn: number,
leftColumn: number,
topRow: number,
bottomRow: number,
parentColumnSpace: number,
parentRowSpace: number,
): {
componentWidth: number;
componentHeight: number;
} {
return {
componentWidth: (rightColumn - leftColumn) * parentColumnSpace,
componentHeight: (bottomRow - topRow) * parentRowSpace,
};
}
getErrorCount = memoize((evalErrors: Record<string, EvaluationError[]>) => {
return Object.values(evalErrors).reduce(
(prev, curr) =>
curr.filter(
(error) => error.errorType !== PropertyEvaluationErrorType.LINT,
).length + prev,
0,
);
}, JSON.stringify);
render() {
return this.getWidgetView();
}
/**
* this function is responsive for making the widget resizable.
* A widget can be made by non-resizable by passing resizeDisabled prop.
*
* @param content
*/
makeResizable(content: ReactNode) {
return (
<ResizableComponent
{...this.props}
paddingOffset={PositionedContainer.padding}
>
{content}
</ResizableComponent>
);
}
/**
* this functions wraps the widget in a component that shows a setting control at the top right
* which gets shown on hover. A widget can enable/disable this by setting `disablePropertyPane` prop
*
* @param content
* @param showControls
*/
showWidgetName(content: ReactNode, showControls = false) {
return (
<>
{!this.props.disablePropertyPane && (
<WidgetNameComponent
errorCount={this.getErrorCount(
get(this.props, EVAL_ERROR_PATH, {}),
)}
parentId={this.props.parentId}
showControls={showControls}
topRow={this.props.detachFromLayout ? 4 : this.props.topRow}
type={this.props.type}
widgetId={this.props.widgetId}
widgetName={this.props.widgetName}
/>
)}
{content}
</>
);
}
/**
* wraps the widget in a draggable component.
* Note: widget drag can be disabled by setting `dragDisabled` prop to true
*
* @param content
*/
makeDraggable(content: ReactNode) {
return <DraggableComponent {...this.props}>{content}</DraggableComponent>;
}
/**
* wraps the widget in a draggable component.
* Note: widget drag can be disabled by setting `dragDisabled` prop to true
*
* @param content
*/
makeSnipeable(content: ReactNode) {
return <SnipeableComponent {...this.props}>{content}</SnipeableComponent>;
}
makePositioned(content: ReactNode) {
const style = this.getPositionStyle();
return (
<PositionedContainer
focused={this.props.focused}
parentId={this.props.parentId}
resizeDisabled={this.props.resizeDisabled}
selected={this.props.selected}
style={style}
widgetId={this.props.widgetId}
widgetType={this.props.type}
>
{content}
</PositionedContainer>
);
}
addErrorBoundary(content: ReactNode) {
return <ErrorBoundary>{content}</ErrorBoundary>;
}
/**
* These comments are rendered using position: absolute over the widget borders,
* they are not aware of the component structure.
* For additional component specific contexts, for eg.
* a comment bound to the scroll position or a specific section
* we would pass comments as props to the components
*/
addOverlayComments(content: ReactNode) {
return (
<OverlayCommentsWrapper
refId={this.props.widgetId}
widgetType={this.props.type}
>
{content}
</OverlayCommentsWrapper>
);
}
addPreventInteractionOverlay(content: ReactNode) {
return (
<PreventInteractionsOverlay widgetType={this.props.type}>
{content}
</PreventInteractionsOverlay>
);
}
addPreviewModeWidget(content: ReactNode): React.ReactElement {
return (
<PreviewModeComponent isVisible={this.props.isVisible}>
{content}
</PreviewModeComponent>
);
}
private getWidgetView(): ReactNode {
let content: ReactNode;
switch (this.props.renderMode) {
case RenderModes.CANVAS:
content = this.getCanvasView();
content = this.addPreviewModeWidget(content);
content = this.addPreventInteractionOverlay(content);
content = this.addOverlayComments(content);
if (!this.props.detachFromLayout) {
if (!this.props.resizeDisabled) content = this.makeResizable(content);
content = this.showWidgetName(content);
content = this.makeDraggable(content);
content = this.makeSnipeable(content);
// NOTE: In sniping mode we are not blocking onClick events from PositionWrapper.
content = this.makePositioned(content);
}
return content;
// return this.getCanvasView();
case RenderModes.PAGE:
content = this.getPageView();
if (this.props.isVisible) {
content = this.addPreventInteractionOverlay(content);
content = this.addOverlayComments(content);
content = this.addErrorBoundary(content);
if (!this.props.detachFromLayout) {
content = this.makePositioned(content);
}
return content;
}
return null;
default:
throw Error("RenderMode not defined");
}
}
abstract getPageView(): ReactNode;
getCanvasView(): ReactNode {
const content = this.getPageView();
return this.addErrorBoundary(content);
}
// TODO(abhinav): Maybe make this a pure component to bailout from updating altogether.
// This would involve making all widgets which have "states" to not have states,
// as they're extending this one.
shouldComponentUpdate(nextProps: WidgetProps, nextState: WidgetState) {
return (
!shallowequal(nextProps, this.props) ||
!shallowequal(nextState, this.state)
);
}
/**
* generates styles that positions the widget
*/
private getPositionStyle(): BaseStyle {
const { componentHeight, componentWidth } = this.getComponentDimensions();
return {
positionType: PositionTypes.ABSOLUTE,
componentHeight,
componentWidth,
yPosition:
this.props.topRow * this.props.parentRowSpace +
(this.props.noContainerOffset ? 0 : CONTAINER_GRID_PADDING),
xPosition:
this.props.leftColumn * this.props.parentColumnSpace +
(this.props.noContainerOffset ? 0 : CONTAINER_GRID_PADDING),
xPositionUnit: CSSUnits.PIXEL,
yPositionUnit: CSSUnits.PIXEL,
};
}
// TODO(abhinav): These defaultProps seem unneccessary. Check it out.
static defaultProps: Partial<WidgetProps> | undefined = {
parentRowSpace: 1,
parentColumnSpace: 1,
topRow: 0,
leftColumn: 0,
isLoading: false,
renderMode: RenderModes.CANVAS,
dragDisabled: false,
dropDisabled: false,
isDeletable: true,
resizeDisabled: false,
disablePropertyPane: false,
};
}
export interface BaseStyle {
componentHeight: number;
componentWidth: number;
positionType: PositionType;
xPosition: number;
yPosition: number;
xPositionUnit: CSSUnit;
yPositionUnit: CSSUnit;
heightUnit?: CSSUnit;
widthUnit?: CSSUnit;
}
export type WidgetState = Record<string, unknown>;
export interface WidgetBuilder<T extends WidgetProps, S extends WidgetState> {
buildWidget(widgetProps: T): JSX.Element;
}
export interface WidgetBaseProps {
widgetId: string;
type: WidgetType;
widgetName: string;
parentId?: string;
renderMode: RenderMode;
version: number;
}
export type WidgetRowCols = {
leftColumn: number;
rightColumn: number;
topRow: number;
bottomRow: number;
minHeight?: number; // Required to reduce the size of CanvasWidgets.
};
export interface WidgetPositionProps extends WidgetRowCols {
parentColumnSpace: number;
parentRowSpace: number;
// The detachFromLayout flag tells use about the following properties when enabled
// 1) Widget does not drag/resize
// 2) Widget CAN (but not neccessarily) be a dropTarget
// Examples: MainContainer is detached from layout,
// MODAL_WIDGET is also detached from layout.
detachFromLayout?: boolean;
noContainerOffset?: boolean; // This won't offset the child in parent
}
export const WIDGET_STATIC_PROPS = {
leftColumn: true,
rightColumn: true,
topRow: true,
bottomRow: true,
minHeight: true,
parentColumnSpace: true,
parentRowSpace: true,
children: true,
type: true,
widgetId: true,
widgetName: true,
parentId: true,
renderMode: true,
detachFromLayout: true,
noContainerOffset: false,
};
export const WIDGET_DISPLAY_PROPS = {
isVisible: true,
isLoading: true,
isDisabled: true,
backgroundColor: true,
};
export interface WidgetDisplayProps {
//TODO(abhinav): Some of these props are mandatory
isVisible?: boolean;
isLoading: boolean;
isDisabled?: boolean;
backgroundColor?: string;
animateLoading?: boolean;
}
export interface WidgetDataProps
extends WidgetBaseProps,
WidgetPositionProps,
WidgetDisplayProps {}
export interface WidgetProps
extends WidgetDataProps,
WidgetDynamicPathListProps,
DataTreeEvaluationProps {
key?: string;
isDefaultClickDisabled?: boolean;
[key: string]: any;
}
export interface WidgetCardProps {
type: WidgetType;
key?: string;
displayName: string;
icon: string;
isBeta?: boolean;
}
export const WidgetOperations = {
MOVE: "MOVE",
RESIZE: "RESIZE",
ADD_CHILD: "ADD_CHILD",
UPDATE_PROPERTY: "UPDATE_PROPERTY",
DELETE: "DELETE",
ADD_CHILDREN: "ADD_CHILDREN",
};
export type WidgetOperation = typeof WidgetOperations[keyof typeof WidgetOperations];
export default BaseWidget;