Skip to content
This repository was archived by the owner on Feb 10, 2025. It is now read-only.

fix: creation of multi-select component #117

Merged
merged 2 commits into from
Jan 17, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions src/client.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,3 +3,4 @@ export * from "./primitives/Markdown";
export * from "./primitives/MarkdownEditor";
export * from "./components/Form";
export * from "./components/GenericProgressForm";
export * from "./components/MultipleSelect";
203 changes: 203 additions & 0 deletions src/components/MultipleSelect/MultipleSelect.mdx
Original file line number Diff line number Diff line change
@@ -0,0 +1,203 @@
import { Meta, Story, Controls, Canvas } from "@storybook/blocks";

import * as MultipleSelectStories from "./MultipleSelect.stories";

<Meta
of={MultipleSelectStories}
parameters={{
layout: "centered",
docs: {
story: {
inline: true,
iframeHeight: 200,
},
},
}}
/>

# MultipleSelect

A flexible multi-select component with grouped options, exclusive selections, and collapsible sections.

## Props

### Component Props

- **`options`**: `MultipleSelectGroup[]` - Array of option groups
- **`defaultValue`**: `Record<string, string[]>` - Initial selections per group
- **`placeholder`**: `string` - Text shown when nothing selected if defaultValue then placeholder will be ignored
- **`className`**: `string` - Popover container CSS classes
- **`variants`**: `object` - Style customizations with the following options:
- **`color`**: `"default" | "grey"` - Controls the overall color scheme
- **`size`**: `"default" | "sm"` - Controls component sizing
- **`rounded`**: `"default"` - Controls border radius styling
- **`itemsPosition`**: `"start" | "end" | "center"` - Controls alignment of items in the list
- **`headerPosition`**: `"start" | "end" | "center"` - Controls alignment of group headers
- **`triggerTextColor`**: `"default" | "red" | "green"` - Controls the trigger text color
- **`itemsColor`**: `"default" | "light-grey"` - Controls the color of items and their icons
- **`onChange`**: `(values: Record<string, string[]>) => void` - Selection change handler

### Group Properties (`MultipleSelectGroup`)

- **`groupLabel?`**: `string` - Group name (omit for ungrouped items)
- **`multiple?`**: `boolean` - Allow multiple selections (default: true)
- **`collapsible?`**: `boolean` - Enable group toggle
- **`items`**: `MultipleSelectItem[]` - Array of options

### Item Properties (`MultipleSelectItem`)

- **`value`**: `string` - Unique identifier
- **`label`**: `string` - Display text
- **`exclusive?`**: `boolean` - Clear other selections when chosen
- **`exclusiveScope?`**: `"group" | "global"` - Clear scope for exclusive items
- **`iconType?`**: `IconType` - Optional icon identifier check `@/primitives/Icon story`
- **`icon?`**: `React.ComponentType` - Optional icon component used if no iconType is provided
- **`itemClassName?`**: `string` - Item-specific CSS classes

---

<Canvas of={MultipleSelectStories.FilterExample}>
<div style={{ display: "flex", justifyContent: "center", alignItems: "center", padding: "1rem" }}>
<Story
of={MultipleSelectStories.FilterExample}
parameters={{
docs: {
canvas: { sourceState: "shown" },
},
}}
/>
</div>
</Canvas>
<Controls of={MultipleSelectStories.FilterExample} />

## Stories

Below are several usage examples showcasing common patterns and advanced features.

### 1. Basic

<Canvas of={MultipleSelectStories.Basic}>
<div style={{ display: "flex", justifyContent: "center", alignItems: "center", padding: "1rem" }}>
<Story
of={MultipleSelectStories.Basic}
parameters={{
layout: "centered",
docs: {
canvas: { sourceState: "shown" },
},
}}
/>
</div>
</Canvas>

**Description**
A single ungrouped set of items (no group label) with simple placeholder text.

- **Options**: 3 items, no exclusivity.
- **`onChange`** logs the selection to console.

### 2. OrderByExample

<Canvas of={MultipleSelectStories.OrderByExample}>
<div style={{ display: "flex", justifyContent: "center", alignItems: "center", padding: "1rem" }}>
<Story
of={MultipleSelectStories.OrderByExample}
parameters={{
layout: "centered",
docs: {
canvas: { sourceState: "shown" },
},
}}
/>
</div>
</Canvas>

**Description** Demonstrates **exclusive items** across two single-select groups (“Order by time” and
“Order by name”). Each item has `exclusive: true, exclusiveScope: 'global'`, ensuring that if you pick
“Recent,” it clears any “A-Z” or “Z-A” selection.

- **`defaultValue`** sets the initial selection to `["Recent"]` in the “ORDER BY TIME” group.
- **`variants`** example usage for adjusting text color, alignment, etc.

### 3. FilterExample

<Canvas of={MultipleSelectStories.FilterExample}>
<div style={{ display: "flex", justifyContent: "center", alignItems: "center", padding: "1rem" }}>
<Story
of={MultipleSelectStories.FilterExample}
parameters={{
layout: "centered",
docs: {
canvas: { sourceState: "shown" },
},
}}
/>
</div>
</Canvas>

**Description** A **filter** scenario with an **“All”** exclusive item for resetting.

- “All” is in an **ungrouped** set with `exclusive: true, exclusiveScope: 'global'`.
- Two collapsible groups, “Network” and “Status,” each with `multiple: true`.
- **`defaultValue`** starts with “All” selected in the ungrouped items.

### 4. MixedSelectionTypes

<Canvas of={MultipleSelectStories.MixedSelectionTypes}>
<div style={{ display: "flex", justifyContent: "center", alignItems: "center", padding: "1rem" }}>
<Story
of={MultipleSelectStories.MixedSelectionTypes}
parameters={{
layout: "centered",
docs: {
canvas: { sourceState: "shown" },
},
}}
/>
</div>
</Canvas>

**Description** Shows a combination of:

- A group with an **exclusive** item called “Reset All.”
- Another group that’s **multi-select** with standard options.

This approach is useful when you have a special action (like “Reset” or “Clear All”) plus normal multi-select items.

### 5. WithVariants

<Canvas of={MultipleSelectStories.WithVariants}>
<div style={{ display: "flex", justifyContent: "center", alignItems: "center", padding: "1rem" }}>
<Story
of={MultipleSelectStories.WithVariants}
parameters={{
layout: "centered",
docs: {
canvas: { sourceState: "shown" },
},
}}
/>
</div>
</Canvas>

**Description**
Showcases using **all variant props** (`triggerTextColor`, `headerPosition`, `itemsPosition`, etc.) or whatever your **MultipleSelect** has to customize.

- You can set text color, alignment, plus a collapsible group.
- Useful to see how to pass styling variants from **tailwind-variants** to the component.

---

## Tips & Best Practices

1. **Use `defaultValue`** for initial selections, or manage selections outside if you want it to be “controlled.”
2. **`exclusive`** items are great for “select all,” “select none,” or “reset.”
3. **Collapsible groups** let you nest large sets of items without overwhelming the user.

---

## Conclusion

`MultipleSelect` is highly **configurable** for your needs: from a simple single group with no advanced logic, to multi-group filters, collapsible sections, exclusive “all” items, or complex order-by pickers.

Check out the source code & stories for additional usage details.
172 changes: 172 additions & 0 deletions src/components/MultipleSelect/MultipleSelect.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
import { action } from "@storybook/addon-actions";
import type { Meta, StoryObj } from "@storybook/react";

import { MultipleSelect } from "./MultipleSelect";

const onChange = action("onChange");

const meta = {
title: "Components/MultipleSelect",
component: MultipleSelect,
parameters: {
layout: "centered",
},
} satisfies Meta<typeof MultipleSelect>;

export default meta;
type Story = StoryObj<typeof MultipleSelect>;

// Basic example with single group
export const Basic: Story = {
args: {
options: [
{
items: [
{ value: "1", label: "Option 1" },
{ value: "2", label: "Option 2" },
{ value: "3", label: "Option 3" },
],
},
],
placeholder: "Select options",
onChange: (values) => onChange(values),
},
};

// Example with exclusive options (like order by)
export const OrderByExample: Story = {
args: {
options: [
{
groupLabel: "ORDER BY TIME",
multiple: false,
items: ["Recent", "Oldest"].map((value) => ({
label: value,
value,
exclusive: true,
exclusiveScope: "global",
})),
},
{
groupLabel: "ORDER BY NAME",
multiple: false,
items: ["A-Z", "Z-A"].map((value) => ({
label: value,
value,
exclusive: true,
exclusiveScope: "global",
})),
},
],
defaultValue: { "ORDER BY TIME": ["Recent"] },
variants: {
triggerTextColor: "green",
headerPosition: "end",
itemsPosition: "end",
},
placeholder: "Order by",
className: "w-40",
onChange: (values) => onChange(values),
},
};

// Example with filters including "All" option and collapsible groups
export const FilterExample: Story = {
args: {
options: [
{
multiple: false,
items: [
{
label: "All",
value: "All-id",
exclusive: true,
exclusiveScope: "global",
},
],
},
{
groupLabel: "Network",
multiple: true,
collapsible: true,
items: [
{ label: "Rounds on Ethereum", value: "1" },
{ label: "Rounds on Polygon", value: "137" },
{ label: "Rounds on Optimism", value: "10" },
],
},
{
groupLabel: "Status",
multiple: true,
collapsible: true,
items: [
{ label: "Active", value: "active" },
{ label: "Taking Applications", value: "applications" },
{ label: "Finished", value: "finished" },
],
},
],
defaultValue: { ungrouped: ["All-id"] },
variants: { triggerTextColor: "red" },
placeholder: "Filter by",
className: "w-64",
onChange: (values) => onChange(values),
},
};

// Example with mixed exclusive and non-exclusive options
export const MixedSelectionTypes: Story = {
args: {
options: [
{
groupLabel: "Special Actions",
items: [
{
label: "Reset All",
value: "reset",
exclusive: true,
exclusiveScope: "global",
},
],
},
{
groupLabel: "Regular Options",
multiple: true,
items: [
{ label: "Option 1", value: "1" },
{ label: "Option 2", value: "2" },
{ label: "Option 3", value: "3" },
],
},
],
placeholder: "Select options",
onChange: (values) => onChange(values),
},
};

// Example with all variant options
export const WithVariants: Story = {
args: {
options: [
{
groupLabel: "Group 1",
multiple: true,
collapsible: true,
items: [
{ label: "Option 1", value: "1" },
{ label: "Option 2", value: "2" },
],
},
],

variants: {
triggerTextColor: "green",
headerPosition: "end",
itemsPosition: "end",
},

placeholder: "Select with variants",
onChange: (values) => onChange(values),
className: "w-40",
},
};
Loading
Loading