Commit 1e5f1eb8 authored by peterhegman's avatar peterhegman

Prevent duplicate tooltips when hovering over status emoji

Previously the user popover and the status tooltip were shown when
hovering over the user status emoji
parent 9191e87f
...@@ -2,6 +2,7 @@ ...@@ -2,6 +2,7 @@
import { mapActions } from 'vuex'; import { mapActions } from 'vuex';
import timeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue'; import timeAgoTooltip from '~/vue_shared/components/time_ago_tooltip.vue';
import GitlabTeamMemberBadge from '~/vue_shared/components/user_avatar/badges/gitlab_team_member_badge.vue'; import GitlabTeamMemberBadge from '~/vue_shared/components/user_avatar/badges/gitlab_team_member_badge.vue';
import $ from 'jquery';
export default { export default {
components: { components: {
...@@ -45,6 +46,11 @@ export default { ...@@ -45,6 +46,11 @@ export default {
default: true, default: true,
}, },
}, },
data() {
return {
isUsernameLinkHovered: false,
};
},
computed: { computed: {
toggleChevronClass() { toggleChevronClass() {
return this.expanded ? 'fa-chevron-up' : 'fa-chevron-down'; return this.expanded ? 'fa-chevron-up' : 'fa-chevron-down';
...@@ -58,6 +64,46 @@ export default { ...@@ -58,6 +64,46 @@ export default {
showGitlabTeamMemberBadge() { showGitlabTeamMemberBadge() {
return this.author?.is_gitlab_employee; return this.author?.is_gitlab_employee;
}, },
authorLinkClasses() {
return {
hover: this.isUsernameLinkHovered,
'text-underline': this.isUsernameLinkHovered,
'author-name-link': true,
'js-user-link': true,
};
},
authorPath() {
return this.author.path;
},
authorName() {
return this.author.name;
},
authorUsername() {
return this.author.username;
},
authorId() {
return this.author.id;
},
authorStatus() {
return this.author.status_tooltip_html;
},
},
mounted() {
// Temporarily remove `title` attribute from emoji when tooltip is open
// Prevents duplicate tooltips (Bootstrap tooltip and browser title tooltip)
const { authorStatus } = this.$refs;
if (authorStatus && authorStatus.querySelector('.has-tooltip')) {
const emoji = authorStatus.querySelector('gl-emoji');
const emojiTitle = emoji.getAttribute('title');
$(this.$refs.authorStatus).on('show.bs.tooltip', () => {
emoji.removeAttribute('title');
});
$(this.$refs.authorStatus).on('hidden.bs.tooltip', () => {
emoji.setAttribute('title', emojiTitle);
});
}
}, },
methods: { methods: {
...mapActions(['setTargetNoteHash']), ...mapActions(['setTargetNoteHash']),
...@@ -69,6 +115,16 @@ export default { ...@@ -69,6 +115,16 @@ export default {
this.setTargetNoteHash(this.noteTimestampLink); this.setTargetNoteHash(this.noteTimestampLink);
} }
}, },
handleUsernameMouseEnter() {
this.$refs.authorNameLink.dispatchEvent(new Event('mouseenter'));
this.isUsernameLinkHovered = true;
},
handleUsernameMouseLeave() {
this.$refs.authorNameLink.dispatchEvent(new Event('mouseleave'));
this.isUsernameLinkHovered = false;
},
}, },
}; };
</script> </script>
...@@ -87,18 +143,27 @@ export default { ...@@ -87,18 +143,27 @@ export default {
</div> </div>
<template v-if="hasAuthor"> <template v-if="hasAuthor">
<a <a
v-once ref="authorNameLink"
:href="author.path" :href="authorPath"
class="js-user-link" :class="authorLinkClasses"
:data-user-id="author.id" :data-user-id="authorId"
:data-username="author.username" :data-username="authorUsername"
> >
<slot name="note-header-info"></slot> <slot name="note-header-info"></slot>
<span class="note-header-author-name bold">{{ author.name }}</span> <span class="note-header-author-name bold">{{ authorName }}</span>
<span v-if="author.status_tooltip_html" v-html="author.status_tooltip_html"></span>
<span class="note-headline-light">@{{ author.username }}</span>
</a> </a>
<gitlab-team-member-badge v-if="showGitlabTeamMemberBadge" /> <span v-if="authorStatus" ref="authorStatus" v-html="authorStatus"></span>
<span class="text-nowrap author-username">
<a
ref="authorUsernameLink"
class="author-username-link"
:href="authorPath"
@mouseenter="handleUsernameMouseEnter"
@mouseleave="handleUsernameMouseLeave"
><span class="note-headline-light">@{{ authorUsername }}</span>
</a>
<gitlab-team-member-badge v-if="showGitlabTeamMemberBadge" />
</span>
</template> </template>
<span v-else>{{ __('A deleted user') }}</span> <span v-else>{{ __('A deleted user') }}</span>
<span class="note-headline-light note-headline-meta"> <span class="note-headline-light note-headline-meta">
......
...@@ -588,7 +588,8 @@ $note-form-margin-left: 72px; ...@@ -588,7 +588,8 @@ $note-form-margin-left: 72px;
a { a {
color: inherit; color: inherit;
&:hover { &:hover,
&.hover {
color: $blue-600; color: $blue-600;
} }
...@@ -605,6 +606,21 @@ $note-form-margin-left: 72px; ...@@ -605,6 +606,21 @@ $note-form-margin-left: 72px;
.author-link { .author-link {
color: $gl-text-color; color: $gl-text-color;
} }
// Prevent flickering of link when hovering between `author-name-link` and `.author-username-link`
.author-name-link + .author-username .author-username-link {
position: relative;
&::before {
content: '';
position: absolute;
right: 100%;
width: 0.25rem;
height: 100%;
top: 0;
bottom: 0;
}
}
} }
.discussion-header { .discussion-header {
......
---
title: Prevent duplicate tooltips when hovering over status emoji in comments
merge_request: 29356
author:
type: fixed
...@@ -179,4 +179,26 @@ describe('NoteHeader component', () => { ...@@ -179,4 +179,26 @@ describe('NoteHeader component', () => {
expect(findTimestamp().exists()).toBe(true); expect(findTimestamp().exists()).toBe(true);
}); });
}); });
describe('author username link', () => {
it('proxies `mouseenter` event to author name link', () => {
createComponent({ author });
const dispatchEvent = jest.spyOn(wrapper.vm.$refs.authorNameLink, 'dispatchEvent');
wrapper.find({ ref: 'authorUsernameLink' }).trigger('mouseenter');
expect(dispatchEvent).toHaveBeenCalledWith(new Event('mouseenter'));
});
it('proxies `mouseleave` event to author name link', () => {
createComponent({ author });
const dispatchEvent = jest.spyOn(wrapper.vm.$refs.authorNameLink, 'dispatchEvent');
wrapper.find({ ref: 'authorUsernameLink' }).trigger('mouseleave');
expect(dispatchEvent).toHaveBeenCalledWith(new Event('mouseleave'));
});
});
}); });
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment