vue-vben-admin/src/hooks/event/useKeyPress.ts

165 lines
4.3 KiB
TypeScript
Raw Normal View History

2020-12-24 22:58:26 +08:00
// https://ahooks.js.org/zh-CN/hooks/dom/use-key-press
import type { Ref } from 'vue';
import { onBeforeUnmount, onMounted, unref } from 'vue';
import { noop } from '/@/utils';
import { isFunction, isString, isNumber, isArray } from '/@/utils/is';
export type KeyPredicate = (event: KeyboardEvent) => boolean;
export type keyType = KeyboardEvent['keyCode'] | KeyboardEvent['key'];
export type KeyFilter = keyType | keyType[] | ((event: KeyboardEvent) => boolean);
export type EventHandler = (event: KeyboardEvent) => void;
export type keyEvent = 'keydown' | 'keyup';
export type TargetElement = HTMLElement | Element | Document | Window;
export type Target = Ref<TargetElement>;
export type EventOption = {
events?: keyEvent[];
target?: Target;
};
const defaultEvents: keyEvent[] = ['keydown'];
// 键盘事件 keyCode 别名
2020-12-25 01:09:44 +08:00
const aliasKeyCodeMap: Recordable<number | number[]> = {
2020-12-24 22:58:26 +08:00
esc: 27,
tab: 9,
enter: 13,
space: 32,
up: 38,
left: 37,
right: 39,
down: 40,
delete: [8, 46],
};
// 键盘事件 key 别名
2020-12-25 01:09:44 +08:00
const aliasKeyMap: Recordable<string | string[]> = {
2020-12-24 22:58:26 +08:00
esc: 'Escape',
tab: 'Tab',
enter: 'Enter',
space: ' ',
// IE11 uses key names without `Arrow` prefix for arrow keys.
up: ['Up', 'ArrowUp'],
left: ['Left', 'ArrowLeft'],
right: ['Right', 'ArrowRight'],
down: ['Down', 'ArrowDown'],
delete: ['Backspace', 'Delete'],
};
// 修饰键
2020-12-25 01:09:44 +08:00
const modifierKey: Recordable<(event: KeyboardEvent) => boolean> = {
2020-12-24 22:58:26 +08:00
ctrl: (event: KeyboardEvent) => event.ctrlKey,
shift: (event: KeyboardEvent) => event.shiftKey,
alt: (event: KeyboardEvent) => event.altKey,
meta: (event: KeyboardEvent) => event.metaKey,
};
/**
*
* @param [event: KeyboardEvent]
* @param [keyFilter: any]
* @returns Boolean
*/
function genFilterKey(event: any, keyFilter: any) {
// 浏览器自动补全 input 的时候,会触发 keyDown、keyUp 事件,但此时 event.key 等为空
if (!event.key) {
return false;
}
// 数字类型直接匹配事件的 keyCode
if (isNumber(keyFilter)) {
return event.keyCode === keyFilter;
}
// 字符串依次判断是否有组合键
const genArr = keyFilter.split('.');
let genLen = 0;
for (const key of genArr) {
// 组合键
const genModifier = modifierKey[key];
// key 别名
const aliasKey = aliasKeyMap[key];
// keyCode 别名
const aliasKeyCode = aliasKeyCodeMap[key];
/**
*
* 1.
* 2. key
* 3. keyCode
* 4. key keyCode
*/
if (
(genModifier && genModifier(event)) ||
(aliasKey && isArray(aliasKey) ? aliasKey.includes(event.key) : aliasKey === event.key) ||
(aliasKeyCode && isArray(aliasKeyCode)
? aliasKeyCode.includes(event.keyCode)
: aliasKeyCode === event.keyCode) ||
event.key.toUpperCase() === key.toUpperCase()
) {
genLen++;
}
}
return genLen === genArr.length;
}
/**
*
*/
function genKeyFormat(keyFilter: any): KeyPredicate {
if (isFunction(keyFilter)) {
return keyFilter;
}
if (isString(keyFilter) || isNumber(keyFilter)) {
return (event: KeyboardEvent) => genFilterKey(event, keyFilter);
}
if (isArray(keyFilter)) {
return (event: KeyboardEvent) => keyFilter.some((item: any) => genFilterKey(event, item));
}
return keyFilter ? () => true : () => false;
}
export function useKeyPress(
keyFilter: KeyFilter,
eventHandler: EventHandler = noop,
option: EventOption = {}
) {
const { events = defaultEvents, target } = option;
let el: TargetElement | null | undefined;
function handler(event: any) {
const genGuard: KeyPredicate = genKeyFormat(keyFilter);
if (genGuard(event)) {
return eventHandler(event);
}
}
onMounted(() => {
el = getTargetElement(target, window);
if (!el) return;
for (const eventName of events) {
el.addEventListener(eventName, handler);
}
});
onBeforeUnmount(() => {
if (!el) return;
for (const eventName of events) {
el.removeEventListener(eventName, handler);
}
});
}
export function getTargetElement(
target?: Target,
defaultElement?: TargetElement
): TargetElement | undefined | null {
if (!target) {
return defaultElement;
}
return isFunction(target) ? target() : unref(target);
2020-12-24 22:58:26 +08:00
}