JFIFxxC      C  " }!1AQa"q2#BR$3br %&'()*456789:CDEFGHIJSTUVWXYZcdefghijstuvwxyz w!1AQaq"2B #3Rbr{ gilour

File "use-keybind.ts"

Full Path: /home/markqprx/iniasli.pro/resources/client/utils/keybinds/use-keybind.ts
File size: 1.53 KB
MIME-type: text/plain
Charset: utf-8

import {useGlobalListeners} from '@react-aria/utils';
import {useCallbackRef} from '@common/utils/hooks/use-callback-ref';
import {useEffect} from 'react';
import {isCtrlKeyPressed} from '@common/utils/keybinds/is-ctrl-key-pressed';
import {isAnyInputFocused} from '@common/utils/dom/is-any-input-focused';

interface Options {
  allowedInputSelector?: string;
}

export function useKeybind(
  el: HTMLElement | 'window',
  shortcut: string,
  userCallback: (e: KeyboardEvent) => void,
  {allowedInputSelector}: Options = {},
) {
  const {addGlobalListener, removeAllGlobalListeners} = useGlobalListeners();
  const callback = useCallbackRef(userCallback);

  useEffect(() => {
    const target = el === 'window' ? window : el;
    addGlobalListener(target, 'keydown', (e: KeyboardEvent) => {
      if (!shouldIgnoreActiveEl(allowedInputSelector) && isAnyInputFocused()) {
        return;
      }
      const matches = shortcut.split('+').every(key => {
        if (key === 'ctrl') {
          return isCtrlKeyPressed(e);
        } else {
          return e.key === key;
        }
      });
      if (matches) {
        e.preventDefault();
        e.stopPropagation();
        callback(e);
      }
    });
    return removeAllGlobalListeners;
  }, [
    addGlobalListener,
    shortcut,
    removeAllGlobalListeners,
    callback,
    el,
    allowedInputSelector,
  ]);
}

function shouldIgnoreActiveEl(selector?: string) {
  if (!selector || !document.activeElement) {
    return false;
  }
  return (document.activeElement as HTMLElement).closest(selector);
}