Стилизация ReactSelect с помощью CSS-in-JS

Вот ссылка на другой хороший stackoverflow сообщение, на котором я основываю свой первоначальный ответ. Я пытаюсь стилизовать свой компонент ReactSelect так, чтобы он выглядел примерно так:

введите здесь описание изображения

На снимке экрана выше это не очевидно, но поле выбора имеет гораздо меньшую высоту (всего 29 пикселей), чем ReactSelect по умолчанию, и это моя первая цель со стилем (уменьшить высоту). Вот мой код прямо сейчас, пытающийся уменьшить высоту, полученный в основном из сообщения stackoverflow, на которое я ссылался:

const customStyles = {
    control: base => ({
        ...base,
        height: 22,
        minHeight: 22
    })
};

const customControlStyles = base => ({
    ...base,
    height: 22,
    minHeight: 22
});

const selectOptions = [
    { value: 'pct', label: 'FG Percentage' },
    { value: 'attFreq', label: 'FG Frequency' },
    { value: 'made', label: 'Total Makes' },
    { value: 'att', label: 'Total Attempts' }];

const shotdistSelect =
    (<Select
        // arrowRenderer={null}
        maxMenuHeight={30}
        placeholder={'Knickerbockers'}
        isClearable={false}
        isDisabled={this.props.loading}
        backspaceRemovesValue={false}
        isSearchable={true}
        value={this.state.shotdistType}
        onChange={this.handleShotdistChange}
        options={selectOptions}
        styles={{ control: customControlStyles }}
        // styles={{ customStyles }}
    />);

И вот результат примера выше:

введите здесь описание изображения

... не совсем то, что я собирался. Кроме того, когда я использую customStyles вместо customControlStyles в приведенном выше примере, стиль больше не работает, и я не уверен, что я сделал неправильно при создании customStyles, из-за чего он не работает. Я полагаю, что мне нужно будет сделать что-то похожее на customStyles, так как кажется, что мне нужно будет стилизовать больше, чем просто часть control ReactSelect.

И во-вторых, я хотел бы удалить как вертикальную черту, так и курсор вниз в ReactSelect, как на исходном снимке экрана.

Любая помощь с этим стилем будет принята с благодарностью! Я работаю над этим в течение довольно долгого времени, пока безуспешно. Спасибо!


person Canovice    schedule 09.02.2019    source источник


Ответы (1)


Вариант 1

Убедитесь, что вы используете самую последнюю версию react-select (v2.3.0). Я смог выполнить то, что вы хотите, используя немного CSS с style keys, предлагаемым react-select.

Рабочий пример: https://codesandbox.io/s/7y6901y950.

контейнеры/форма/Form.js

import React, { Component } from "react";
import CustomSelect from "../../components/CustomSelect/CustomSelect";

const fgOptions = [
  { value: "pct", label: "FG Percentage" },
  { value: "attFreq", label: "FG Frequency" },
  { value: "made", label: "Total Makes" },
  { value: "att", label: "Total Attempts" }
];

const saveOptions = [
  { value: "pct", label: "Save Percentage" },
  { value: "sFreq", label: "Save Frequency" },
  { value: "tSaves", label: "Total Saves" }
];

const assistOptions = [
  { value: "pct", label: "Assist Percentage" },
  { value: "aFreq", label: "Assist Frequency" },
  { value: "tAssist", label: "Total Assists" }
];

export default class Form extends Component {
  handleChange = (name, value) => {
    this.setState({ [name]: value });
  };

  handleSubmit = e => {
    e.preventDefault();
    alert(JSON.stringify(this.state, null, 4));
  };

  render = () => (
    <form onSubmit={this.handleSubmit} className="app-container">
      <h1>React Select Styling</h1>
      <CustomSelect
        name="fg"
        label="FG:"
        placeholder="Field Goals"
        handleChange={this.handleChange}
        selectOptions={fgOptions}
      />
      <CustomSelect
        name="assists"
        label="AS:"
        placeholder="Assists"
        handleChange={this.handleChange}
        selectOptions={assistOptions}
      />
      <CustomSelect
        name="saves"
        label="SV:"
        placeholder="Saves"
        handleChange={this.handleChange}
        selectOptions={saveOptions}
      />
      <button type="submit" className="submit">
        Submit
      </button>
    </form>
  );
}

components/CustomSelect/CustomSelect.js

import React from "react";
import PropTypes from "prop-types";
import Select from "react-select";
import { labelStyles, selectStyles } from "./styles/styles";

const CustomSelect = ({
  handleChange,
  label,
  name,
  placeholder,
  selectOptions,
  value
}) => (
  <div className="select-container">
    <label htmlFor={name} style={labelStyles}>
      {label}
    </label>
    <Select
      name={name}
      placeholder={placeholder}
      isClearable={false}
      backspaceRemovesValue={false}
      isSearchable={true}
      value={value}
      onChange={value => handleChange(name, value)}
      options={selectOptions}
      styles={selectStyles}
    />
  </div>
);

CustomSelect.propTypes = {
  handleChange: PropTypes.func.isRequired,
  label: PropTypes.string.isRequired,
  name: PropTypes.string.isRequired,
  placeholder: PropTypes.string.isRequired,
  selectOptions: PropTypes.arrayOf(
    PropTypes.shape({
      label: PropTypes.string.isRequired,
      value: PropTypes.string.isRequired
    })
  ),
  value: PropTypes.objectOf({
    value: PropTypes.string,
    label: PropTypes.string
  })
};

export default CustomSelect;

components/CustomSelect/styles/styles.js (см. документацию для ключей стилей -- если вы добавляете более длинные метки, то вы должны настроить свойство labelStyles width, иначе соотношение label к select будет другим)

export const selectStyles = {
  option: (provided, state) => ({
    ...provided,
    borderBottom: "1px dotted pink",
    color: state.isSelected ? "blue" : "",
    fontSize: 16,
    backgroundColor: state.isSelected ? "#eee" : "",
    textAlign: "left",
    cursor: "pointer"
  }),
  container: base => ({
    ...base,
    width: "100%"
  }),
  control: base => ({
    ...base,
    height: 32,
    minHeight: 32,
    fontSize: 16,
    borderRadius: 0,
    width: "100%",
    textAlign: "left",
    cursor: "pointer"
  }),
  dropdownIndicator: base => ({
    ...base,
    display: "none"
  }),
  indicatorSeparator: base => ({
    ...base,
    display: "none"
  }),
  valueContainer: base => ({
    ...base,
    padding: 0,
    paddingLeft: 2
  })
};

export const labelStyles = {
  fontSize: 16,
  paddingTop: 8,
  marginRight: 5,
  width: 50,
  textAlign: "right"
};

styles.css

.app-container {
  padding: 0px 20px;
  text-align: center;
  font-family: sans-serif;
}

.select-container {
  display: flex;
  margin: 0 auto;
  width: 100%;
  max-width: 500px;
  -webkit-box-orient: vertical;
  -webkit-box-direction: normal;
  -webkit-flex-direction: row;
  -ms-flex-direction: row;
  flex-direction: row;
  -webkit-box-flex: 1;
  margin-bottom: 10px;
}

.submit {
  cursor: pointer;
  background-color: #1e87f0;
  color: #fff;
  border: 1px solid transparent;
  box-sizing: border-box;
  padding: 0 30px;
  vertical-align: middle;
  font-size: 14px;
  line-height: 38px;
  text-transform: uppercase;
  transition: 0.1s ease-in-out;
  transition-property: color, background-color, border-color;
}

.submit:hover {
  background-color: #0f7ae5;
  color: #fff;
}

.submit:focus {
  background-color: #1e87f0;
  color: #fff;
  outline: none;
}

Вариант 2

Или вы можете делать все, не используя react-select, что я очень рекомендую, так как это исключает еще одну зависимость! Таким образом, у вас есть возможность оформить его по своему усмотрению (полностью через css, полностью через css-in-js или их комбинацию).

Рабочий пример: https://codesandbox.io/s/w72k49nn27 ( в этом примере используется только css)

person Matt Carlotta    schedule 10.02.2019
comment
Обновлен ответ, чтобы включить 2-й вариант (который я настоятельно рекомендую вместо 1-го). - person Matt Carlotta; 10.02.2019