Skip to content

Commit

Permalink
Revert "fix(Combobox): make value as string of integer work (#2095)"
Browse files Browse the repository at this point in the history
This reverts commit c25f51e.
  • Loading branch information
mimarz committed Jun 7, 2024
1 parent 2ede277 commit 7b8695e
Show file tree
Hide file tree
Showing 6 changed files with 29 additions and 75 deletions.
21 changes: 0 additions & 21 deletions packages/react/src/components/form/Combobox/Combobox.stories.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -580,24 +580,3 @@ export const RemoveAllOptions: StoryFn<typeof Combobox> = (args) => {
</>
);
};

export const WithNumberValues: StoryFn<typeof Combobox> = () => {
return (
<Combobox initialValue={['2000']}>
<Combobox.Option
id={'3000'}
key={'3000'}
value={'3000'}
>
some value
</Combobox.Option>
<Combobox.Option
id={'2000'}
key={'2000'}
value={'2000'}
>
some other value
</Combobox.Option>
</Combobox>
);
};
30 changes: 13 additions & 17 deletions packages/react/src/components/form/Combobox/Combobox.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import { omit } from '../../../utilities';
import { Spinner } from '../../Spinner';

import type { Option } from './useCombobox';
import useCombobox, { prefix, removePrefix } from './useCombobox';
import useCombobox from './useCombobox';
import ComboboxInput from './internal/ComboboxInput';
import ComboboxLabel from './internal/ComboboxLabel';
import ComboboxError from './internal/ComboboxError';
Expand Down Expand Up @@ -205,23 +205,23 @@ export const ComboboxComponent = forwardRef<HTMLInputElement, ComboboxProps>(
// if value is set, set input value to the label of the value
useEffect(() => {
if (value && value.length > 0 && !multiple) {
const option = options[prefix(value[0])];
const option = options[value[0]];
setInputValue(option?.label || '');
}
}, [multiple, value, options]);

useEffect(() => {
if (value && Object.keys(options).length >= 0) {
const updatedSelectedOptions = value.map((option) => {
const value = options[prefix(option)];
const value = options[option];
return value;
});

setSelectedOptions(
updatedSelectedOptions.reduce<{
[key: string]: Option;
}>((acc, value) => {
acc[prefix(value.value)] = value;
acc[value.value] = value;
return acc;
}, {}),
);
Expand All @@ -246,21 +246,19 @@ export const ComboboxComponent = forwardRef<HTMLInputElement, ComboboxProps>(

if (remove) {
const newSelectedOptions = { ...selectedOptions };
delete newSelectedOptions[prefix(option.value)];
delete newSelectedOptions[option.value];
setSelectedOptions(newSelectedOptions);
onValueChange?.(
Object.keys(newSelectedOptions).map((key) => removePrefix(key)),
);
onValueChange?.(Object.keys(newSelectedOptions));
return;
}

const newSelectedOptions = { ...selectedOptions };

if (multiple) {
if (newSelectedOptions[prefix(option.value)]) {
delete newSelectedOptions[prefix(option.value)];
if (newSelectedOptions[option.value]) {
delete newSelectedOptions[option.value];
} else {
newSelectedOptions[prefix(option.value)] = option;
newSelectedOptions[option.value] = option;
}
setInputValue('');
inputRef.current?.focus();
Expand All @@ -269,7 +267,7 @@ export const ComboboxComponent = forwardRef<HTMLInputElement, ComboboxProps>(
Object.keys(newSelectedOptions).forEach((key) => {
delete newSelectedOptions[key];
});
newSelectedOptions[prefix(option.value)] = option;
newSelectedOptions[option.value] = option;
setInputValue(option?.label || '');
// move cursor to the end of the input
setTimeout(() => {
Expand All @@ -281,9 +279,7 @@ export const ComboboxComponent = forwardRef<HTMLInputElement, ComboboxProps>(
}

setSelectedOptions(newSelectedOptions);
onValueChange?.(
Object.keys(newSelectedOptions).map((key) => removePrefix(key)),
);
onValueChange?.(Object.keys(newSelectedOptions));

!multiple && setOpen(false);
refs.domReference.current?.focus();
Expand Down Expand Up @@ -313,7 +309,7 @@ export const ComboboxComponent = forwardRef<HTMLInputElement, ComboboxProps>(
measureElement: (elem) => {
return elem.getBoundingClientRect().height;
},
overscan: 7,
overscan: 1,
});

return (
Expand Down Expand Up @@ -342,7 +338,7 @@ export const ComboboxComponent = forwardRef<HTMLInputElement, ComboboxProps>(
onOptionClick: (value: string) => {
if (readOnly) return;
if (disabled) return;
const option = options[prefix(value)];
const option = options[value];
debouncedHandleSelectOption({ option: option });
},
handleSelectOption: debouncedHandleSelectOption,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,6 @@ import { useMergeRefs } from '@floating-ui/react';
import { ComboboxContext } from '../ComboboxContext';
import useDebounce from '../../../../utilities/useDebounce';
import { useComboboxId, useComboboxIdDispatch } from '../ComboboxIdContext';
import { prefix } from '../useCombobox';

type UseComboboxOptionProps = {
id?: string;
Expand Down Expand Up @@ -35,7 +34,7 @@ export default function useComboboxOption({
} = context;

const index = useMemo(
() => filteredOptions.indexOf(prefix(String(value))) + customIds.length,
() => filteredOptions.indexOf(value) + customIds.length,
[customIds.length, filteredOptions, value],
);

Expand All @@ -50,7 +49,7 @@ export default function useComboboxOption({
throw new Error('Internal error: ComboboxOption did not find index');
}

const selected = selectedOptions[prefix(value)];
const selected = selectedOptions[value];
const active = activeIndex === index;

useEffect(() => {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,6 @@ import { Box } from '../../../Box';
import { omit } from '../../../../utilities';
import { useComboboxIdDispatch } from '../ComboboxIdContext';
import type { ComboboxProps } from '../Combobox';
import { prefix } from '../useCombobox';

import ComboboxChips from './ComboboxChips';
import ComboboxClearButton from './ComboboxClearButton';
Expand Down Expand Up @@ -69,9 +68,9 @@ export const ComboboxInput = ({
setActiveIndex(0);

// check if input value is the same as a label, if so, select it
const option = options[prefix(value.toLowerCase())];
const option = options[value.toLowerCase()];
if (!option) return;
if (selectedOptions[prefix(option.value)]) return;
if (selectedOptions[option.value]) return;

handleSelectOption({ option: option });
};
Expand Down
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
import type { Option } from '../useCombobox';
import type { ComboboxProps } from '../Combobox';
import { removePrefix } from '../useCombobox';

type ComboboxNativeProps = {
selectedOptions: {
Expand All @@ -15,17 +14,19 @@ export const ComboboxNative = ({
multiple,
name,
}: ComboboxNativeProps) => {
const VALUE = Object.keys(selectedOptions).map((key) => removePrefix(key));

return (
<select
name={name}
multiple={multiple}
style={{ display: 'none' }}
value={multiple ? VALUE : VALUE[0]}
value={
multiple
? Object.keys(selectedOptions)
: Object.keys(selectedOptions)[0]
}
onChange={() => {}}
>
{VALUE.map((value) => (
{Object.keys(selectedOptions).map((value) => (
<option
key={value}
value={value}
Expand Down
32 changes: 6 additions & 26 deletions packages/react/src/components/form/Combobox/useCombobox.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -42,21 +42,6 @@ export function isInteractiveComboboxCustom(
return isComboboxCustom(child) && child.props.interactive === true;
}

const INTERNAL_OPTION_PREFIX = 'internal-option-';

/**
* We use this function to prefix the value of the options so we can make sure numbers as strings are not parsed as numbers in objects
* @param value
* @returns
*/
export const prefix = (value?: string): string => {
return INTERNAL_OPTION_PREFIX + value;
};

export const removePrefix = (value: string): string => {
return value.slice(INTERNAL_OPTION_PREFIX.length);
};

export default function useCombobox({
children,
inputValue,
Expand Down Expand Up @@ -130,8 +115,8 @@ export default function useCombobox({
label = childrenLabel;
}

allOptions[prefix(String(props.value))] = {
value: String(props.value),
allOptions[props.value] = {
value: props.value,
label,
displayValue: props.displayValue,
description: props.description,
Expand All @@ -143,11 +128,7 @@ export default function useCombobox({

const preSelectedOptions = useMemo(
() =>
(
initialValue?.map((key) => {
return prefix(key);
}) || []
).reduce<{
(initialValue || []).reduce<{
[key: string]: Option;
}>((acc, value) => {
const option = options[value];
Expand All @@ -170,17 +151,16 @@ export default function useCombobox({
(option, index) => {
/* If we have a selected value in single mode and the input matches an option, return all children */
if (!multiple && Object.keys(selectedOptions).length === 1) {
filteredOptions.push(option);
filteredOptions.push(options[option].value);
return optionsChildren[index];
}

if (multiple && selectedOptions[option]) {
filteredOptions.push(option);
filteredOptions.push(options[option].value);
return optionsChildren[index];
}

if (filter(inputValue, options[option])) {
filteredOptions.push(option);
filteredOptions.push(options[option].value);
return optionsChildren[index];
}
},
Expand Down

0 comments on commit 7b8695e

Please sign in to comment.