mirror of
https://github.com/coder/coder.git
synced 2025-07-09 11:45:56 +00:00
262 lines
5.8 KiB
TypeScript
262 lines
5.8 KiB
TypeScript
import Link from "@material-ui/core/Link"
|
|
import Popover from "@material-ui/core/Popover"
|
|
import { makeStyles } from "@material-ui/core/styles"
|
|
import HelpIcon from "@material-ui/icons/HelpOutline"
|
|
import OpenInNewIcon from "@material-ui/icons/OpenInNew"
|
|
import React, { createContext, useContext, useRef, useState } from "react"
|
|
import { Stack } from "../../Stack/Stack"
|
|
|
|
type Icon = typeof HelpIcon
|
|
|
|
type Size = "small" | "medium"
|
|
export interface HelpTooltipProps {
|
|
// Useful to test on storybook
|
|
open?: boolean
|
|
size?: Size
|
|
}
|
|
|
|
export const Language = {
|
|
ariaLabel: "tooltip",
|
|
}
|
|
|
|
const HelpTooltipContext = createContext<{ open: boolean; onClose: () => void } | undefined>(
|
|
undefined,
|
|
)
|
|
|
|
const useHelpTooltip = () => {
|
|
const helpTooltipContext = useContext(HelpTooltipContext)
|
|
|
|
if (!helpTooltipContext) {
|
|
throw new Error("This hook should be used in side of the HelpTooltipContext.")
|
|
}
|
|
|
|
return helpTooltipContext
|
|
}
|
|
|
|
export const HelpTooltip: React.FC<React.PropsWithChildren<HelpTooltipProps>> = ({
|
|
children,
|
|
open,
|
|
size = "medium",
|
|
}) => {
|
|
const styles = useStyles({ size })
|
|
const anchorRef = useRef<HTMLButtonElement>(null)
|
|
const [isOpen, setIsOpen] = useState(!!open)
|
|
const id = isOpen ? "help-popover" : undefined
|
|
|
|
const onClose = () => {
|
|
setIsOpen(false)
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<button
|
|
ref={anchorRef}
|
|
aria-describedby={id}
|
|
className={styles.button}
|
|
onClick={(event) => {
|
|
event.stopPropagation()
|
|
setIsOpen(true)
|
|
}}
|
|
onMouseEnter={() => {
|
|
setIsOpen(true)
|
|
}}
|
|
aria-label={Language.ariaLabel}
|
|
>
|
|
<HelpIcon className={styles.icon} />
|
|
</button>
|
|
<Popover
|
|
className={styles.popover}
|
|
classes={{ paper: styles.popoverPaper }}
|
|
id={id}
|
|
open={isOpen}
|
|
anchorEl={anchorRef.current}
|
|
onClose={onClose}
|
|
anchorOrigin={{
|
|
vertical: "bottom",
|
|
horizontal: "left",
|
|
}}
|
|
transformOrigin={{
|
|
vertical: "top",
|
|
horizontal: "left",
|
|
}}
|
|
PaperProps={{
|
|
onMouseEnter: () => {
|
|
setIsOpen(true)
|
|
},
|
|
onMouseLeave: () => {
|
|
setIsOpen(false)
|
|
},
|
|
}}
|
|
>
|
|
<HelpTooltipContext.Provider value={{ open: isOpen, onClose }}>
|
|
{children}
|
|
</HelpTooltipContext.Provider>
|
|
</Popover>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export const HelpTooltipTitle: React.FC<React.PropsWithChildren<unknown>> = ({ children }) => {
|
|
const styles = useStyles()
|
|
|
|
return <h4 className={styles.title}>{children}</h4>
|
|
}
|
|
|
|
export const HelpTooltipText: React.FC<React.PropsWithChildren<unknown>> = ({ children }) => {
|
|
const styles = useStyles()
|
|
|
|
return <p className={styles.text}>{children}</p>
|
|
}
|
|
|
|
export const HelpTooltipLink: React.FC<React.PropsWithChildren<{ href: string }>> = ({
|
|
children,
|
|
href,
|
|
}) => {
|
|
const styles = useStyles()
|
|
|
|
return (
|
|
<Link href={href} target="_blank" rel="noreferrer" className={styles.link}>
|
|
<OpenInNewIcon className={styles.linkIcon} />
|
|
{children}
|
|
</Link>
|
|
)
|
|
}
|
|
|
|
export const HelpTooltipAction: React.FC<
|
|
React.PropsWithChildren<{
|
|
icon: Icon
|
|
onClick: () => void
|
|
ariaLabel?: string
|
|
}>
|
|
> = ({ children, icon: Icon, onClick, ariaLabel }) => {
|
|
const styles = useStyles()
|
|
const tooltip = useHelpTooltip()
|
|
|
|
return (
|
|
<button
|
|
aria-label={ariaLabel ?? ""}
|
|
className={styles.action}
|
|
onClick={(event) => {
|
|
event.stopPropagation()
|
|
onClick()
|
|
tooltip.onClose()
|
|
}}
|
|
>
|
|
<Icon className={styles.actionIcon} />
|
|
{children}
|
|
</button>
|
|
)
|
|
}
|
|
|
|
export const HelpTooltipLinksGroup: React.FC<React.PropsWithChildren<unknown>> = ({ children }) => {
|
|
const styles = useStyles()
|
|
|
|
return (
|
|
<Stack spacing={1} className={styles.linksGroup}>
|
|
{children}
|
|
</Stack>
|
|
)
|
|
}
|
|
|
|
const getButtonSpacingFromSize = (size?: Size): number => {
|
|
switch (size) {
|
|
case "small":
|
|
return 2.75
|
|
case "medium":
|
|
default:
|
|
return 3
|
|
}
|
|
}
|
|
|
|
const getIconSpacingFromSize = (size?: Size): number => {
|
|
switch (size) {
|
|
case "small":
|
|
return 1.75
|
|
case "medium":
|
|
default:
|
|
return 2
|
|
}
|
|
}
|
|
|
|
const useStyles = makeStyles((theme) => ({
|
|
button: {
|
|
display: "flex",
|
|
alignItems: "center",
|
|
justifyContent: "center",
|
|
width: ({ size }: { size?: Size }) => theme.spacing(getButtonSpacingFromSize(size)),
|
|
height: ({ size }: { size?: Size }) => theme.spacing(getButtonSpacingFromSize(size)),
|
|
padding: 0,
|
|
border: 0,
|
|
background: "transparent",
|
|
color: theme.palette.text.primary,
|
|
opacity: 0.5,
|
|
cursor: "pointer",
|
|
|
|
"&:hover": {
|
|
opacity: 0.75,
|
|
},
|
|
},
|
|
|
|
icon: {
|
|
width: ({ size }: { size?: Size }) => theme.spacing(getIconSpacingFromSize(size)),
|
|
height: ({ size }: { size?: Size }) => theme.spacing(getIconSpacingFromSize(size)),
|
|
},
|
|
|
|
popover: {
|
|
pointerEvents: "none",
|
|
},
|
|
|
|
popoverPaper: {
|
|
marginTop: theme.spacing(0.5),
|
|
width: theme.spacing(38),
|
|
padding: theme.spacing(2.5),
|
|
color: theme.palette.text.secondary,
|
|
pointerEvents: "auto",
|
|
},
|
|
|
|
title: {
|
|
marginTop: 0,
|
|
marginBottom: theme.spacing(1),
|
|
color: theme.palette.text.primary,
|
|
},
|
|
|
|
text: {
|
|
marginTop: theme.spacing(0.5),
|
|
marginBottom: theme.spacing(0.5),
|
|
},
|
|
|
|
link: {
|
|
display: "flex",
|
|
alignItems: "center",
|
|
},
|
|
|
|
linkIcon: {
|
|
color: "inherit",
|
|
width: 14,
|
|
height: 14,
|
|
marginRight: theme.spacing(1),
|
|
},
|
|
|
|
linksGroup: {
|
|
marginTop: theme.spacing(2),
|
|
},
|
|
|
|
action: {
|
|
display: "flex",
|
|
alignItems: "center",
|
|
background: "none",
|
|
border: 0,
|
|
color: theme.palette.primary.light,
|
|
padding: 0,
|
|
cursor: "pointer",
|
|
fontSize: 14,
|
|
},
|
|
|
|
actionIcon: {
|
|
color: "inherit",
|
|
width: 14,
|
|
height: 14,
|
|
marginRight: theme.spacing(1),
|
|
},
|
|
}))
|