Skip to content

Commit

Permalink
Merge pull request #548 from sebgroup/feature/dropdown-custom-labels
Browse files Browse the repository at this point in the history
feat(dropdown): adds custom lables for selected item
  • Loading branch information
mario-subo authored Mar 23, 2021
2 parents c029079 + 5c22af3 commit c349c0c
Show file tree
Hide file tree
Showing 2 changed files with 72 additions and 3 deletions.
60 changes: 60 additions & 0 deletions lib/src/Dropdown/Dropdown.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -181,6 +181,66 @@ describe("Component: Dropdown", () => {
expect(container.querySelector("select").multiple).toBe(true);
});

it("Should render custom static label", () => {
const hardcodedLabel: string = "Always show this text";
act(() => {
render(
<Dropdown selectedLabel={hardcodedLabel} multiple>
{testOptions}
</Dropdown>,
container
);
});

expect(container.querySelector(".rc.custom-dropdown > .dropdown > .btn.dropdown-toggle > span").textContent).toBe(hardcodedLabel);
});

it("Should render custom label for selected element", () => {
const customLabels: { [k: string]: string } = {
1: "One",
2: "Two",
3: "Three",
};

const getLabel = (value: string) => {
return customLabels[value];
};

act(() => {
render(
<Dropdown selectedLabel={getLabel} value="3">
{testOptions}
</Dropdown>,
container
);
});

expect(container.querySelector(".rc.custom-dropdown > .dropdown > .btn.dropdown-toggle > span").textContent).toBe(customLabels["3"]);
});

it("Should render custom label and separator for selected elements", () => {
const customLabels: { [k: string]: string } = {
1: "One",
2: "Two",
3: "Three",
};

const getLabel = (value: string[]) => {
return value.map((e) => customLabels[e]).join(" + ");
};

act(() => {
render(
<Dropdown selectedLabel={getLabel} multiple value={["1", "2"]}>
{testOptions}
</Dropdown>,
container
);
});

expect(container.querySelector(".rc.custom-dropdown > .dropdown > .btn.dropdown-toggle > span").textContent).toBe("One + Two");
});

it("Should render native only when used in a mobile device", () => {
const userAgent: string = window.navigator.userAgent;
const onChange: jest.Mock = jest.fn();
Expand Down
15 changes: 12 additions & 3 deletions lib/src/Dropdown/Dropdown.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -42,11 +42,13 @@ export type DropdownProps = Omit<JSX.IntrinsicElements["select"], "value"> & {
searchable?: boolean;
/** Allows clearing the dropdown with a clear button */
clearable?: boolean;
/** Allows setting custom label to be displayed for selected item */
selectedLabel?: string | ((value: string | string[]) => string | string[]);
/** Custom texts to be dispalyed in different parts of the dropdown */
text?: DropdownText;
};

export const Dropdown: React.FC<DropdownProps> = React.forwardRef(({ wrapperProps = {}, text = {}, onMultipleChange, searchable, clearable, ...props }: DropdownProps, ref) => {
export const Dropdown: React.FC<DropdownProps> = React.forwardRef(({ wrapperProps = {}, text = {}, onMultipleChange, searchable, clearable, selectedLabel, ...props }: DropdownProps, ref) => {
const [toggleId] = React.useState<string>(randomId("ddt-"));
const [selectAllId] = React.useState<string>(randomId("sa-"));
const [show, setShow] = React.useState<boolean>(false);
Expand Down Expand Up @@ -210,8 +212,15 @@ export const Dropdown: React.FC<DropdownProps> = React.forwardRef(({ wrapperProp
}, [show]);

React.useEffect(() => {
!isMobile && setLabel((Array.isArray(props.value) ? props.value.join(", ") : props.value) || props.placeholder);
}, [props.value, props.placeholder]);
if (selectedLabel && typeof selectedLabel === "string") {
!isMobile && setLabel(selectedLabel || props.placeholder);
} else if (selectedLabel && typeof selectedLabel === "function") {
const newLabel: string | string[] = selectedLabel(props.value);
!isMobile && setLabel((Array.isArray(newLabel) ? newLabel.join(", ") : newLabel) || props.placeholder);
} else {
!isMobile && setLabel((Array.isArray(props.value) ? props.value.join(", ") : props.value) || props.placeholder);
}
}, [props.value, props.placeholder, selectedLabel]);

return (
<div {...wrapperProps} className={classnames("rc custom-dropdown", wrapperProps.className)}>
Expand Down

0 comments on commit c349c0c

Please sign in to comment.