debian-mirror-gitlab/app/assets/javascripts/tooltips/components/tooltips.vue

122 lines
2.8 KiB
Vue
Raw Normal View History

2020-11-24 15:15:51 +05:30
<script>
import { GlTooltip, GlSafeHtmlDirective as SafeHtml } from '@gitlab/ui';
import { uniqueId } from 'lodash';
2021-03-08 18:12:59 +05:30
const getTooltipTitle = (element) => {
2020-11-24 15:15:51 +05:30
return element.getAttribute('title') || element.dataset.title;
};
const newTooltip = (element, config = {}) => {
const { placement, container, boundary, html, triggers } = element.dataset;
const title = getTooltipTitle(element);
return {
id: uniqueId('gl-tooltip'),
target: element,
title,
html,
placement,
container,
boundary,
triggers,
disabled: !title,
...config,
};
};
export default {
components: {
GlTooltip,
},
directives: {
SafeHtml,
},
data() {
return {
tooltips: [],
};
},
created() {
2021-03-08 18:12:59 +05:30
this.observer = new MutationObserver((mutations) => {
mutations.forEach((mutation) => {
2020-11-24 15:15:51 +05:30
mutation.removedNodes.forEach(this.dispose);
});
});
},
beforeDestroy() {
this.observer.disconnect();
},
methods: {
addTooltips(elements, config) {
const newTooltips = elements
2021-03-08 18:12:59 +05:30
.filter((element) => !this.tooltipExists(element))
.map((element) => newTooltip(element, config))
.filter((tooltip) => tooltip.title);
2020-11-24 15:15:51 +05:30
2021-03-08 18:12:59 +05:30
newTooltips.forEach((tooltip) => this.observe(tooltip));
2020-11-24 15:15:51 +05:30
this.tooltips.push(...newTooltips);
},
observe(tooltip) {
this.observer.observe(tooltip.target.parentElement, {
childList: true,
});
},
dispose(target) {
if (!target) {
this.tooltips = [];
} else {
const index = this.tooltips.indexOf(this.findTooltipByTarget(target));
if (index > -1) {
this.tooltips.splice(index, 1);
}
}
},
fixTitle(target) {
const tooltip = this.findTooltipByTarget(target);
if (tooltip) {
tooltip.title = target.getAttribute('title');
}
},
triggerEvent(target, event) {
const tooltip = this.findTooltipByTarget(target);
if (tooltip) {
this.$refs[tooltip.id][0].$emit(event);
}
},
tooltipExists(element) {
return Boolean(this.findTooltipByTarget(element));
},
findTooltipByTarget(element) {
2021-03-08 18:12:59 +05:30
return this.tooltips.find((tooltip) => tooltip.target === element);
2020-11-24 15:15:51 +05:30
},
},
2021-03-08 18:12:59 +05:30
safeHtmlConfig: {
ADD_TAGS: ['gl-emoji'],
},
2020-11-24 15:15:51 +05:30
};
</script>
<template>
<div>
<gl-tooltip
v-for="(tooltip, index) in tooltips"
:id="tooltip.id"
:ref="tooltip.id"
:key="index"
:target="tooltip.target"
:triggers="tooltip.triggers"
:placement="tooltip.placement"
:container="tooltip.container"
:boundary="tooltip.boundary"
:disabled="tooltip.disabled"
2021-01-29 00:20:46 +05:30
:show="tooltip.show"
2020-11-24 15:15:51 +05:30
>
2021-03-08 18:12:59 +05:30
<span v-if="tooltip.html" v-safe-html:[$options.safeHtmlConfig]="tooltip.title"></span>
2020-11-24 15:15:51 +05:30
<span v-else>{{ tooltip.title }}</span>
</gl-tooltip>
</div>
</template>