2023-06-20 00:43:36 +05:30
|
|
|
import { nextTick } from 'vue';
|
|
|
|
import { GlCollapse } from '@gitlab/ui';
|
2023-03-17 16:20:25 +05:30
|
|
|
import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
|
|
|
|
import SuperSidebar from '~/super_sidebar/components/super_sidebar.vue';
|
2023-04-23 21:23:45 +05:30
|
|
|
import HelpCenter from '~/super_sidebar/components/help_center.vue';
|
2023-03-17 16:20:25 +05:30
|
|
|
import UserBar from '~/super_sidebar/components/user_bar.vue';
|
2023-05-27 22:25:52 +05:30
|
|
|
import SidebarPortalTarget from '~/super_sidebar/components/sidebar_portal_target.vue';
|
2023-06-20 00:43:36 +05:30
|
|
|
import ContextSwitcher from '~/super_sidebar/components/context_switcher.vue';
|
|
|
|
import {
|
|
|
|
SUPER_SIDEBAR_PEEK_OPEN_DELAY,
|
|
|
|
SUPER_SIDEBAR_PEEK_CLOSE_DELAY,
|
|
|
|
} from '~/super_sidebar/constants';
|
|
|
|
import { stubComponent } from 'helpers/stub_component';
|
2023-03-17 16:20:25 +05:30
|
|
|
import { sidebarData } from '../mock_data';
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
const focusInputMock = jest.fn();
|
2023-05-27 22:25:52 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
describe('SuperSidebar component', () => {
|
|
|
|
let wrapper;
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
const findSidebar = () => wrapper.findByTestId('super-sidebar');
|
|
|
|
const findHoverArea = () => wrapper.findByTestId('super-sidebar-hover-area');
|
2023-03-17 16:20:25 +05:30
|
|
|
const findUserBar = () => wrapper.findComponent(UserBar);
|
2023-04-23 21:23:45 +05:30
|
|
|
const findHelpCenter = () => wrapper.findComponent(HelpCenter);
|
2023-05-27 22:25:52 +05:30
|
|
|
const findSidebarPortalTarget = () => wrapper.findComponent(SidebarPortalTarget);
|
2023-03-17 16:20:25 +05:30
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
const createWrapper = ({ props = {}, provide = {}, sidebarState = {} } = {}) => {
|
2023-03-17 16:20:25 +05:30
|
|
|
wrapper = shallowMountExtended(SuperSidebar, {
|
2023-06-20 00:43:36 +05:30
|
|
|
data() {
|
|
|
|
return {
|
|
|
|
...sidebarState,
|
|
|
|
};
|
|
|
|
},
|
2023-03-17 16:20:25 +05:30
|
|
|
propsData: {
|
|
|
|
sidebarData,
|
|
|
|
...props,
|
|
|
|
},
|
2023-06-20 00:43:36 +05:30
|
|
|
stubs: {
|
|
|
|
ContextSwitcher: stubComponent(ContextSwitcher, {
|
|
|
|
methods: { focusInput: focusInputMock },
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
provide,
|
2023-03-17 16:20:25 +05:30
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
describe('default', () => {
|
2023-06-20 00:43:36 +05:30
|
|
|
it('adds inert attribute when collapsed', () => {
|
|
|
|
createWrapper({ sidebarState: { isCollapsed: true } });
|
2023-05-27 22:25:52 +05:30
|
|
|
expect(findSidebar().attributes('inert')).toBe('inert');
|
|
|
|
});
|
|
|
|
|
2023-06-20 00:43:36 +05:30
|
|
|
it('does not add inert attribute when expanded', () => {
|
2023-05-27 22:25:52 +05:30
|
|
|
createWrapper();
|
|
|
|
expect(findSidebar().attributes('inert')).toBe(undefined);
|
2023-03-17 16:20:25 +05:30
|
|
|
});
|
|
|
|
|
|
|
|
it('renders UserBar with sidebarData', () => {
|
2023-05-27 22:25:52 +05:30
|
|
|
createWrapper();
|
2023-03-17 16:20:25 +05:30
|
|
|
expect(findUserBar().props('sidebarData')).toBe(sidebarData);
|
|
|
|
});
|
2023-04-23 21:23:45 +05:30
|
|
|
|
|
|
|
it('renders HelpCenter with sidebarData', () => {
|
2023-05-27 22:25:52 +05:30
|
|
|
createWrapper();
|
2023-04-23 21:23:45 +05:30
|
|
|
expect(findHelpCenter().props('sidebarData')).toBe(sidebarData);
|
|
|
|
});
|
2023-05-27 22:25:52 +05:30
|
|
|
|
|
|
|
it('renders SidebarPortalTarget', () => {
|
|
|
|
createWrapper();
|
|
|
|
expect(findSidebarPortalTarget().exists()).toBe(true);
|
|
|
|
});
|
2023-06-20 00:43:36 +05:30
|
|
|
|
|
|
|
it("does not call the context switcher's focusInput method initially", () => {
|
|
|
|
createWrapper();
|
|
|
|
|
|
|
|
expect(focusInputMock).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('renders hidden shortcut links', () => {
|
|
|
|
createWrapper();
|
|
|
|
const [linkAttrs] = sidebarData.shortcut_links;
|
|
|
|
const link = wrapper.find(`.${linkAttrs.css_class}`);
|
|
|
|
|
|
|
|
expect(link.exists()).toBe(true);
|
|
|
|
expect(link.attributes('href')).toBe(linkAttrs.href);
|
|
|
|
expect(link.attributes('class')).toContain('gl-display-none');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when peeking on hover', () => {
|
|
|
|
const peekClass = 'super-sidebar-peek';
|
|
|
|
|
|
|
|
it('updates inert attribute and peek class', async () => {
|
|
|
|
createWrapper({
|
|
|
|
provide: { glFeatures: { superSidebarPeek: true } },
|
|
|
|
sidebarState: { isCollapsed: true },
|
|
|
|
});
|
|
|
|
|
|
|
|
findHoverArea().trigger('mouseenter');
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(SUPER_SIDEBAR_PEEK_OPEN_DELAY - 1);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Not quite enough time has elapsed yet for sidebar to open
|
|
|
|
expect(findSidebar().classes()).not.toContain(peekClass);
|
|
|
|
expect(findSidebar().attributes('inert')).toBe('inert');
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(1);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Exactly enough time has elapsed to open
|
|
|
|
expect(findSidebar().classes()).toContain(peekClass);
|
|
|
|
expect(findSidebar().attributes('inert')).toBe(undefined);
|
|
|
|
|
|
|
|
// Important: assume the cursor enters the sidebar
|
|
|
|
findSidebar().trigger('mouseenter');
|
|
|
|
|
|
|
|
jest.runAllTimers();
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Sidebar remains peeked open indefinitely without a mouseleave
|
|
|
|
expect(findSidebar().classes()).toContain(peekClass);
|
|
|
|
expect(findSidebar().attributes('inert')).toBe(undefined);
|
|
|
|
|
|
|
|
findSidebar().trigger('mouseleave');
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(SUPER_SIDEBAR_PEEK_CLOSE_DELAY - 1);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Not quite enough time has elapsed yet for sidebar to hide
|
|
|
|
expect(findSidebar().classes()).toContain(peekClass);
|
|
|
|
expect(findSidebar().attributes('inert')).toBe(undefined);
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(1);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Exactly enough time has elapsed for sidebar to hide
|
|
|
|
expect(findSidebar().classes()).not.toContain('super-sidebar-peek');
|
|
|
|
expect(findSidebar().attributes('inert')).toBe('inert');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('eventually closes the sidebar if cursor never enters sidebar', async () => {
|
|
|
|
createWrapper({
|
|
|
|
provide: { glFeatures: { superSidebarPeek: true } },
|
|
|
|
sidebarState: { isCollapsed: true },
|
|
|
|
});
|
|
|
|
|
|
|
|
findHoverArea().trigger('mouseenter');
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(SUPER_SIDEBAR_PEEK_OPEN_DELAY);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Sidebar is now open
|
|
|
|
expect(findSidebar().classes()).toContain(peekClass);
|
|
|
|
expect(findSidebar().attributes('inert')).toBe(undefined);
|
|
|
|
|
|
|
|
// Important: do *not* fire a mouseenter event on the sidebar here. This
|
|
|
|
// imitates what happens if the cursor moves away from the sidebar before
|
|
|
|
// it actually appears.
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(SUPER_SIDEBAR_PEEK_CLOSE_DELAY - 1);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Not quite enough time has elapsed yet for sidebar to hide
|
|
|
|
expect(findSidebar().classes()).toContain(peekClass);
|
|
|
|
expect(findSidebar().attributes('inert')).toBe(undefined);
|
|
|
|
|
|
|
|
jest.advanceTimersByTime(1);
|
|
|
|
await nextTick();
|
|
|
|
|
|
|
|
// Exactly enough time has elapsed for sidebar to hide
|
|
|
|
expect(findSidebar().classes()).not.toContain('super-sidebar-peek');
|
|
|
|
expect(findSidebar().attributes('inert')).toBe('inert');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('when opening the context switcher', () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
createWrapper();
|
|
|
|
wrapper.findComponent(GlCollapse).vm.$emit('input', true);
|
|
|
|
wrapper.findComponent(GlCollapse).vm.$emit('shown');
|
|
|
|
});
|
|
|
|
|
|
|
|
it("calls the context switcher's focusInput method", () => {
|
|
|
|
expect(focusInputMock).toHaveBeenCalledTimes(1);
|
|
|
|
});
|
2023-03-17 16:20:25 +05:30
|
|
|
});
|
|
|
|
});
|