Skip to content

smackjax/ez-react-select

Repository files navigation

ez-react-select-input

An easy select input limited only by imagination!!!(...and css)

Description

The major difference between this and a vanilla HTML select element, aside from styling(see below), is this <Select> holds any element type you want in it's <Option> children, or even the <Select> itself. I've intentionally kept it as mechanically un-opinionated as I can. The upside is, each <Option> can look however you want, containing whatever you want, and you can even have children besides <Option>s inside the main <Select>. The downside being any styling that needs to be done is on the developer, although I've tried to give a couple useful examples.

Examples

Frameworks/examples can be seen by importing ez-react-select/frameworks.

import { BlankSelect, BasicSelect, ColorSelect } from 'ez-react-select/frameworks';

Usage

npm install --save ez-react-select

I tried to make this as easy as possible to use, and as close to standard html as I could get. I.E:

import { Select, Option } from 'ez-react-select';
<Select
name="(Form value name)"
id="(Optional, but if set then the dropdown id will be this + '-dropdown')"
className="(Applied to main container)"
selectedClassName="(Applied to the 'selected' item)"
dropdownClassName="(Applied to the hidden dropdown)"
optionClassName="(Applied to all Option element children)"
onChange={Required}
>
    <Option
    className="(optional)"
    value="option1"
    placeholder={true}
    selected={true}
    >
        (anything here)
    </Option>

    <hr className="Check it out! Totally fine!" />

    <Option
    className="(still optional)"
    value="option2"
    selected={false}
    >
        (anything here)
    </Option>

</Select>

The guts

  1. All children inside the <Select> tags are enumerated over, extracting the <Option> with the 'placeholder' attribute. Any additional <Option> with 'placeholder' will overwrite the previous one. Note This removes any Options with placeholder from the select dropdown.
  2. The children are enumerated over again, this time creating a reference list of <Option> elements, appending classes and initializing onClick.
  3. When an <Option> is clicked, the <Option> is retrieved from the reference list held in state by its index.
  4. The 'selected' <Option> is cloned, put in the main wrapper, appending the 'selectedClassName' attribute and changing its onClick to a toggle function. This also 'checks' the hidden radio input, changing the form value.

References

Predefined className's

These are always defined and appended, but custom classes added will override(not overwrite) them.

  • .ez-select-wrapper - <Select> wrapper
  • .ez-select-dropdown - Dropdown div
  • .ez-select-item - All <Option> elements
  • .selected - Child being displayed, including placeholders

<Select> className props:

The main <Select> element accepts these className props and appends them onto the various elements

  • className - applied to the main div containing everything. And I do mean everything.
  • selectedClassName - applied to whatever item is selected, including placeholders.
  • dropdownClassName - applied to the div that's toggled open by clicking whatever item is selected/a placeholder.
  • optionClassName - applied to all Option elements.

Useful info

  • There has to be an <Option> with 'placeholder', otherwise there won't be an initial <Option> to click on.
  • An <Option> with a 'placeholder' attribute but not 'selected' will not set a value by default, but will be shown by default.
  • When an <Option> is 'selected' the whole child is copied and placed in main div.
  • An <Option> can have it's own onChange, which will override the <Select> onChange. This could change in the future, right now it stops event propagation with no way to continue it, which cancels the child from being selected. That's obviously not ideal.

Contributing

The Core

<Select>, <SelectDropdown>, <Option>

If you feel like taking a crack at any of these, or think it would benefit from a different feature, feel free. I just ask if it limits the customization(for example, limits children to only be <Options>), you make a 'framework' instead of changing the core.

  • Accessibility - I have no idea how to make this accessible; but it is a priority.
  • Tabbable - ...and currently, it's not. Which can be really annoying if you have a good tab flow through your form.
  • Soft navigation - With up/down arrow keys and enter.
  • props.style - Add to handling to <Option> and <SelectDropdown>
  • e.target - Currently e.target is pointing at a hidden radio input, which works for getting an onChange value, but not much else. It would be nice to have it pointed at something more useful, but I'm open as to what that should be.
  • Add/Remove <Option> after mount - I haven't tested this at all. Could open up some interesting possibilities.
  • OptGroup - I can't think of an implementation that would play nice with the level of customization <Select> allows, but it wouldn't hurt to explore the idea in order to bring this closer to standard HTML.
  • Tests - Yes. Yes I should have.

Frameworks

If you're not sure how to test a component with ref, @gaearon gave an awesome(and easy) answer here.

I tried to make this easy, but here's the process:

  • Setup
    • Copy ./_src/frameworks/_blank-select or
    • Create your own folder in ./_src/frameworks/
  • Once inside your folder, name and structure things however you want
  • Create your Select framework component
  • Make it the default export of the your main file
  • Add it to /_src/frameworks/index.js with PascalCase
  • npm run build, if you haven't already

Inspiration

In making my app (shameless plug: Incooknito Kitchen), I thought it would be awesome to have an input where I could show color splotches next to their corresponding text options; but all the select input components I found only showed text, or didn't allow for the usual HTML-style markup. This project turned from a select input into more of a dropdown-esque framework, but I guess that's what a select input is at heart anyway.

Releases

No releases published

Packages

No packages published