mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2026-01-23 19:54:08 +00:00
Compare commits
2 Commits
master
...
users/nish
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
14ed7454fc | ||
|
|
387575ae46 |
194
src/Common/SearchableDropdown.tsx
Normal file
194
src/Common/SearchableDropdown.tsx
Normal file
@@ -0,0 +1,194 @@
|
|||||||
|
import { Callout, DirectionalHint, ISearchBoxStyles, Label, SearchBox } from "@fluentui/react";
|
||||||
|
import * as React from "react";
|
||||||
|
import { useCallback, useRef, useState } from "react";
|
||||||
|
|
||||||
|
interface SearchableDropdownProps<T> {
|
||||||
|
label: string;
|
||||||
|
items: T[];
|
||||||
|
selectedItem: T | null;
|
||||||
|
onSelect: (item: T) => void;
|
||||||
|
getKey: (item: T) => string;
|
||||||
|
getDisplayText: (item: T) => string;
|
||||||
|
placeholder?: string;
|
||||||
|
filterPlaceholder?: string;
|
||||||
|
className?: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
onDismiss?: () => void;
|
||||||
|
buttonStyles?: React.CSSProperties;
|
||||||
|
searchBoxStyles?: Partial<ISearchBoxStyles>;
|
||||||
|
listStyles?: React.CSSProperties;
|
||||||
|
itemStyles?: React.CSSProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const SearchableDropdown = <T,>({
|
||||||
|
label,
|
||||||
|
items,
|
||||||
|
selectedItem,
|
||||||
|
onSelect,
|
||||||
|
getKey,
|
||||||
|
getDisplayText,
|
||||||
|
placeholder = "Select an item",
|
||||||
|
filterPlaceholder = "Filter items",
|
||||||
|
className,
|
||||||
|
disabled = false,
|
||||||
|
onDismiss,
|
||||||
|
buttonStyles: customButtonStyles,
|
||||||
|
searchBoxStyles: customSearchBoxStyles,
|
||||||
|
listStyles: customListStyles,
|
||||||
|
itemStyles: customItemStyles,
|
||||||
|
}: SearchableDropdownProps<T>): React.ReactElement => {
|
||||||
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
|
const [filterText, setFilterText] = useState("");
|
||||||
|
const buttonRef = useRef<HTMLButtonElement>(null);
|
||||||
|
|
||||||
|
const defaultButtonStyles: React.CSSProperties = {
|
||||||
|
width: "100%",
|
||||||
|
height: "32px",
|
||||||
|
padding: "0 28px 0 8px",
|
||||||
|
border: "1px solid #8a8886",
|
||||||
|
background: "#fff",
|
||||||
|
color: "#323130",
|
||||||
|
textAlign: "left",
|
||||||
|
cursor: disabled ? "not-allowed" : "pointer",
|
||||||
|
position: "relative",
|
||||||
|
fontFamily: "inherit",
|
||||||
|
fontSize: "14px",
|
||||||
|
};
|
||||||
|
|
||||||
|
const defaultListStyles: React.CSSProperties = {
|
||||||
|
maxHeight: "300px",
|
||||||
|
overflowY: "auto",
|
||||||
|
};
|
||||||
|
|
||||||
|
const defaultItemStyles: React.CSSProperties = {
|
||||||
|
padding: "8px 12px",
|
||||||
|
cursor: "pointer",
|
||||||
|
fontSize: "14px",
|
||||||
|
};
|
||||||
|
|
||||||
|
// Merge custom styles with defaults
|
||||||
|
const buttonStyles = { ...defaultButtonStyles, ...customButtonStyles };
|
||||||
|
const listStyles = { ...defaultListStyles, ...customListStyles };
|
||||||
|
const itemStyles = { ...defaultItemStyles, ...customItemStyles };
|
||||||
|
|
||||||
|
const closeDropdown = useCallback(() => {
|
||||||
|
setIsOpen(false);
|
||||||
|
setFilterText("");
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const filteredItems = items?.filter((item) => getDisplayText(item).toLowerCase().includes(filterText.toLowerCase()));
|
||||||
|
|
||||||
|
const handleDismiss = useCallback(() => {
|
||||||
|
closeDropdown();
|
||||||
|
onDismiss?.();
|
||||||
|
}, [closeDropdown, onDismiss]);
|
||||||
|
|
||||||
|
const handleButtonClick = useCallback(
|
||||||
|
(event: React.MouseEvent) => {
|
||||||
|
if (disabled) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
|
||||||
|
if (isOpen) {
|
||||||
|
closeDropdown();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setIsOpen(true);
|
||||||
|
return;
|
||||||
|
},
|
||||||
|
[isOpen, closeDropdown, disabled],
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleSelect = useCallback(
|
||||||
|
(item: T) => {
|
||||||
|
onSelect(item);
|
||||||
|
closeDropdown();
|
||||||
|
},
|
||||||
|
[onSelect, closeDropdown],
|
||||||
|
);
|
||||||
|
|
||||||
|
const buttonLabel = selectedItem
|
||||||
|
? getDisplayText(selectedItem)
|
||||||
|
: items?.length === 0
|
||||||
|
? `No ${label}s Found`
|
||||||
|
: placeholder;
|
||||||
|
|
||||||
|
const buttonId = `${className}-button`;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<Label htmlFor={buttonId}>{label}</Label>
|
||||||
|
<button
|
||||||
|
id={buttonId}
|
||||||
|
ref={buttonRef}
|
||||||
|
className={className}
|
||||||
|
onClick={handleButtonClick}
|
||||||
|
style={buttonStyles}
|
||||||
|
disabled={disabled}
|
||||||
|
>
|
||||||
|
<span style={{ overflow: "hidden", textOverflow: "ellipsis", whiteSpace: "nowrap", display: "block" }}>
|
||||||
|
{buttonLabel}
|
||||||
|
</span>
|
||||||
|
<span
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
right: "8px",
|
||||||
|
top: "50%",
|
||||||
|
transform: "translateY(-50%)",
|
||||||
|
pointerEvents: "none",
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
▼
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
{isOpen && (
|
||||||
|
<Callout
|
||||||
|
target={buttonRef.current}
|
||||||
|
onDismiss={handleDismiss}
|
||||||
|
directionalHint={DirectionalHint.bottomLeftEdge}
|
||||||
|
isBeakVisible={false}
|
||||||
|
gapSpace={0}
|
||||||
|
setInitialFocus
|
||||||
|
>
|
||||||
|
<div style={{ width: buttonRef.current?.offsetWidth || 300, display: "flex", flexDirection: "column" }}>
|
||||||
|
<SearchBox
|
||||||
|
placeholder={filterPlaceholder}
|
||||||
|
value={filterText}
|
||||||
|
onChange={(_, newValue) => setFilterText(newValue || "")}
|
||||||
|
styles={customSearchBoxStyles}
|
||||||
|
/>
|
||||||
|
<div style={listStyles}>
|
||||||
|
{filteredItems && filteredItems.length > 0 ? (
|
||||||
|
filteredItems.map((item) => {
|
||||||
|
const key = getKey(item);
|
||||||
|
const isSelected = selectedItem ? getKey(selectedItem) === key : false;
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
key={key}
|
||||||
|
onClick={() => handleSelect(item)}
|
||||||
|
style={{
|
||||||
|
...itemStyles,
|
||||||
|
backgroundColor: isSelected ? "#e6e6e6" : "transparent",
|
||||||
|
}}
|
||||||
|
onMouseEnter={(e) => (e.currentTarget.style.backgroundColor = "#f3f2f1")}
|
||||||
|
onMouseLeave={(e) =>
|
||||||
|
(e.currentTarget.style.backgroundColor = isSelected ? "#e6e6e6" : "transparent")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{getDisplayText(item)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})
|
||||||
|
) : (
|
||||||
|
<div style={{ padding: "8px 12px", color: "#605e5c" }}>No items found</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Callout>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import { Dropdown } from "@fluentui/react";
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { FunctionComponent } from "react";
|
import { FunctionComponent } from "react";
|
||||||
|
import { SearchableDropdown } from "../../../Common/SearchableDropdown";
|
||||||
import { DatabaseAccount } from "../../../Contracts/DataModels";
|
import { DatabaseAccount } from "../../../Contracts/DataModels";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -17,23 +17,18 @@ export const SwitchAccount: FunctionComponent<Props> = ({
|
|||||||
dismissMenu,
|
dismissMenu,
|
||||||
}: Props) => {
|
}: Props) => {
|
||||||
return (
|
return (
|
||||||
<Dropdown
|
<SearchableDropdown<DatabaseAccount>
|
||||||
label="Cosmos DB Account Name"
|
label="Cosmos DB Account Name"
|
||||||
|
items={accounts}
|
||||||
|
selectedItem={selectedAccount}
|
||||||
|
onSelect={(account) => setSelectedAccountName(account.name)}
|
||||||
|
getKey={(account) => account.name}
|
||||||
|
getDisplayText={(account) => account.name}
|
||||||
|
placeholder="Select an Account"
|
||||||
|
filterPlaceholder="Filter accounts"
|
||||||
className="accountSwitchAccountDropdown"
|
className="accountSwitchAccountDropdown"
|
||||||
options={accounts?.map((account) => ({
|
disabled={!accounts || accounts.length === 0}
|
||||||
key: account.name,
|
onDismiss={dismissMenu}
|
||||||
text: account.name,
|
|
||||||
data: account,
|
|
||||||
}))}
|
|
||||||
onChange={(_, option) => {
|
|
||||||
setSelectedAccountName(String(option?.key));
|
|
||||||
dismissMenu();
|
|
||||||
}}
|
|
||||||
defaultSelectedKey={selectedAccount?.name}
|
|
||||||
placeholder={accounts && accounts.length === 0 ? "No Accounts Found" : "Select an Account"}
|
|
||||||
styles={{
|
|
||||||
callout: "accountSwitchAccountDropdownMenu",
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { Dropdown } from "@fluentui/react";
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { FunctionComponent } from "react";
|
import { FunctionComponent } from "react";
|
||||||
|
import { SearchableDropdown } from "../../../Common/SearchableDropdown";
|
||||||
import { Subscription } from "../../../Contracts/DataModels";
|
import { Subscription } from "../../../Contracts/DataModels";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
@@ -15,24 +15,16 @@ export const SwitchSubscription: FunctionComponent<Props> = ({
|
|||||||
selectedSubscription,
|
selectedSubscription,
|
||||||
}: Props) => {
|
}: Props) => {
|
||||||
return (
|
return (
|
||||||
<Dropdown
|
<SearchableDropdown<Subscription>
|
||||||
label="Subscription"
|
label="Subscription"
|
||||||
|
items={subscriptions}
|
||||||
|
selectedItem={selectedSubscription}
|
||||||
|
onSelect={(sub) => setSelectedSubscriptionId(sub.subscriptionId)}
|
||||||
|
getKey={(sub) => sub.subscriptionId}
|
||||||
|
getDisplayText={(sub) => sub.displayName}
|
||||||
|
placeholder="Select a Subscription"
|
||||||
|
filterPlaceholder="Filter subscriptions"
|
||||||
className="accountSwitchSubscriptionDropdown"
|
className="accountSwitchSubscriptionDropdown"
|
||||||
options={subscriptions?.map((sub) => {
|
|
||||||
return {
|
|
||||||
key: sub.subscriptionId,
|
|
||||||
text: sub.displayName,
|
|
||||||
data: sub,
|
|
||||||
};
|
|
||||||
})}
|
|
||||||
onChange={(_, option) => {
|
|
||||||
setSelectedSubscriptionId(String(option?.key));
|
|
||||||
}}
|
|
||||||
defaultSelectedKey={selectedSubscription?.subscriptionId}
|
|
||||||
placeholder={subscriptions && subscriptions.length === 0 ? "No Subscriptions Found" : "Select a Subscription"}
|
|
||||||
styles={{
|
|
||||||
callout: "accountSwitchSubscriptionDropdownMenu",
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user