import * as React from 'react'; import { _ } from '@joplin/lib/locale'; import { themeStyle } from '@joplin/lib/theme'; import time from '@joplin/lib/time'; const Datetime = require('react-datetime').default; import CreatableSelect from 'react-select/creatable'; import Select from 'react-select'; import makeAnimated from 'react-select/animated'; import { focus } from '@joplin/lib/utils/focusHandler'; interface Props { themeId: number; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied defaultValue: any; visible: boolean; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied style: any; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied buttons: any[]; // eslint-disable-next-line @typescript-eslint/ban-types -- Old code before rule was applied onClose: Function; inputType: string; description: string; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied answer?: any; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied autocomplete: any; label: string; } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied export default class PromptDialog extends React.Component { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied private answerInput_: any; private focusInput_: boolean; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied private styles_: any; private styleKey_: string; private menuIsOpened_ = false; public constructor(props: Props) { super(props); this.answerInput_ = React.createRef(); this.select_menuOpen = this.select_menuOpen.bind(this); this.select_menuClose = this.select_menuClose.bind(this); } public UNSAFE_componentWillMount() { this.setState({ visible: false, answer: this.props.defaultValue ? this.props.defaultValue : '', }); this.focusInput_ = true; this.menuIsOpened_ = false; } public UNSAFE_componentWillReceiveProps(newProps: Props) { if ('visible' in newProps && newProps.visible !== this.props.visible) { this.setState({ visible: newProps.visible }); if (newProps.visible) this.focusInput_ = true; } if ('defaultValue' in newProps && newProps.defaultValue !== this.props.defaultValue) { this.setState({ answer: newProps.defaultValue }); } } private select_menuOpen() { this.menuIsOpened_ = true; } private select_menuClose() { this.menuIsOpened_ = false; } public componentDidUpdate() { if (this.focusInput_ && this.answerInput_.current) focus('PromptDialog::componentDidUpdate', this.answerInput_.current); this.focusInput_ = false; } public styles(themeId: number, width: number, height: number, visible: boolean) { const styleKey = `${themeId}_${width}_${height}_${visible}`; if (styleKey === this.styleKey_) return this.styles_; const theme = themeStyle(themeId); this.styleKey_ = styleKey; this.styles_ = {}; const paddingTop = 20; this.styles_.modalLayer = { zIndex: 9999, position: 'absolute', top: 0, left: 0, width: width, height: height, boxSizing: 'border-box', backgroundColor: 'rgba(0,0,0,0.6)', display: visible ? 'flex' : 'none', alignItems: 'flex-start', justifyContent: 'center', paddingTop: `${paddingTop}px`, }; this.styles_.promptDialog = { backgroundColor: theme.backgroundColor, padding: 16, display: 'inline-block', maxWidth: width * 0.5, boxShadow: '6px 6px 20px rgba(0,0,0,0.5)', }; this.styles_.button = { minWidth: theme.buttonMinWidth, minHeight: theme.buttonMinHeight, marginLeft: 5, color: theme.color, backgroundColor: theme.backgroundColor, border: '1px solid', borderColor: theme.dividerColor, }; this.styles_.label = { marginRight: 5, fontSize: theme.fontSize, color: theme.color, fontFamily: theme.fontFamily, verticalAlign: 'middle', }; this.styles_.input = { width: 0.5 * width, maxWidth: 400, color: theme.color, backgroundColor: theme.backgroundColor, border: '1px solid', borderColor: theme.dividerColor, }; this.styles_.select = { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied control: (provided: any) => { return { ...provided, minWidth: width * 0.2, maxWidth: width * 0.5, fontFamily: theme.fontFamily, }; }, // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied input: (provided: any) => { return { ...provided, minWidth: '20px', color: theme.color, }; }, // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied menu: (provided: any) => { return { ...provided, color: theme.color, fontFamily: theme.fontFamily, backgroundColor: theme.backgroundColor, }; }, // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied option: (provided: any, state: any) => { return { ...provided, color: theme.color, fontFamily: theme.fontFamily, paddingLeft: `${10 + (state.data.indentDepth || 0) * 20}px`, }; }, // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied multiValueLabel: (provided: any) => { return { ...provided, fontFamily: theme.fontFamily, }; }, // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied multiValueRemove: (provided: any) => { return { ...provided, color: theme.color, }; }, }; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied this.styles_.selectTheme = (tagTheme: any) => { return { ...tagTheme, borderRadius: 2, colors: { ...tagTheme.colors, primary: theme.raisedBackgroundColor, primary25: theme.raisedBackgroundColor, neutral0: theme.backgroundColor, neutral5: theme.backgroundColor, neutral10: theme.raisedBackgroundColor, neutral20: theme.raisedBackgroundColor, neutral30: theme.raisedBackgroundColor, neutral40: theme.color, neutral50: theme.color, neutral60: theme.color, neutral70: theme.color, neutral80: theme.color, neutral90: theme.color, danger: theme.backgroundColor, dangerLight: theme.colorError2, }, }; }; this.styles_.desc = { ...theme.textStyle, marginTop: 10 }; return this.styles_; } public render() { const style = this.props.style; const theme = themeStyle(this.props.themeId); const buttonTypes = this.props.buttons ? this.props.buttons : ['ok', 'cancel']; const styles = this.styles(this.props.themeId, style.width, style.height, this.state.visible); const onClose = (accept: boolean, buttonType: string = null) => { if (this.props.onClose) { let outputAnswer = this.state.answer; if (this.props.inputType === 'datetime') { // outputAnswer = anythingToDate(outputAnswer); outputAnswer = time.anythingToDateTime(outputAnswer); } this.props.onClose(accept ? outputAnswer : null, buttonType); } this.setState({ visible: false, answer: '' }); }; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied const onChange = (event: any) => { this.setState({ answer: event.target.value }); }; // const anythingToDate = (o) => { // if (o && o.toDate) return o.toDate(); // if (!o) return null; // let m = moment(o, time.dateTimeFormat()); // if (m.isValid()) return m.toDate(); // m = moment(o, time.dateFormat()); // return m.isValid() ? m.toDate() : null; // } // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied const onDateTimeChange = (momentObject: any) => { this.setState({ answer: momentObject }); }; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied const onSelectChange = (newValue: any) => { this.setState({ answer: newValue }); this.focusInput_ = true; }; // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied const onKeyDown = (event: any) => { if (event.key === 'Enter') { // If the dropdown is open, we don't close the dialog - instead // the currently item will be selected. If it is closed however // we confirm the dialog. if ((this.props.inputType === 'tags' || this.props.inputType === 'dropdown') && this.menuIsOpened_) { // Do nothing } else { onClose(true); } } else if (event.key === 'Escape') { onClose(false); } }; const descComp = this.props.description ?
{this.props.description}
: null; let inputComp = null; if (this.props.inputType === 'datetime') { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied inputComp = onDateTimeChange(momentObject)} />; } else if (this.props.inputType === 'tags') { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied inputComp = onKeyDown(event)} />; } else if (this.props.inputType === 'dropdown') { // eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied inputComp = onChange(event)} onKeyDown={event => onKeyDown(event)} />; } const buttonComps = []; if (buttonTypes.indexOf('create') >= 0) { buttonComps.push( , ); } if (buttonTypes.indexOf('ok') >= 0) { buttonComps.push( , ); } if (buttonTypes.indexOf('cancel') >= 0) { buttonComps.push( , ); } if (buttonTypes.indexOf('clear') >= 0) { buttonComps.push( , ); } return (
{inputComp} {descComp}
{buttonComps}
); } }