2018-03-17 18:26:18 +05:30
|
|
|
import Vue from 'vue';
|
|
|
|
|
|
|
|
import bp from '~/breakpoints';
|
|
|
|
import appComponent from '~/projects_dropdown/components/app.vue';
|
|
|
|
import eventHub from '~/projects_dropdown/event_hub';
|
|
|
|
import ProjectsStore from '~/projects_dropdown/store/projects_store';
|
|
|
|
import ProjectsService from '~/projects_dropdown/service/projects_service';
|
|
|
|
|
2018-03-27 19:54:05 +05:30
|
|
|
import mountComponent from 'spec/helpers/vue_mount_component_helper';
|
2018-03-17 18:26:18 +05:30
|
|
|
import { currentSession, mockProject, mockRawProject } from '../mock_data';
|
|
|
|
|
|
|
|
const createComponent = () => {
|
|
|
|
gon.api_version = currentSession.apiVersion;
|
|
|
|
const Component = Vue.extend(appComponent);
|
|
|
|
const store = new ProjectsStore();
|
|
|
|
const service = new ProjectsService(currentSession.username);
|
|
|
|
|
|
|
|
return mountComponent(Component, {
|
|
|
|
store,
|
|
|
|
service,
|
|
|
|
currentUserName: currentSession.username,
|
|
|
|
currentProject: currentSession.project,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const returnServicePromise = (data, failed) => new Promise((resolve, reject) => {
|
|
|
|
if (failed) {
|
|
|
|
reject(data);
|
|
|
|
} else {
|
|
|
|
resolve({
|
|
|
|
json() {
|
|
|
|
return data;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('AppComponent', () => {
|
|
|
|
describe('computed', () => {
|
|
|
|
let vm;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
vm = createComponent();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
vm.$destroy();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('frequentProjects', () => {
|
|
|
|
it('should return list of frequently accessed projects from store', () => {
|
|
|
|
expect(vm.frequentProjects).toBeDefined();
|
|
|
|
expect(vm.frequentProjects.length).toBe(0);
|
|
|
|
|
|
|
|
vm.store.setFrequentProjects([mockProject]);
|
|
|
|
expect(vm.frequentProjects).toBeDefined();
|
|
|
|
expect(vm.frequentProjects.length).toBe(1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('searchProjects', () => {
|
|
|
|
it('should return list of frequently accessed projects from store', () => {
|
|
|
|
expect(vm.searchProjects).toBeDefined();
|
|
|
|
expect(vm.searchProjects.length).toBe(0);
|
|
|
|
|
|
|
|
vm.store.setSearchedProjects([mockRawProject]);
|
|
|
|
expect(vm.searchProjects).toBeDefined();
|
|
|
|
expect(vm.searchProjects.length).toBe(1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('methods', () => {
|
|
|
|
let vm;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
vm = createComponent();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
vm.$destroy();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('toggleFrequentProjectsList', () => {
|
|
|
|
it('should toggle props which control visibility of Frequent Projects list from state passed', () => {
|
|
|
|
vm.toggleFrequentProjectsList(true);
|
|
|
|
expect(vm.isLoadingProjects).toBeFalsy();
|
|
|
|
expect(vm.isSearchListVisible).toBeFalsy();
|
|
|
|
expect(vm.isFrequentsListVisible).toBeTruthy();
|
|
|
|
|
|
|
|
vm.toggleFrequentProjectsList(false);
|
|
|
|
expect(vm.isLoadingProjects).toBeTruthy();
|
|
|
|
expect(vm.isSearchListVisible).toBeTruthy();
|
|
|
|
expect(vm.isFrequentsListVisible).toBeFalsy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('toggleSearchProjectsList', () => {
|
|
|
|
it('should toggle props which control visibility of Searched Projects list from state passed', () => {
|
|
|
|
vm.toggleSearchProjectsList(true);
|
|
|
|
expect(vm.isLoadingProjects).toBeFalsy();
|
|
|
|
expect(vm.isFrequentsListVisible).toBeFalsy();
|
|
|
|
expect(vm.isSearchListVisible).toBeTruthy();
|
|
|
|
|
|
|
|
vm.toggleSearchProjectsList(false);
|
|
|
|
expect(vm.isLoadingProjects).toBeTruthy();
|
|
|
|
expect(vm.isFrequentsListVisible).toBeTruthy();
|
|
|
|
expect(vm.isSearchListVisible).toBeFalsy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('toggleLoader', () => {
|
|
|
|
it('should toggle props which control visibility of list loading animation from state passed', () => {
|
|
|
|
vm.toggleLoader(true);
|
|
|
|
expect(vm.isFrequentsListVisible).toBeFalsy();
|
|
|
|
expect(vm.isSearchListVisible).toBeFalsy();
|
|
|
|
expect(vm.isLoadingProjects).toBeTruthy();
|
|
|
|
|
|
|
|
vm.toggleLoader(false);
|
|
|
|
expect(vm.isFrequentsListVisible).toBeTruthy();
|
|
|
|
expect(vm.isSearchListVisible).toBeTruthy();
|
|
|
|
expect(vm.isLoadingProjects).toBeFalsy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('fetchFrequentProjects', () => {
|
|
|
|
it('should set props for loading animation to `true` while frequent projects list is being loaded', () => {
|
|
|
|
spyOn(vm, 'toggleLoader');
|
|
|
|
|
|
|
|
vm.fetchFrequentProjects();
|
|
|
|
expect(vm.isLocalStorageFailed).toBeFalsy();
|
|
|
|
expect(vm.toggleLoader).toHaveBeenCalledWith(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set props for loading animation to `false` and props for frequent projects list to `true` once data is loaded', () => {
|
|
|
|
const mockData = [mockProject];
|
|
|
|
|
|
|
|
spyOn(vm.service, 'getFrequentProjects').and.returnValue(mockData);
|
|
|
|
spyOn(vm.store, 'setFrequentProjects');
|
|
|
|
spyOn(vm, 'toggleFrequentProjectsList');
|
|
|
|
|
|
|
|
vm.fetchFrequentProjects();
|
|
|
|
expect(vm.service.getFrequentProjects).toHaveBeenCalled();
|
|
|
|
expect(vm.store.setFrequentProjects).toHaveBeenCalledWith(mockData);
|
|
|
|
expect(vm.toggleFrequentProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set props for failure message to `true` when method fails to fetch frequent projects list', () => {
|
|
|
|
spyOn(vm.service, 'getFrequentProjects').and.returnValue(null);
|
|
|
|
spyOn(vm.store, 'setFrequentProjects');
|
|
|
|
spyOn(vm, 'toggleFrequentProjectsList');
|
|
|
|
|
|
|
|
expect(vm.isLocalStorageFailed).toBeFalsy();
|
|
|
|
|
|
|
|
vm.fetchFrequentProjects();
|
|
|
|
expect(vm.service.getFrequentProjects).toHaveBeenCalled();
|
|
|
|
expect(vm.store.setFrequentProjects).toHaveBeenCalledWith([]);
|
|
|
|
expect(vm.toggleFrequentProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
expect(vm.isLocalStorageFailed).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set props for search results list to `true` if search query was already made previously', () => {
|
|
|
|
spyOn(bp, 'getBreakpointSize').and.returnValue('md');
|
|
|
|
spyOn(vm.service, 'getFrequentProjects');
|
|
|
|
spyOn(vm, 'toggleSearchProjectsList');
|
|
|
|
|
|
|
|
vm.searchQuery = 'test';
|
|
|
|
vm.fetchFrequentProjects();
|
|
|
|
expect(vm.service.getFrequentProjects).not.toHaveBeenCalled();
|
|
|
|
expect(vm.toggleSearchProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should set props for frequent projects list to `true` if search query was already made but screen size is less than 768px', () => {
|
|
|
|
spyOn(bp, 'getBreakpointSize').and.returnValue('sm');
|
|
|
|
spyOn(vm, 'toggleSearchProjectsList');
|
|
|
|
spyOn(vm.service, 'getFrequentProjects');
|
|
|
|
|
|
|
|
vm.searchQuery = 'test';
|
|
|
|
vm.fetchFrequentProjects();
|
|
|
|
expect(vm.service.getFrequentProjects).toHaveBeenCalled();
|
|
|
|
expect(vm.toggleSearchProjectsList).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('fetchSearchedProjects', () => {
|
|
|
|
const searchQuery = 'test';
|
|
|
|
|
|
|
|
it('should perform search with provided search query', (done) => {
|
|
|
|
const mockData = [mockRawProject];
|
|
|
|
spyOn(vm, 'toggleLoader');
|
|
|
|
spyOn(vm, 'toggleSearchProjectsList');
|
|
|
|
spyOn(vm.service, 'getSearchedProjects').and.returnValue(returnServicePromise(mockData));
|
|
|
|
spyOn(vm.store, 'setSearchedProjects');
|
|
|
|
|
|
|
|
vm.fetchSearchedProjects(searchQuery);
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(vm.searchQuery).toBe(searchQuery);
|
|
|
|
expect(vm.toggleLoader).toHaveBeenCalledWith(true);
|
|
|
|
expect(vm.service.getSearchedProjects).toHaveBeenCalledWith(searchQuery);
|
|
|
|
expect(vm.toggleSearchProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
expect(vm.store.setSearchedProjects).toHaveBeenCalledWith(mockData);
|
|
|
|
done();
|
|
|
|
}, 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should update props for showing search failure', (done) => {
|
|
|
|
spyOn(vm, 'toggleSearchProjectsList');
|
|
|
|
spyOn(vm.service, 'getSearchedProjects').and.returnValue(returnServicePromise({}, true));
|
|
|
|
|
|
|
|
vm.fetchSearchedProjects(searchQuery);
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(vm.searchQuery).toBe(searchQuery);
|
|
|
|
expect(vm.service.getSearchedProjects).toHaveBeenCalledWith(searchQuery);
|
|
|
|
expect(vm.isSearchFailed).toBeTruthy();
|
|
|
|
expect(vm.toggleSearchProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
done();
|
|
|
|
}, 0);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('logCurrentProjectAccess', () => {
|
|
|
|
it('should log current project access via service', (done) => {
|
|
|
|
spyOn(vm.service, 'logProjectAccess');
|
|
|
|
|
|
|
|
vm.currentProject = mockProject;
|
|
|
|
vm.logCurrentProjectAccess();
|
|
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
expect(vm.service.logProjectAccess).toHaveBeenCalledWith(mockProject);
|
|
|
|
done();
|
|
|
|
}, 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleSearchClear', () => {
|
|
|
|
it('should show frequent projects list when search input is cleared', () => {
|
|
|
|
spyOn(vm.store, 'clearSearchedProjects');
|
|
|
|
spyOn(vm, 'toggleFrequentProjectsList');
|
|
|
|
|
|
|
|
vm.handleSearchClear();
|
|
|
|
|
|
|
|
expect(vm.toggleFrequentProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
expect(vm.store.clearSearchedProjects).toHaveBeenCalled();
|
|
|
|
expect(vm.searchQuery).toBe('');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('handleSearchFailure', () => {
|
|
|
|
it('should show failure message within dropdown', () => {
|
|
|
|
spyOn(vm, 'toggleSearchProjectsList');
|
|
|
|
|
|
|
|
vm.handleSearchFailure();
|
|
|
|
expect(vm.toggleSearchProjectsList).toHaveBeenCalledWith(true);
|
|
|
|
expect(vm.isSearchFailed).toBeTruthy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('created', () => {
|
|
|
|
it('should bind event listeners on eventHub', (done) => {
|
|
|
|
spyOn(eventHub, '$on');
|
|
|
|
|
|
|
|
createComponent().$mount();
|
|
|
|
|
|
|
|
Vue.nextTick(() => {
|
|
|
|
expect(eventHub.$on).toHaveBeenCalledWith('dropdownOpen', jasmine.any(Function));
|
|
|
|
expect(eventHub.$on).toHaveBeenCalledWith('searchProjects', jasmine.any(Function));
|
|
|
|
expect(eventHub.$on).toHaveBeenCalledWith('searchCleared', jasmine.any(Function));
|
|
|
|
expect(eventHub.$on).toHaveBeenCalledWith('searchFailed', jasmine.any(Function));
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('beforeDestroy', () => {
|
|
|
|
it('should unbind event listeners on eventHub', (done) => {
|
|
|
|
const vm = createComponent();
|
|
|
|
spyOn(eventHub, '$off');
|
|
|
|
|
|
|
|
vm.$mount();
|
|
|
|
vm.$destroy();
|
|
|
|
|
|
|
|
Vue.nextTick(() => {
|
|
|
|
expect(eventHub.$off).toHaveBeenCalledWith('dropdownOpen', jasmine.any(Function));
|
|
|
|
expect(eventHub.$off).toHaveBeenCalledWith('searchProjects', jasmine.any(Function));
|
|
|
|
expect(eventHub.$off).toHaveBeenCalledWith('searchCleared', jasmine.any(Function));
|
|
|
|
expect(eventHub.$off).toHaveBeenCalledWith('searchFailed', jasmine.any(Function));
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('template', () => {
|
|
|
|
let vm;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
vm = createComponent();
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
vm.$destroy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should render search input', () => {
|
|
|
|
expect(vm.$el.querySelector('.search-input-container')).toBeDefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should render loading animation', (done) => {
|
|
|
|
vm.toggleLoader(true);
|
|
|
|
Vue.nextTick(() => {
|
|
|
|
const loadingEl = vm.$el.querySelector('.loading-animation');
|
|
|
|
|
|
|
|
expect(loadingEl).toBeDefined();
|
|
|
|
expect(loadingEl.classList.contains('prepend-top-20')).toBeTruthy();
|
|
|
|
expect(loadingEl.querySelector('i').getAttribute('aria-label')).toBe('Loading projects');
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should render frequent projects list header', (done) => {
|
|
|
|
vm.toggleFrequentProjectsList(true);
|
|
|
|
Vue.nextTick(() => {
|
|
|
|
const sectionHeaderEl = vm.$el.querySelector('.section-header');
|
|
|
|
|
|
|
|
expect(sectionHeaderEl).toBeDefined();
|
|
|
|
expect(sectionHeaderEl.innerText.trim()).toBe('Frequently visited');
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should render frequent projects list', (done) => {
|
|
|
|
vm.toggleFrequentProjectsList(true);
|
|
|
|
Vue.nextTick(() => {
|
|
|
|
expect(vm.$el.querySelector('.projects-list-frequent-container')).toBeDefined();
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should render searched projects list', (done) => {
|
|
|
|
vm.toggleSearchProjectsList(true);
|
|
|
|
Vue.nextTick(() => {
|
|
|
|
expect(vm.$el.querySelector('.section-header')).toBe(null);
|
|
|
|
expect(vm.$el.querySelector('.projects-list-search-container')).toBeDefined();
|
|
|
|
done();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|