-
-
Notifications
You must be signed in to change notification settings - Fork 3.2k
/
Copy pathusePermission.ts
61 lines (49 loc) · 1.51 KB
/
usePermission.ts
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
import { useEffect, useState } from 'react';
import { noop, off, on } from './misc/util';
export type IState = PermissionState | '';
interface IPushPermissionDescriptor extends PermissionDescriptor {
name: 'push';
userVisibleOnly?: boolean;
}
interface IMidiPermissionDescriptor extends PermissionDescriptor {
name: 'midi';
sysex?: boolean;
}
interface IDevicePermissionDescriptor extends PermissionDescriptor {
name: 'camera' | 'microphone' | 'speaker';
deviceId?: string;
}
export type IPermissionDescriptor =
| PermissionDescriptor
| IPushPermissionDescriptor
| IMidiPermissionDescriptor
| IDevicePermissionDescriptor;
// const usePermission = <T extends PermissionDescriptor>(permissionDesc: T): IState => {
const usePermission = (permissionDesc: IPermissionDescriptor): IState => {
const [state, setState] = useState<IState>('');
useEffect(() => {
let mounted = true;
let permissionStatus: PermissionStatus | null = null;
const onChange = () => {
if (!mounted) {
return;
}
setState(() => permissionStatus?.state ?? '');
};
navigator.permissions
.query(permissionDesc)
.then((status) => {
permissionStatus = status;
on(permissionStatus, 'change', onChange);
onChange();
})
.catch(noop);
return () => {
permissionStatus && off(permissionStatus, 'change', onChange);
mounted = false;
permissionStatus = null;
};
}, [permissionDesc]);
return state;
};
export default usePermission;