Commit 78cd69f9 authored by Thomas Randolph's avatar Thomas Randolph

Fix slight scroll bounce when jumping between unresolved discussions

Changelog: fixed
parent b829ab9f
import { mapGetters, mapActions, mapState } from 'vuex'; import { mapGetters, mapActions, mapState } from 'vuex';
import { scrollToElementWithContext, scrollToElement } from '~/lib/utils/common_utils'; import { scrollToElementWithContext, scrollToElement } from '~/lib/utils/common_utils';
import { updateHistory } from '../../lib/utils/url_utility';
import eventHub from '../event_hub'; import eventHub from '../event_hub';
const isDiffsVirtualScrollingEnabled = () => window.gon?.features?.diffsVirtualScrolling; const isDiffsVirtualScrollingEnabled = () => window.gon?.features?.diffsVirtualScrolling;
...@@ -22,13 +23,43 @@ function scrollTo(selector, { withoutContext = false } = {}) { ...@@ -22,13 +23,43 @@ function scrollTo(selector, { withoutContext = false } = {}) {
return false; return false;
} }
function updateUrlWithNoteId(noteId) {
const newHistoryEntry = {
state: null,
title: window.title,
url: `#note_${noteId}`,
replace: true,
};
if (noteId && isDiffsVirtualScrollingEnabled()) {
// Temporarily mask the ID to avoid the browser default
// scrolling taking over which is broken with virtual
// scrolling enabled.
const note = document.querySelector(`#note_${noteId}`);
note?.setAttribute('id', `masked::${note.id}`);
// Update the hash now that the ID "doesn't exist" in the page
updateHistory(newHistoryEntry);
// Unmask the note's ID
note?.setAttribute('id', `note_${noteId}`);
} else if (noteId) {
updateHistory(newHistoryEntry);
}
}
/** /**
* @param {object} self Component instance with mixin applied * @param {object} self Component instance with mixin applied
* @param {string} id Discussion id we are jumping to * @param {string} id Discussion id we are jumping to
*/ */
function diffsJump({ expandDiscussion }, id) { function diffsJump({ expandDiscussion }, id, firstNoteId) {
const selector = `ul.notes[data-discussion-id="${id}"]`; const selector = `ul.notes[data-discussion-id="${id}"]`;
eventHub.$once('scrollToDiscussion', () => scrollTo(selector));
eventHub.$once('scrollToDiscussion', () => {
scrollTo(selector);
// Wait for the discussion scroll before updating to the more specific ID
setTimeout(() => updateUrlWithNoteId(firstNoteId), 0);
});
expandDiscussion({ discussionId: id }); expandDiscussion({ discussionId: id });
} }
...@@ -60,12 +91,13 @@ function switchToDiscussionsTabAndJumpTo(self, id) { ...@@ -60,12 +91,13 @@ function switchToDiscussionsTabAndJumpTo(self, id) {
* @param {object} discussion Discussion we are jumping to * @param {object} discussion Discussion we are jumping to
*/ */
function jumpToDiscussion(self, discussion) { function jumpToDiscussion(self, discussion) {
const { id, diff_discussion: isDiffDiscussion } = discussion; const { id, diff_discussion: isDiffDiscussion, notes } = discussion;
const firstNoteId = notes?.[0]?.id;
if (id) { if (id) {
const activeTab = window.mrTabs.currentAction; const activeTab = window.mrTabs.currentAction;
if (activeTab === 'diffs' && isDiffDiscussion) { if (activeTab === 'diffs' && isDiffDiscussion) {
diffsJump(self, id); diffsJump(self, id, firstNoteId);
} else if (activeTab === 'show') { } else if (activeTab === 'show') {
discussionJump(self, id); discussionJump(self, id);
} else { } else {
...@@ -83,6 +115,7 @@ function handleDiscussionJump(self, fn, discussionId = self.currentDiscussionId) ...@@ -83,6 +115,7 @@ function handleDiscussionJump(self, fn, discussionId = self.currentDiscussionId)
const isDiffView = window.mrTabs.currentAction === 'diffs'; const isDiffView = window.mrTabs.currentAction === 'diffs';
const targetId = fn(discussionId, isDiffView); const targetId = fn(discussionId, isDiffView);
const discussion = self.getDiscussion(targetId); const discussion = self.getDiscussion(targetId);
const setHash = !isDiffView && !isDiffsVirtualScrollingEnabled();
const discussionFilePath = discussion?.diff_file?.file_path; const discussionFilePath = discussion?.diff_file?.file_path;
if (isDiffsVirtualScrollingEnabled()) { if (isDiffsVirtualScrollingEnabled()) {
...@@ -92,7 +125,7 @@ function handleDiscussionJump(self, fn, discussionId = self.currentDiscussionId) ...@@ -92,7 +125,7 @@ function handleDiscussionJump(self, fn, discussionId = self.currentDiscussionId)
if (discussionFilePath) { if (discussionFilePath) {
self.scrollToFile({ self.scrollToFile({
path: discussionFilePath, path: discussionFilePath,
setHash: !isDiffsVirtualScrollingEnabled(), setHash,
}); });
} }
......
...@@ -226,16 +226,13 @@ describe('Discussion navigation mixin', () => { ...@@ -226,16 +226,13 @@ describe('Discussion navigation mixin', () => {
${false} ${false}
${true} ${true}
`('virtual scrolling feature is $diffsVirtualScrolling', ({ diffsVirtualScrolling }) => { `('virtual scrolling feature is $diffsVirtualScrolling', ({ diffsVirtualScrolling }) => {
beforeEach(async () => { beforeEach(() => {
window.gon = { features: { diffsVirtualScrolling } }; window.gon = { features: { diffsVirtualScrolling } };
jest.spyOn(store, 'dispatch'); jest.spyOn(store, 'dispatch');
store.state.notes.currentDiscussionId = 'a'; store.state.notes.currentDiscussionId = 'a';
window.location.hash = 'test'; window.location.hash = 'test';
wrapper.vm.jumpToNextDiscussion();
await nextTick();
}); });
afterEach(() => { afterEach(() => {
...@@ -243,16 +240,34 @@ describe('Discussion navigation mixin', () => { ...@@ -243,16 +240,34 @@ describe('Discussion navigation mixin', () => {
window.location.hash = ''; window.location.hash = '';
}); });
it('resets location hash if diffsVirtualScrolling flag is true', () => { it('resets location hash if diffsVirtualScrolling flag is true', async () => {
wrapper.vm.jumpToNextDiscussion();
await nextTick();
expect(window.location.hash).toBe(diffsVirtualScrolling ? '' : '#test'); expect(window.location.hash).toBe(diffsVirtualScrolling ? '' : '#test');
}); });
it(`calls scrollToFile with setHash as ${diffsVirtualScrolling ? 'false' : 'true'}`, () => { it.each`
expect(store.dispatch).toHaveBeenCalledWith('diffs/scrollToFile', { tabValue | hashValue
path: 'test.js', ${'diffs'} | ${false}
setHash: !diffsVirtualScrolling, ${'show'} | ${!diffsVirtualScrolling}
}); ${'other'} | ${!diffsVirtualScrolling}
}); `(
'calls scrollToFile with setHash as $hashValue when the tab is $tabValue',
async ({ hashValue, tabValue }) => {
window.mrTabs.currentAction = tabValue;
wrapper.vm.jumpToNextDiscussion();
await nextTick();
expect(store.dispatch).toHaveBeenCalledWith('diffs/scrollToFile', {
path: 'test.js',
setHash: hashValue,
});
},
);
}); });
}); });
}); });
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