import React, { MutableRefObject } from "react";
import styled from "styled-components";
import {
Alignment,
Intent,
NumericInput,
IconName,
InputGroup,
Classes,
ControlGroup,
TextArea,
Tag,
IRef,
} from "@blueprintjs/core";
import _, { isNil } from "lodash";
import { ComponentProps } from "widgets/BaseComponent";
import { TextSize, TEXT_SIZES } from "constants/WidgetConstants";
import { Colors } from "constants/Colors";
import {
createMessage,
INPUT_WIDGET_DEFAULT_VALIDATION_ERROR,
} from "@appsmith/constants/messages";
import { InputTypes } from "../constants";
// TODO(abhinav): All of the following imports should not be in widgets.
import ErrorTooltip from "components/editorComponents/ErrorTooltip";
import Icon from "components/ads/Icon";
import { InputType } from "widgets/InputWidget/constants";
import { LabelPosition } from "components/constants";
import LabelWithTooltip, {
labelLayoutStyles,
LABEL_CONTAINER_CLASS,
} from "components/ads/LabelWithTooltip";
/**
* All design system component specific logic goes here.
* Ex. Blueprint has a separate numeric input and text input so switching between them goes here
* Ex. To set the icon as currency, blue print takes in a set of defined types
* All generic logic like max characters for phone numbers should be 10, should go in the widget
*/
const InputComponentWrapper = styled((props) => (
))<{
numeric: boolean;
multiline: string;
hasError: boolean;
allowCurrencyChange?: boolean;
disabled?: boolean;
inputType: InputType;
compactMode: boolean;
labelPosition: LabelPosition;
}>`
${labelLayoutStyles}
&&&& {
${({ inputType, labelPosition }) => {
if (!labelPosition && inputType !== InputTypes.TEXT) {
return "flex-direction: row";
}
}};
& .${LABEL_CONTAINER_CLASS} {
flex-grow: 0;
${({ inputType, labelPosition }) => {
if (!labelPosition && inputType !== InputTypes.TEXT) {
return "flex: 1; margin-right: 5px; label { margin-right: 5px; margin-bottom: 0;}";
}
}}
align-items: flex-start;
${({ compactMode, labelPosition }) => {
if (!labelPosition && !compactMode) {
return "max-height: 20px; .bp3-popover-wrapper {max-height: 20px}";
}
}};
}
.currency-type-filter,
.country-type-filter {
width: fit-content;
height: 36px;
display: inline-block;
left: 0;
z-index: 16;
&:hover {
border: 1px solid ${Colors.GREY_5} !important;
}
}
.${Classes.INPUT} {
min-height: 36px;
${(props) =>
props.inputType === InputTypes.CURRENCY &&
props.allowCurrencyChange &&
`
padding-left: 45px;`};
${(props) =>
props.inputType === InputTypes.CURRENCY &&
!props.allowCurrencyChange &&
`
padding-left: 35px;`};
${(props) =>
props.inputType === InputTypes.PHONE_NUMBER &&
`padding-left: 85px;
`};
box-shadow: none;
border: 1px solid;
border-radius: 0;
height: ${(props) => (props.multiline === "true" ? "100%" : "inherit")};
width: 100%;
border-color: ${({ hasError }) => {
return hasError
? `${Colors.DANGER_SOLID} !important;`
: `${Colors.GREY_3};`;
}}
${(props) =>
props.numeric &&
`
border-top-right-radius: 0px;
border-bottom-right-radius: 0px;
${props.hasError ? "" : "border-right-width: 0px;"}
`}
${(props) =>
props.inputType === "PASSWORD" &&
`
& + .bp3-input-action {
height: 36px;
width: 36px;
cursor: pointer;
padding: 1px;
.password-input {
color: ${Colors.GREY_6};
justify-content: center;
height: 100%;
svg {
width: 20px;
height: 20px;
}
&:hover {
background-color: ${Colors.GREY_2};
color: ${Colors.GREY_10};
}
}
}
`}
transition: border-color .15s ease-in-out,box-shadow .15s ease-in-out;
&:active {
border-color: ${({ hasError }) =>
hasError ? Colors.DANGER_SOLID : Colors.HIT_GRAY};
}
&:hover {
border-left: 1px solid ${Colors.GREY_5};
border-right: 1px solid ${Colors.GREY_5};
border-color: ${Colors.GREY_5};
}
&:focus {
border-color: ${({ hasError }) =>
hasError ? Colors.DANGER_SOLID : Colors.MYSTIC};
&:focus {
outline: 0;
border: 1px solid ${Colors.GREEN_1};
box-shadow: 0px 0px 0px 2px ${Colors.GREEN_2} !important;
}
}
&:disabled {
background-color: ${Colors.GREY_1};
border: 1.2px solid ${Colors.GREY_3};
& + .bp3-input-action {
pointer-events: none;
}
}
}
.${Classes.INPUT_GROUP} {
display: block;
margin: 0;
.bp3-tag {
background-color: transparent;
color: #5c7080;
margin-top: 8px;
}
&.${Classes.DISABLED} + .bp3-button-group.bp3-vertical {
pointer-events: none;
button {
background: ${Colors.GREY_1};
}
}
}
.${Classes.CONTROL_GROUP} {
justify-content: flex-start;
}
height: 100%;
align-items: ${({ compactMode, inputType, labelPosition }) => {
if (!labelPosition && inputType !== InputTypes.TEXT) {
return "center";
}
if (labelPosition === LabelPosition.Top) {
return "flex-start";
}
if (compactMode) {
return "center";
}
if (labelPosition === LabelPosition.Left) {
if (inputType === InputTypes.TEXT) {
return "stretch";
}
return "center";
}
return "flex-start";
}};
}
&&&& .bp3-input-group {
display: flex;
> {
&.bp3-icon:first-child {
top: 3px;
}
input:not(:first-child) {
line-height: 16px;
&:hover:not(:focus) {
border-left: 1px solid ${Colors.GREY_5};
}
}
}
${(props) => {
if (props.inputType === InputTypes.PHONE_NUMBER) {
return `
> {
input:not(:first-child) {
padding-left: 10px;
}
.currency-type-filter,
.currency-type-trigger,
.country-type-filter,
.country-type-trigger {
position: static;
background: rgb(255, 255, 255);
border-width: 1.2px 0px 1.2px 1.2px;
border-top-style: solid;
border-bottom-style: solid;
border-left-style: solid;
border-top-color: rgb(235, 235, 235);
border-bottom-color: rgb(235, 235, 235);
border-left-color: rgb(235, 235, 235);
border-image: initial;
color: rgb(9, 7, 7);
border-right-style: initial;
border-right-color: initial;
}
}
`;
}
}}
}
`;
const StyledNumericInput = styled(NumericInput)`
&&&& .bp3-button-group.bp3-vertical {
border: 1.2px solid ${Colors.GREY_3};
border-left: none;
button {
background: ${Colors.WHITE};
box-shadow: none;
min-width: 24px;
width: 24px;
border-radius: 0;
&:hover {
background: ${Colors.GREY_2};
span {
color: ${Colors.GREY_10};
}
}
&:focus {
border: 1px solid ${Colors.GREEN_1};
box-shadow: 0px 0px 0px 2px ${Colors.GREEN_2};
}
span {
color: ${Colors.GREY_6};
svg {
width: 14px;
}
}
}
}
`;
const TextInputWrapper = styled.div<{
inputHtmlType?: InputHTMLType;
compact: boolean;
labelPosition?: LabelPosition;
}>`
width: 100%;
display: flex;
flex: 1;
min-height: 36px;
${({ inputHtmlType }) =>
inputHtmlType && inputHtmlType !== InputTypes.TEXT && `&&& {flex-grow: 0;}`}
`;
export type InputHTMLType = "TEXT" | "NUMBER" | "PASSWORD" | "EMAIL" | "TEL";
export const isNumberInputType = (inputHTMLType: InputHTMLType = "TEXT") => {
return inputHTMLType === "NUMBER";
};
class BaseInputComponent extends React.Component<
BaseInputComponentProps,
InputComponentState
> {
constructor(props: BaseInputComponentProps) {
super(props);
this.state = { showPassword: false };
}
componentDidMount() {
if (isNumberInputType(this.props.inputHTMLType) && this.props.onStep) {
const element = document.querySelector(
`.appsmith_widget_${this.props.widgetId} .bp3-button-group`,
);
if (element !== null && element.childNodes) {
element.childNodes[0].addEventListener(
"mousedown",
this.onStepIncrement,
);
element.childNodes[1].addEventListener(
"mousedown",
this.onStepDecrement,
);
}
}
}
componentWillUnmount() {
if (isNumberInputType(this.props.inputHTMLType) && this.props.onStep) {
const element = document.querySelector(
`.appsmith_widget_${this.props.widgetId} .bp3-button-group`,
);
if (element !== null && element.childNodes) {
element.childNodes[0].removeEventListener(
"click",
this.onStepIncrement,
);
element.childNodes[1].removeEventListener(
"click",
this.onStepDecrement,
);
}
}
}
setFocusState = (isFocused: boolean) => {
this.props.onFocusChange(isFocused);
};
onTextChange = (
event:
| React.ChangeEvent
| React.ChangeEvent,
) => {
this.props.onValueChange(event.target.value);
};
onNumberChange = (valueAsNum: number, valueAsString: string) => {
this.props.onValueChange(valueAsString);
};
getLeftIcon = () => {
if (this.props.iconName && this.props.iconAlign === "left") {
return this.props.iconName;
}
return this.props.leftIcon;
};
getType(inputType: InputHTMLType = "TEXT") {
switch (inputType) {
case "PASSWORD":
return this.state.showPassword ? "text" : "password";
case "TEL":
return "tel";
case "EMAIL":
return "email";
default:
return "text";
}
}
onKeyDownTextArea = (e: React.KeyboardEvent) => {
const isEnterKey = e.key === "Enter" || e.keyCode === 13;
const { disableNewLineOnPressEnterKey } = this.props;
if (isEnterKey && disableNewLineOnPressEnterKey && !e.shiftKey) {
e.preventDefault();
}
if (typeof this.props.onKeyDown === "function") {
this.props.onKeyDown(e);
}
};
onKeyDown = (e: React.KeyboardEvent) => {
if (typeof this.props.onKeyDown === "function") {
this.props.onKeyDown(e);
}
};
private numericInputComponent = () => {
const leftIcon = this.getLeftIcon();
const conditionalProps: Record = {};
if (!isNil(this.props.maxNum)) {
conditionalProps.max = this.props.maxNum;
}
if (!isNil(this.props.minNum)) {
conditionalProps.min = this.props.minNum;
}
return (
{
if (this.props.inputRef && el) {
this.props.inputRef.current = el;
}
}}
intent={this.props.intent}
leftIcon={leftIcon}
majorStepSize={null}
minorStepSize={null}
onBlur={() => this.setFocusState(false)}
onFocus={() => this.setFocusState(true)}
onKeyDown={this.onKeyDown}
onValueChange={this.onNumberChange}
placeholder={this.props.placeholder}
stepSize={this.props.stepSize}
value={this.props.value}
{...conditionalProps}
/>
);
};
private textAreaInputComponent = () => (