Current File : /home/pacjaorg/wpt.pacja.org/wp-content/plugins/leadin/scripts/shared/Common/AsyncSelect.tsx
import React, { useRef, useState, useEffect } from 'react';
import { styled } from '@linaria/react';
import {
  CALYPSO,
  CALYPSO_LIGHT,
  CALYPSO_MEDIUM,
  OBSIDIAN,
} from '../UIComponents/colors';
import UISpinner from '../UIComponents/UISpinner';
import LoadState, { LoadStateType } from '../enums/loadState';

const Container = styled.div`
  color: ${OBSIDIAN};
  font-family: 'Lexend Deca', Helvetica, Arial, sans-serif;
  font-size: 14px;
  position: relative;
`;

interface IControlContainerProps {
  focused: boolean;
}

const ControlContainer = styled.div<IControlContainerProps>`
  align-items: center;
  background-color: hsl(0, 0%, 100%);
  border-color: hsl(0, 0%, 80%);
  border-radius: 4px;
  border-style: solid;
  border-width: ${props => (props.focused ? '0' : '1px')};
  cursor: default;
  display: flex;
  flex-wrap: wrap;
  justify-content: space-between;
  min-height: 38px;
  outline: 0 !important;
  position: relative;
  transition: all 100ms;
  box-sizing: border-box;
  box-shadow: ${props =>
    props.focused ? `0 0 0 2px ${CALYPSO_MEDIUM}` : 'none'};
  &:hover {
    border-color: hsl(0, 0%, 70%);
  }
`;

const ValueContainer = styled.div`
  align-items: center;
  display: flex;
  flex: 1;
  flex-wrap: wrap;
  padding: 2px 8px;
  position: relative;
  overflow: hidden;
  box-sizing: border-box;
`;

const Placeholder = styled.div`
  color: hsl(0, 0%, 50%);
  margin-left: 2px;
  margin-right: 2px;
  position: absolute;
  top: 50%;
  transform: translateY(-50%);
  box-sizing: border-box;
  font-size: 16px;
`;

const SingleValue = styled.div`
  color: hsl(0, 0%, 20%);
  margin-left: 2px;
  margin-right: 2px;
  max-width: calc(100% - 8px);
  overflow: hidden;
  position: absolute;
  text-overflow: ellipsis;
  white-space: nowrap;
  top: 50%;
  transform: translateY(-50%);
  box-sizing: border-box;
`;

const IndicatorContainer = styled.div`
  align-items: center;
  align-self: stretch;
  display: flex;
  flex-shrink: 0;
  box-sizing: border-box;
`;

const DropdownIndicator = styled.div`
  border-top: 8px solid ${CALYPSO};
  border-left: 6px solid transparent;
  border-right: 6px solid transparent;
  width: 0px;
  height: 0px;
  margin: 10px;
`;

const InputContainer = styled.div`
  margin: 2px;
  padding-bottom: 2px;
  padding-top: 2px;
  visibility: visible;
  color: hsl(0, 0%, 20%);
  box-sizing: border-box;
`;

const Input = styled.input`
  box-sizing: content-box;
  background: rgba(0, 0, 0, 0) none repeat scroll 0px center;
  border: 0px none;
  font-size: inherit;
  opacity: 1;
  outline: currentcolor none 0px;
  padding: 0px;
  color: inherit;
  font-family: inherit;
`;

const InputShadow = styled.div`
  position: absolute;
  opacity: 0;
  font-size: inherit;
`;

const MenuContainer = styled.div`
  position: absolute;
  top: 100%;
  background-color: #fff;
  border-radius: 4px;
  margin-bottom: 8px;
  margin-top: 8px;
  z-index: 9999;
  box-shadow: 0 0 0 1px hsla(0, 0%, 0%, 0.1), 0 4px 11px hsla(0, 0%, 0%, 0.1);
  width: 100%;
`;

const MenuList = styled.div`
  max-height: 300px;
  overflow-y: auto;
  padding-bottom: 4px;
  padding-top: 4px;
  position: relative;
`;

const MenuGroup = styled.div`
  padding-bottom: 8px;
  padding-top: 8px;
`;

const MenuGroupHeader = styled.div`
  color: #999;
  cursor: default;
  font-size: 75%;
  font-weight: 500;
  margin-bottom: 0.25em;
  text-transform: uppercase;
  padding-left: 12px;
  padding-left: 12px;
`;

interface IMenuItemProps {
  selected: boolean;
}

const MenuItem = styled.div<IMenuItemProps>`
  display: block;
  background-color: ${props =>
    props.selected ? CALYPSO_MEDIUM : 'transparent'};
  color: ${props => (props.selected ? '#fff' : 'inherit')};
  cursor: default;
  font-size: inherit;
  width: 100%;
  padding: 8px 12px;
  &:hover {
    background-color: ${props =>
      props.selected ? CALYPSO_MEDIUM : CALYPSO_LIGHT};
  }
`;

interface IAsyncSelectProps {
  placeholder: string;
  value: any;
  loadOptions?: Function;
  defaultOptions?: any[];
  onChange: Function;
}

export default function AsyncSelect({
  placeholder,
  value,
  loadOptions,
  onChange,
  defaultOptions,
}: IAsyncSelectProps) {
  const inputEl = useRef<HTMLInputElement>(null);
  const inputShadowEl = useRef<HTMLDivElement>(null);
  const [isFocused, setFocus] = useState(false);
  const [loadState, setLoadState] = useState<LoadStateType>(
    LoadState.NotLoaded
  );
  const [localValue, setLocalValue] = useState('');
  const [options, setOptions] = useState(defaultOptions);

  const inputSize = `${
    inputShadowEl.current ? inputShadowEl.current.clientWidth + 10 : 2
  }px`;

  useEffect(() => {
    if (loadOptions && loadState === LoadState.NotLoaded) {
      loadOptions('', (result: any) => {
        setOptions(result);
        setLoadState(LoadState.Idle);
      });
    }
  }, [loadOptions, loadState]);

  const renderItems = (items: any[] = [], parentKey?: number) => {
    return items.map((item, index) => {
      if (item.options) {
        return (
          <MenuGroup key={`async-select-item-${index}`}>
            <MenuGroupHeader id={`${index}-heading`}>
              {item.label}
            </MenuGroupHeader>
            <div>{renderItems(item.options, index)}</div>
          </MenuGroup>
        );
      } else {
        const key = `async-select-item-${
          parentKey !== undefined ? `${parentKey}-${index}` : index
        }`;
        return (
          <MenuItem
            key={key}
            id={key}
            selected={value && item.value === value.value}
            onClick={() => {
              onChange(item);
              setFocus(false);
            }}
          >
            {item.label}
          </MenuItem>
        );
      }
    });
  };

  return (
    <Container>
      <ControlContainer
        id="leadin-async-selector"
        focused={isFocused}
        onClick={() => {
          if (isFocused) {
            if (inputEl.current) {
              inputEl.current.blur();
            }
            setFocus(false);
            setLocalValue('');
          } else {
            if (inputEl.current) {
              inputEl.current.focus();
            }
            setFocus(true);
          }
        }}
      >
        <ValueContainer>
          {localValue === '' &&
            (!value ? (
              <Placeholder>{placeholder}</Placeholder>
            ) : (
              <SingleValue>{value.label}</SingleValue>
            ))}
          <InputContainer>
            <Input
              ref={inputEl}
              onFocus={() => {
                setFocus(true);
              }}
              onChange={e => {
                setLocalValue(e.target.value);
                setLoadState(LoadState.Loading);
                loadOptions &&
                  loadOptions(e.target.value, (result: any) => {
                    setOptions(result);
                    setLoadState(LoadState.Idle);
                  });
              }}
              value={localValue}
              width={inputSize}
              id="asycn-select-input"
            />
            <InputShadow ref={inputShadowEl}>{localValue}</InputShadow>
          </InputContainer>
        </ValueContainer>
        <IndicatorContainer>
          {loadState === LoadState.Loading && <UISpinner />}
          <DropdownIndicator />
        </IndicatorContainer>
      </ControlContainer>
      {isFocused && (
        <MenuContainer>
          <MenuList>{renderItems(options)}</MenuList>
        </MenuContainer>
      )}
    </Container>
  );
}
Site is undergoing maintenance

PACJA Events

Maintenance mode is on

Site will be available soon. Thank you for your patience!