Skip to content

Latest commit

 

History

History
248 lines (194 loc) · 5.92 KB

File metadata and controls

248 lines (194 loc) · 5.92 KB

@straw-hat/react-hooks

A collection of React hooks.

Usage

  • Reference guides Please use yarn docs:reference to generate the reference docs.

useIsomorphicLayoutEffect

React currently throws a warning when using useLayoutEffect on the server. To get around it, we can conditionally useEffect on the server (no-op) and useLayoutEffect in the browser.

import { useIsomorphicLayoutEffect } from '@straw-hat/react-hooks';

function MyComponent() {
  useIsomorphicLayoutEffect(() => {
    // ...
  }, deps);
}

useControlled

Used to create controlled values, useful for building controlled Form inputs.

import CheckIcon from '@material-ui/icons/Check';
import * as React from 'react';
import { FormState } from 'react-hook-form';

import { useControlled } from '@straw-hat/react-hooks';

interface CheckboxInputProps {
  label: string;
  name: string;
  checked?: boolean;
  defaultChecked?: boolean;
  formState?: FormState<any>;
  onChange?: React.ChangeEventHandler<any>;
}

export const CheckboxInput = React.forwardRef<HTMLInputElement, CheckboxInputProps>((props, ref) => {
  const [checked, setCheckedState] = useControlled({
    controlled: props.checked,
    default: Boolean(props.defaultChecked),
  });

  function onChange(event) {
    const newChecked = event.target.checked;
    setCheckedState(newChecked);
    props.onChange?.(event);
  }

  return (
    <label className="flex items-center cursor-pointer">
      <input
        className="hidden leading-tight"
        type="checkbox"
        name={props.name}
        checked={props.checked}
        defaultChecked={props.defaultChecked}
        ref={ref}
        onChange={onChange}
      />
      <div className="w-4 h-4 p-3  bg-red-700 rounded-md flex items-center justify-center">
        {checked && <CheckIcon className="text-white" color="inherit" fontSize="inherit" />}
      </div>
      <span className="ml-3 text-sm md:text-base font-semibold text-gray-600">{props.label}</span>
    </label>
  );
});

useScript

Used to inject script tags.

import { useScript } from '@straw-hat/react-hooks';
import * as React from 'react';

function MyComponent() {
  const { status, isReady } = useScript({ src: 'https://bit.ly/3se7YYw' });

  React.useEffect(() => {
    if (isReady) {
      // Do stuff from the script if you need to
    }
  }, [status]);

  return <>...</>;
}

usePrevious

React state hook that returns the previous state as described in the React hooks FAQ.

import { usePrevious } from '@straw-hat/react-hooks';
import * as React from 'react';

function MyComponent() {
  // State value and setter for our example
  const [count, setCount] = useState<number>(0);
  // Get the previous value (was passed into hook on last render)
  const prevCount: number = usePrevious<number>(count);
  // Display both current and previous count value
  return (
    <div>
      <h1>
        Now: {count}, before: {prevCount}
      </h1>
      <button onClick={() => setCount(count + 1)}>Increment</button>
    </div>
  );
}

useLatest

React state hook that returns the latest state as described in the React hooks FAQ.

This is mostly useful to get access to the latest value of some props or state inside an asynchronous callback, instead of that value at the time the callback was created from.

import { useLatest } from '@straw-hat/react-hooks';
import * as React from 'react';

function MyComponent() {
  const [count, setCount] = React.useState(0);
  const latestCount = useLatest(count);

  function handleAlertClick() {
    setTimeout(() => {
      console.log('You clicked on (Count): ', count);
      console.log('You clicked on: (Latest Count)', latestCount.current);
    }, 3000);
  }

  return (
    <div>
      <p>You clicked {count} times</p>
      <button onClick={() => setCount(count + 1)}>Click me</button>
      <button onClick={handleAlertClick}>Show alert</button>
    </div>
  );
}

useEventListener

Register an event handler.

import { useEventListener } from '@straw-hat/react-hooks';
import * as React from 'react';

function MyComponent() {
  useEventListener({
    element: window,
    type: 'resize',
    listener() {
      console.log('resizing');
    },
  });

  return <>...</>;
}

createContextHook

Creates a Context from a regular hook, following a particular boilerplate pattern. It will return the "context hook" and the Provider component.

// use-my-hook.ts
import { createContextHook } from '@straw-hat/react-hooks';
import * as React from 'react';

// 1. Create your hook
type UseMyHookProps = {
  initialValue: number;
};

function useMyHook(props: UseMyHookProps) {
  const [value, setValue] = React.useState(props.initialValue);
  return { value, setValue };
}

// 2. Create the Context Hook and Provider

const [MyHookProvider, useMyHookContext] = createContextHook<
  // the return value of the hook
  ReturnType<typeof useMyHook>,
  // the hook input parameter
  UseMyHookProps
>(
  // Pass your hook reference
  useMyHook,
  {
    // `name` is used for debugging propose when you failed to connect the hook
    name: 'MyHook',
  },
);

export { MyHookProvider, useMyHookContext };

useCallOnce

Calls the function callback only once.

import { useCallOnce } from '@straw-hat/react-hooks';
import * as React from 'react';

function MyComponent() {
  const onSuccess = useCallOnce(() => console.count('Only Runs Once'));
  const query = useQuery({
    onSuccess: onSuccess,
  });

  return <>...</>;
}

useBoolean

Tracks state of a boolean value.

import { useBoolean } from '@straw-hat/react-hooks';
import * as React from 'react';

function MyComponent() {
  const { value, toggle, setTrue, setFalse } = useBoolean(true);
  return <button onClick={toggle}>...</button>;
}