2020-08-26 11:10:29 -04:00
|
|
|
import Vue from 'vue';
|
2020-09-16 20:09:34 -04:00
|
|
|
import jQuery from 'jquery';
|
2020-10-14 11:08:42 -04:00
|
|
|
import { toArray, isFunction, isElement } from 'lodash';
|
2020-08-26 11:10:29 -04:00
|
|
|
import Tooltips from './components/tooltips.vue';
|
|
|
|
|
|
|
|
let app;
|
|
|
|
|
|
|
|
const EVENTS_MAP = {
|
|
|
|
hover: 'mouseenter',
|
|
|
|
click: 'click',
|
|
|
|
focus: 'focus',
|
|
|
|
};
|
|
|
|
|
|
|
|
const DEFAULT_TRIGGER = 'hover focus';
|
2020-08-30 23:10:13 -04:00
|
|
|
const APP_ELEMENT_ID = 'gl-tooltips-app';
|
2020-08-26 11:10:29 -04:00
|
|
|
|
|
|
|
const tooltipsApp = () => {
|
|
|
|
if (!app) {
|
2020-08-30 23:10:13 -04:00
|
|
|
const container = document.createElement('div');
|
|
|
|
|
|
|
|
container.setAttribute('id', APP_ELEMENT_ID);
|
|
|
|
document.body.appendChild(container);
|
|
|
|
|
2020-08-26 11:10:29 -04:00
|
|
|
app = new Vue({
|
|
|
|
render(h) {
|
|
|
|
return h(Tooltips, {
|
|
|
|
props: {
|
|
|
|
elements: this.elements,
|
|
|
|
},
|
|
|
|
ref: 'tooltips',
|
|
|
|
});
|
|
|
|
},
|
2020-08-30 23:10:13 -04:00
|
|
|
}).$mount(container);
|
2020-08-26 11:10:29 -04:00
|
|
|
}
|
|
|
|
|
2020-09-11 05:08:44 -04:00
|
|
|
return app.$refs.tooltips;
|
2020-08-26 11:10:29 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
const isTooltip = (node, selector) => node.matches && node.matches(selector);
|
|
|
|
|
|
|
|
const addTooltips = (elements, config) => {
|
2020-09-11 05:08:44 -04:00
|
|
|
tooltipsApp().addTooltips(toArray(elements), config);
|
2020-08-26 11:10:29 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
const handleTooltipEvent = (rootTarget, e, selector, config = {}) => {
|
|
|
|
for (let { target } = e; target && target !== rootTarget; target = target.parentNode) {
|
|
|
|
if (isTooltip(target, selector)) {
|
|
|
|
addTooltips([target], {
|
|
|
|
show: true,
|
|
|
|
...config,
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-10-14 11:08:42 -04:00
|
|
|
const applyToElements = (elements, handler) => {
|
|
|
|
const iterable = isElement(elements) ? [elements] : toArray(elements);
|
|
|
|
|
|
|
|
toArray(iterable).forEach(handler);
|
|
|
|
};
|
2020-08-26 11:10:29 -04:00
|
|
|
|
2020-09-16 20:09:34 -04:00
|
|
|
const invokeBootstrapApi = (elements, method) => {
|
|
|
|
if (isFunction(elements.tooltip)) {
|
2020-10-01 08:10:14 -04:00
|
|
|
elements.tooltip(method);
|
|
|
|
} else {
|
2020-09-16 20:09:34 -04:00
|
|
|
jQuery(elements).tooltip(method);
|
|
|
|
}
|
|
|
|
};
|
2020-08-26 11:10:29 -04:00
|
|
|
|
2020-12-23 10:09:54 -05:00
|
|
|
const isGlTooltipsEnabled = () => Boolean(window.gon.features?.glTooltips);
|
2020-09-16 20:09:34 -04:00
|
|
|
|
|
|
|
const tooltipApiInvoker = ({ glHandler, bsHandler }) => (elements, ...params) => {
|
|
|
|
if (isGlTooltipsEnabled()) {
|
|
|
|
applyToElements(elements, glHandler);
|
|
|
|
} else {
|
|
|
|
bsHandler(elements, ...params);
|
|
|
|
}
|
2020-08-26 11:10:29 -04:00
|
|
|
};
|
2020-08-30 23:10:13 -04:00
|
|
|
|
2020-09-16 20:09:34 -04:00
|
|
|
export const initTooltips = (config = {}) => {
|
|
|
|
if (isGlTooltipsEnabled()) {
|
|
|
|
const triggers = config?.triggers || DEFAULT_TRIGGER;
|
2020-12-23 16:10:24 -05:00
|
|
|
const events = triggers.split(' ').map((trigger) => EVENTS_MAP[trigger]);
|
2020-09-11 05:08:44 -04:00
|
|
|
|
2020-12-23 16:10:24 -05:00
|
|
|
events.forEach((event) => {
|
2020-09-16 20:09:34 -04:00
|
|
|
document.addEventListener(
|
|
|
|
event,
|
2020-12-23 16:10:24 -05:00
|
|
|
(e) => handleTooltipEvent(document, e, config.selector, config),
|
2020-09-16 20:09:34 -04:00
|
|
|
true,
|
|
|
|
);
|
|
|
|
});
|
2020-08-30 23:10:13 -04:00
|
|
|
|
2020-09-16 20:09:34 -04:00
|
|
|
return tooltipsApp();
|
|
|
|
}
|
|
|
|
|
|
|
|
return invokeBootstrapApi(document.body, config);
|
|
|
|
};
|
2020-10-21 17:09:00 -04:00
|
|
|
export const add = (elements, config = {}) => {
|
|
|
|
if (isGlTooltipsEnabled()) {
|
|
|
|
return addTooltips(elements, config);
|
|
|
|
}
|
|
|
|
return invokeBootstrapApi(elements, config);
|
|
|
|
};
|
2020-09-16 20:09:34 -04:00
|
|
|
export const dispose = tooltipApiInvoker({
|
2020-12-23 16:10:24 -05:00
|
|
|
glHandler: (element) => tooltipsApp().dispose(element),
|
|
|
|
bsHandler: (elements) => invokeBootstrapApi(elements, 'dispose'),
|
2020-09-16 20:09:34 -04:00
|
|
|
});
|
|
|
|
export const fixTitle = tooltipApiInvoker({
|
2020-12-23 16:10:24 -05:00
|
|
|
glHandler: (element) => tooltipsApp().fixTitle(element),
|
|
|
|
bsHandler: (elements) => invokeBootstrapApi(elements, '_fixTitle'),
|
2020-09-16 20:09:34 -04:00
|
|
|
});
|
|
|
|
export const enable = tooltipApiInvoker({
|
2020-12-23 16:10:24 -05:00
|
|
|
glHandler: (element) => tooltipsApp().triggerEvent(element, 'enable'),
|
|
|
|
bsHandler: (elements) => invokeBootstrapApi(elements, 'enable'),
|
2020-09-16 20:09:34 -04:00
|
|
|
});
|
|
|
|
export const disable = tooltipApiInvoker({
|
2020-12-23 16:10:24 -05:00
|
|
|
glHandler: (element) => tooltipsApp().triggerEvent(element, 'disable'),
|
|
|
|
bsHandler: (elements) => invokeBootstrapApi(elements, 'disable'),
|
2020-09-16 20:09:34 -04:00
|
|
|
});
|
|
|
|
export const hide = tooltipApiInvoker({
|
2020-12-23 16:10:24 -05:00
|
|
|
glHandler: (element) => tooltipsApp().triggerEvent(element, 'close'),
|
|
|
|
bsHandler: (elements) => invokeBootstrapApi(elements, 'hide'),
|
2020-09-16 20:09:34 -04:00
|
|
|
});
|
|
|
|
export const show = tooltipApiInvoker({
|
2020-12-23 16:10:24 -05:00
|
|
|
glHandler: (element) => tooltipsApp().triggerEvent(element, 'open'),
|
|
|
|
bsHandler: (elements) => invokeBootstrapApi(elements, 'show'),
|
2020-09-16 20:09:34 -04:00
|
|
|
});
|
2020-08-30 23:10:13 -04:00
|
|
|
export const destroy = () => {
|
|
|
|
tooltipsApp().$destroy();
|
|
|
|
app = null;
|
|
|
|
};
|