-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathCopyToClipboard.js
113 lines (98 loc) · 2.36 KB
/
CopyToClipboard.js
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
111
112
113
import React, { useRef, useState } from "react";
import { Button, Classes } from "@blueprintjs/core";
import { VscClippy, VscCheck, VscClose } from "react-icons/vsc";
import copyToClipboard from "clipboard-copy";
import { ButtonLink } from "./ButtonLink.js";
import "./CopyToClipboard.css";
const useCopy = contentProvider => {
const [copyStatus, setCopyStatus] = useState("none");
const timeout = useRef();
const copy = async () => {
const content = await contentProvider();
let success;
try {
await copyToClipboard(content);
success = true;
} catch (ignored) {
success = false;
}
setCopyStatus(success ? "success" : "error");
if (timeout.current) {
clearTimeout(timeout.current);
}
timeout.current = setTimeout(() => {
setCopyStatus("none");
timeout.current = undefined;
}, 1000);
};
return [copy, copyStatus];
};
const CopyStatusInline = ({ copyStatus }) => {
switch (copyStatus) {
case "success":
return (
<span className="CopyStatusInline">
(<VscCheck /> copied)
</span>
);
case "error":
return (
<span className="CopyStatusInline">
(<VscClose /> couldn't copy)
</span>
);
default:
return null;
}
};
export const CopyToClipboardLink = ({
contentProvider,
text = "Copy to clipboard"
}) => {
const [copy, copyStatus] = useCopy(contentProvider);
return (
<>
<ButtonLink onClick={copy}>{text}</ButtonLink>
<CopyStatusInline copyStatus={copyStatus} />
</>
);
};
export const CopyToClipboard = ({
contentProvider,
buttonText = "Copy to clipboard",
buttonProps = { small: true, minimal: true }
}) => {
const [copy, copyStatus] = useCopy(contentProvider);
let iconProps;
switch (copyStatus) {
case "success":
iconProps = {
icon: <VscCheck />,
intent: "success",
text: "Copied"
};
break;
case "error":
iconProps = {
icon: <VscClose />,
intent: "error",
text: "Couldn't copy"
};
break;
default:
iconProps = {
icon: <VscClippy />,
text: buttonText,
title: "Copy to clipboard"
};
break;
}
return (
<Button
{...iconProps}
{...buttonProps}
className={Classes.FIXED + " CopyToClipboard"}
onClick={copy}
/>
);
};