# ISSUE - Closes #6734 - Closes #6633 - Closes #6733 - Closes #6816 # Description - [x] Don't allow Empty (whitespaces) Objects to Create, all the keyboard shortcuts are also handled for this. https://github.com/user-attachments/assets/1c9add4e-f13f-458b-8f76-63bd868413a2 https://github.com/user-attachments/assets/e72b6ee3-74e4-4517-a230-3eb10db80dc7 Note: we do have one other issue with FullName field #6740 Inorder to test use **shift**. - [x] Api Keys Input Name Field -> New and Detail View Name field shouldn't be empty or string with whitespaces, we won't able to have whitespaces in both. **Try Entering just spaces** https://github.com/user-attachments/assets/b521b49f-648c-4585-9d15-8ff4faed3c3a - [x] Similar to above, Empty webhook endpoint url under **/settings/developers/webhooks/new** won't be created. **Try Entering just spaces** - [x] New Functions or Updating Functions will not able to have whitespaces, empty string as Name. **Try Entering just spaces** https://github.com/user-attachments/assets/09fcf394-c6d9-4080-8efd-462b054a22d0 - [x] under **settings/workspace-members** changes will lead and solve that user won't be able to enter Invite by email as just whitespaces + button is now getting disabled when there is no correct email. **Try Entering just spaces** https://github.com/user-attachments/assets/b352edfa-113b-4645-80fd-db6f120ab5db - [x] Text Input Field, will not allow to start entering with whitespaces and won't take just whitespaces as value spaces between words will work. https://github.com/user-attachments/assets/8c1a0812-45be-4ed2-bd3d-bb4f92147976 - [x] Similarly Number works as per above including shortcuts. https://github.com/user-attachments/assets/9f69cc87-5c3c-43ee-93c4-fa887bc0d7ee - [x] Similarly FullName field works as per above including shortcuts https://github.com/user-attachments/assets/7bb006b2-abf7-44cd-a214-7a2fc68df169 - [x] Pasting fullName is been Improved. - Case 1 (Two Words): If there are exactly two words, return them as is. - Case 2 (More than Two Words): If there are more than two words, return the first two words only. - Case 3 (One Word): If there is only one word, return it as the first name, with an empty string as the last name. - WhiteSpaces have been handled. ``` console.log(splitFullName("John Doe")); // ["John", "Doe"] console.log(splitFullName(" ")); // ["", ""] console.log(splitFullName("John")); // ["John", ""] console.log(splitFullName(" John Doe ")); // ["John", "Doe"] console.log(splitFullName("John Michael Andrew Doe")); // ["John", "Michael"] ``` --------- Co-authored-by: Lucas Bordeau <bordeau.lucas@gmail.com>
224 lines
5.9 KiB
TypeScript
224 lines
5.9 KiB
TypeScript
import styled from '@emotion/styled';
|
|
import {
|
|
ChangeEvent,
|
|
ClipboardEvent,
|
|
useEffect,
|
|
useRef,
|
|
useState,
|
|
} from 'react';
|
|
import { Key } from 'ts-key-enum';
|
|
|
|
import { FieldDoubleText } from '@/object-record/record-field/types/FieldDoubleText';
|
|
import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys';
|
|
import { useListenClickOutside } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside';
|
|
import { isDefined } from '~/utils/isDefined';
|
|
|
|
import { splitFullName } from '~/utils/format/spiltFullName';
|
|
import { turnIntoEmptyStringIfWhitespacesOnly } from '~/utils/string/turnIntoEmptyStringIfWhitespacesOnly';
|
|
import { StyledTextInput } from './TextInput';
|
|
|
|
const StyledContainer = styled.div`
|
|
align-items: center;
|
|
display: flex;
|
|
justify-content: space-between;
|
|
|
|
input {
|
|
width: 100%;
|
|
}
|
|
|
|
& > input:last-child {
|
|
border-left: 1px solid ${({ theme }) => theme.border.color.medium};
|
|
padding-left: ${({ theme }) => theme.spacing(2)};
|
|
}
|
|
`;
|
|
|
|
type DoubleTextInputProps = {
|
|
firstValue: string;
|
|
secondValue: string;
|
|
firstValuePlaceholder: string;
|
|
secondValuePlaceholder: string;
|
|
hotkeyScope: string;
|
|
onEnter: (newDoubleTextValue: FieldDoubleText) => void;
|
|
onEscape: (newDoubleTextValue: FieldDoubleText) => void;
|
|
onTab?: (newDoubleTextValue: FieldDoubleText) => void;
|
|
onShiftTab?: (newDoubleTextValue: FieldDoubleText) => void;
|
|
onClickOutside: (
|
|
event: MouseEvent | TouchEvent,
|
|
newDoubleTextValue: FieldDoubleText,
|
|
) => void;
|
|
onChange?: (newDoubleTextValue: FieldDoubleText) => void;
|
|
onPaste?: (newDoubleTextValue: FieldDoubleText) => void;
|
|
};
|
|
|
|
export const DoubleTextInput = ({
|
|
firstValue,
|
|
secondValue,
|
|
firstValuePlaceholder,
|
|
secondValuePlaceholder,
|
|
hotkeyScope,
|
|
onClickOutside,
|
|
onEnter,
|
|
onEscape,
|
|
onShiftTab,
|
|
onTab,
|
|
onChange,
|
|
onPaste,
|
|
}: DoubleTextInputProps) => {
|
|
const [firstInternalValue, setFirstInternalValue] = useState(firstValue);
|
|
const [secondInternalValue, setSecondInternalValue] = useState(secondValue);
|
|
|
|
const firstValueInputRef = useRef<HTMLInputElement>(null);
|
|
const secondValueInputRef = useRef<HTMLInputElement>(null);
|
|
const containerRef = useRef<HTMLInputElement>(null);
|
|
|
|
useEffect(() => {
|
|
setFirstInternalValue(firstValue);
|
|
setSecondInternalValue(secondValue);
|
|
}, [firstValue, secondValue]);
|
|
|
|
const handleChange = (
|
|
newFirstValue: string,
|
|
newSecondValue: string,
|
|
): void => {
|
|
setFirstInternalValue(newFirstValue);
|
|
setSecondInternalValue(newSecondValue);
|
|
|
|
onChange?.({
|
|
firstValue: newFirstValue,
|
|
secondValue: newSecondValue,
|
|
});
|
|
};
|
|
|
|
const [focusPosition, setFocusPosition] = useState<'left' | 'right'>('left');
|
|
|
|
useScopedHotkeys(
|
|
Key.Enter,
|
|
() => {
|
|
onEnter({
|
|
firstValue: firstInternalValue,
|
|
secondValue: secondInternalValue,
|
|
});
|
|
},
|
|
hotkeyScope,
|
|
[onEnter, firstInternalValue, secondInternalValue],
|
|
);
|
|
|
|
useScopedHotkeys(
|
|
[Key.Escape],
|
|
() => {
|
|
onEscape({
|
|
firstValue: firstInternalValue,
|
|
secondValue: secondInternalValue,
|
|
});
|
|
},
|
|
hotkeyScope,
|
|
[onEscape, firstInternalValue, secondInternalValue],
|
|
);
|
|
|
|
useScopedHotkeys(
|
|
'tab',
|
|
() => {
|
|
if (focusPosition === 'left') {
|
|
setFocusPosition('right');
|
|
secondValueInputRef.current?.focus();
|
|
} else {
|
|
onTab?.({
|
|
firstValue: firstInternalValue,
|
|
secondValue: secondInternalValue,
|
|
});
|
|
}
|
|
},
|
|
hotkeyScope,
|
|
[onTab, firstInternalValue, secondInternalValue, focusPosition],
|
|
);
|
|
|
|
useScopedHotkeys(
|
|
'shift+tab',
|
|
() => {
|
|
if (focusPosition === 'right') {
|
|
setFocusPosition('left');
|
|
firstValueInputRef.current?.focus();
|
|
} else {
|
|
onShiftTab?.({
|
|
firstValue: firstInternalValue,
|
|
secondValue: secondInternalValue,
|
|
});
|
|
}
|
|
},
|
|
hotkeyScope,
|
|
[onShiftTab, firstInternalValue, secondInternalValue, focusPosition],
|
|
);
|
|
|
|
useListenClickOutside({
|
|
refs: [containerRef],
|
|
callback: (event) => {
|
|
onClickOutside?.(event, {
|
|
firstValue: firstInternalValue,
|
|
secondValue: secondInternalValue,
|
|
});
|
|
},
|
|
enabled: isDefined(onClickOutside),
|
|
});
|
|
|
|
const handleOnPaste = (event: ClipboardEvent<HTMLInputElement>) => {
|
|
if (firstInternalValue.length > 0 || secondInternalValue.length > 0) {
|
|
return;
|
|
}
|
|
|
|
event.preventDefault();
|
|
|
|
const name = event.clipboardData.getData('Text');
|
|
|
|
const splittedName = splitFullName(name);
|
|
|
|
onPaste?.({
|
|
firstValue: splittedName[0],
|
|
secondValue: splittedName[1],
|
|
});
|
|
};
|
|
|
|
const handleClickToPreventParentClickEvents = (
|
|
event: React.MouseEvent<HTMLInputElement>,
|
|
) => {
|
|
event.stopPropagation();
|
|
event.preventDefault();
|
|
};
|
|
|
|
return (
|
|
<StyledContainer ref={containerRef}>
|
|
<StyledTextInput
|
|
autoComplete="off"
|
|
autoFocus
|
|
onFocus={() => setFocusPosition('left')}
|
|
ref={firstValueInputRef}
|
|
placeholder={firstValuePlaceholder}
|
|
value={firstInternalValue}
|
|
onChange={(event: ChangeEvent<HTMLInputElement>) => {
|
|
handleChange(
|
|
turnIntoEmptyStringIfWhitespacesOnly(event.target.value),
|
|
secondInternalValue,
|
|
);
|
|
}}
|
|
onPaste={(event: ClipboardEvent<HTMLInputElement>) =>
|
|
handleOnPaste(event)
|
|
}
|
|
onClick={handleClickToPreventParentClickEvents}
|
|
/>
|
|
<StyledTextInput
|
|
autoComplete="off"
|
|
onFocus={() => setFocusPosition('right')}
|
|
ref={secondValueInputRef}
|
|
placeholder={secondValuePlaceholder}
|
|
value={secondInternalValue}
|
|
onChange={(event: ChangeEvent<HTMLInputElement>) => {
|
|
handleChange(
|
|
firstInternalValue,
|
|
turnIntoEmptyStringIfWhitespacesOnly(event.target.value),
|
|
);
|
|
}}
|
|
onClick={handleClickToPreventParentClickEvents}
|
|
/>
|
|
</StyledContainer>
|
|
);
|
|
};
|