import type {AnchorAlignment, AnchorSide} from '@primer/behaviors'
import {getAnchoredPosition} from '@primer/behaviors'
const TOOLTIP_OPEN_CLASS = 'tooltip-open'
const TOOLTIP_ARROW_EDGE_OFFSET = 6
const TOOLTIP_SR_ONLY_CLASS = 'sr-only'
type Direction = 'n' | 's' | 'e' | 'w' | 'ne' | 'se' | 'nw' | 'sw'
const DIRECTION_CLASSES = [
'tooltip-n',
'tooltip-s',
'tooltip-e',
'tooltip-w',
'tooltip-ne',
'tooltip-se',
'tooltip-nw',
'tooltip-sw'
]
class ToolTipElement extends HTMLElement {
styles() {
return `
:host {
position: absolute;
z-index: 1000000;
padding: .5em .75em;
font: normal normal 11px/1.5 -apple-system, BlinkMacSystemFont, "Segoe UI", Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji";
-webkit-font-smoothing: subpixel-antialiased;
color: var(--color-fg-on-emphasis);
text-align: center;
text-decoration: none;
text-shadow: none;
text-transform: none;
letter-spacing: normal;
word-wrap: break-word;
white-space: pre;
background: var(--color-neutral-emphasis-plus);
border-radius: 6px;
opacity: 0;
max-width: 250px;
word-wrap: break-word;
white-space: normal;
width: max-content;
}
:host:before{
position: absolute;
z-index: 1000001;
color: var(--color-neutral-emphasis-plus);
content: "";
border: 6px solid transparent;
opacity: 0
}
@keyframes tooltip-appear {
from {
opacity: 0
}
to {
opacity: 1
}
}
:host:after{
position: absolute;
display: block;
right: 0;
left: 0;
height: 12px;
content: ""
}
:host(.${TOOLTIP_OPEN_CLASS}),
:host(.${TOOLTIP_OPEN_CLASS}):before {
animation-name: tooltip-appear;
animation-duration: .1s;
animation-fill-mode: forwards;
animation-timing-function: ease-in;
animation-delay: .4s
}
:host(.tooltip-s):before,
:host(.tooltip-n):before {
right: 50%;
margin-right: -${TOOLTIP_ARROW_EDGE_OFFSET}px;
}
:host(.tooltip-s):before,
:host(.tooltip-se):before,
:host(.tooltip-sw):before {
bottom: 100%;
border-bottom-color: var(--color-neutral-emphasis-plus)
}
:host(.tooltip-s):after,
:host(.tooltip-se):after,
:host(.tooltip-sw):after {
bottom: 100%
}
:host(.tooltip-n):before,
:host(.tooltip-ne):before,
:host(.tooltip-nw):before {
top: 100%;
border-top-color: var(--color-neutral-emphasis-plus)
}
:host(.tooltip-n):after,
:host(.tooltip-ne):after,
:host(.tooltip-nw):after {
top: 100%
}
:host(.tooltip-se):before,
:host(.tooltip-ne):before {
left: 0;
margin-left: ${TOOLTIP_ARROW_EDGE_OFFSET}px;
}
:host(.tooltip-sw):before,
:host(.tooltip-nw):before {
right: 0;
margin-right: ${TOOLTIP_ARROW_EDGE_OFFSET}px;
}
:host(.tooltip-w):before {
top: 50%;
bottom: 50%;
left: 100%;
margin-top: -6px;
border-left-color: var(--color-neutral-emphasis-plus)
}
:host(.tooltip-e):before {
top: 50%;
right: 100%;
bottom: 50%;
margin-top: -6px;
border-right-color: var(--color-neutral-emphasis-plus)
}
`
}
#abortController: AbortController | undefined
#align: AnchorAlignment = 'center'
#side: AnchorSide = 'outside-bottom'
#allowUpdatePosition = false
get htmlFor(): string {
return this.getAttribute('for') || ''
}
set htmlFor(value: string) {
this.setAttribute('for', value)
}
get type(): 'description' | 'label' {
const type = this.getAttribute('data-type')
return type === 'label' ? 'label' : 'description'
}
set type(value: 'description' | 'label') {
this.setAttribute('data-type', value)
}
get direction(): Direction {
return (this.getAttribute('data-direction') || 's') as Direction
}
set direction(value: Direction) {
this.setAttribute('data-direction', value)
}
get control(): HTMLElement | null {
return this.ownerDocument.getElementById(this.htmlFor)
}
set hiddenFromView(value: true | false) {
this.classList.toggle(TOOLTIP_SR_ONLY_CLASS, value)
if (this.isConnected) this.#update()
}
get hiddenFromView() {
return this.classList.contains(TOOLTIP_SR_ONLY_CLASS)
}
connectedCallback() {
if (!this.shadowRoot) {
const shadow = this.attachShadow({mode: 'open'})
// eslint-disable-next-line github/no-inner-html
shadow.innerHTML = `
`
}
this.hiddenFromView = true
this.#allowUpdatePosition = true
if (!this.id) {
this.id = `tooltip-${Date.now()}-${(Math.random() * 10000).toFixed(0)}`
}
if (!this.control) return
this.setAttribute('role', 'tooltip')
this.#abortController?.abort()
this.#abortController = new AbortController()
const {signal} = this.#abortController
this.addEventListener('mouseleave', this, {signal})
this.control.addEventListener('mouseenter', this, {signal})
this.control.addEventListener('mouseleave', this, {signal})
this.control.addEventListener('focus', this, {signal})
this.control.addEventListener('blur', this, {signal})
this.ownerDocument.addEventListener('keydown', this, {signal})
this.#update()
}
disconnectedCallback() {
this.#abortController?.abort()
}
handleEvent(event: Event) {
if (!this.control) return
// Ensures that tooltip stays open when hovering between tooltip and element
// WCAG Success Criterion 1.4.13 Hoverable
if ((event.type === 'mouseenter' || event.type === 'focus') && this.hiddenFromView) {
this.hiddenFromView = false
} else if (event.type === 'blur') {
this.hiddenFromView = true
} else if (
event.type === 'mouseleave' &&
(event as MouseEvent).relatedTarget !== this.control &&
(event as MouseEvent).relatedTarget !== this
) {
this.hiddenFromView = true
} else if (event.type === 'keydown' && (event as KeyboardEvent).key === 'Escape' && !this.hiddenFromView) {
this.hiddenFromView = true
}
}
static observedAttributes = ['data-type', 'data-direction', 'id']
#update() {
if (this.hiddenFromView) {
this.classList.remove(TOOLTIP_OPEN_CLASS, ...DIRECTION_CLASSES)
} else {
this.classList.add(TOOLTIP_OPEN_CLASS)
for (const tooltip of this.ownerDocument.querySelectorAll(this.tagName)) {
if (tooltip !== this) tooltip.hiddenFromView = true
}
this.#updatePosition()
}
}
attributeChangedCallback(name: string) {
if (name === 'id' || name === 'data-type') {
if (!this.id || !this.control) return
if (this.type === 'label') {
let labelledBy = this.control.getAttribute('aria-labelledby')
if (labelledBy) {
labelledBy = `${labelledBy} ${this.id}`
} else {
labelledBy = this.id
}
this.control.setAttribute('aria-labelledby', labelledBy)
// Prevent duplicate accessible name announcements.
this.setAttribute('aria-hidden', 'true')
} else {
let describedBy = this.control.getAttribute('aria-describedby')
describedBy ? (describedBy = `${describedBy} ${this.id}`) : (describedBy = this.id)
this.control.setAttribute('aria-describedby', describedBy)
}
} else if (name === 'data-direction') {
this.classList.remove(...DIRECTION_CLASSES)
const direction = this.direction
if (direction === 'n') {
this.#align = 'center'
this.#side = 'outside-top'
} else if (direction === 'ne') {
this.#align = 'start'
this.#side = 'outside-top'
} else if (direction === 'e') {
this.#align = 'center'
this.#side = 'outside-right'
} else if (direction === 'se') {
this.#align = 'start'
this.#side = 'outside-bottom'
} else if (direction === 's') {
this.#align = 'center'
this.#side = 'outside-bottom'
} else if (direction === 'sw') {
this.#align = 'end'
this.#side = 'outside-bottom'
} else if (direction === 'w') {
this.#align = 'center'
this.#side = 'outside-left'
} else if (direction === 'nw') {
this.#align = 'end'
this.#side = 'outside-top'
}
}
}
#updatePosition() {
if (!this.control) return
if (!this.#allowUpdatePosition || this.hiddenFromView) return
const TOOLTIP_OFFSET = 10
this.style.left = `0px` // Ensures we have reliable tooltip width in `getAnchoredPosition`
const position = getAnchoredPosition(this, this.control, {
side: this.#side,
align: this.#align,
anchorOffset: TOOLTIP_OFFSET
})
const anchorSide = position.anchorSide
const align = position.anchorAlign
this.style.top = `${position.top}px`
this.style.left = `${position.left}px`
let direction: Direction = 's'
if (anchorSide === 'outside-left') {
direction = 'w'
} else if (anchorSide === 'outside-right') {
direction = 'e'
} else if (anchorSide === 'outside-top') {
if (align === 'center') {
direction = 'n'
} else if (align === 'start') {
direction = 'ne'
} else {
direction = 'nw'
}
} else {
if (align === 'center') {
direction = 's'
} else if (align === 'start') {
direction = 'se'
} else {
direction = 'sw'
}
}
this.classList.add(`tooltip-${direction}`)
}
}
if (!window.customElements.get('tool-tip')) {
window.ToolTipElement = ToolTipElement
window.customElements.define('tool-tip', ToolTipElement)
}
declare global {
interface Window {
ToolTipElement: typeof ToolTipElement
}
}