forked from algolia/docsearch
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathDocSearchButton.tsx
More file actions
110 lines (91 loc) · 3.22 KB
/
DocSearchButton.tsx
File metadata and controls
110 lines (91 loc) · 3.22 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
import React, { type JSX, useEffect, useState } from 'react';
import { ControlKeyIcon } from './icons/ControlKeyIcon';
import { SearchIcon } from './icons/SearchIcon';
export type ButtonTranslations = Partial<{
buttonText: string;
buttonAriaLabel: string;
}>;
export type DocSearchButtonProps = React.ComponentProps<'button'> & {
translations?: ButtonTranslations;
};
const ACTION_KEY_DEFAULT = 'Ctrl' as const;
const ACTION_KEY_APPLE = '⌘' as const;
function isAppleDevice(): boolean {
return /(Mac|iPhone|iPod|iPad)/i.test(navigator.platform);
}
export const DocSearchButton = React.forwardRef<HTMLButtonElement, DocSearchButtonProps>(
({ translations = {}, ...props }, ref) => {
const { buttonText = 'Search', buttonAriaLabel = 'Search' } = translations;
const [key, setKey] = useState<typeof ACTION_KEY_APPLE | typeof ACTION_KEY_DEFAULT | null>(null);
useEffect(() => {
if (typeof navigator !== 'undefined') {
isAppleDevice() ? setKey(ACTION_KEY_APPLE) : setKey(ACTION_KEY_DEFAULT);
}
}, []);
const [actionKeyReactsTo, actionKeyAltText, actionKeyChild] =
key === ACTION_KEY_DEFAULT
? // eslint-disable-next-line react/jsx-key -- false flag
([ACTION_KEY_DEFAULT, 'Ctrl', <ControlKeyIcon />] as const)
: (['Meta', 'Command', key] as const);
return (
<button
type="button"
className="DocSearch DocSearch-Button"
aria-label={`${buttonAriaLabel} (${actionKeyAltText}+K)`}
{...props}
ref={ref}
>
<span className="DocSearch-Button-Container">
<SearchIcon />
<span className="DocSearch-Button-Placeholder">{buttonText}</span>
</span>
<span className="DocSearch-Button-Keys">
{key !== null && (
<>
<DocSearchButtonKey reactsToKey={actionKeyReactsTo}>{actionKeyChild}</DocSearchButtonKey>
<DocSearchButtonKey reactsToKey="k">K</DocSearchButtonKey>
</>
)}
</span>
</button>
);
},
);
type DocSearchButtonKeyProps = {
reactsToKey?: string;
};
function DocSearchButtonKey({ reactsToKey, children }: React.PropsWithChildren<DocSearchButtonKeyProps>): JSX.Element {
const [isKeyDown, setIsKeyDown] = useState(false);
useEffect(() => {
if (!reactsToKey) {
return undefined;
}
function handleKeyDown(e: KeyboardEvent): void {
if (e.key === reactsToKey) {
setIsKeyDown(true);
}
}
function handleKeyUp(e: KeyboardEvent): void {
if (
e.key === reactsToKey ||
// keyup doesn't fire when Command is held down,
// workaround is to mark key as also released when Command is released
// See https://stackoverflow.com/a/73419500
e.key === 'Meta'
) {
setIsKeyDown(false);
}
}
window.addEventListener('keydown', handleKeyDown);
window.addEventListener('keyup', handleKeyUp);
return (): void => {
window.removeEventListener('keydown', handleKeyDown);
window.removeEventListener('keyup', handleKeyUp);
};
}, [reactsToKey]);
return (
<kbd className={isKeyDown ? 'DocSearch-Button-Key DocSearch-Button-Key--pressed' : 'DocSearch-Button-Key'}>
{children}
</kbd>
);
}