Commit ce8e4343 authored by GitLab Bot's avatar GitLab Bot

Merge remote-tracking branch 'upstream/master' into ce-to-ee-2018-07-11

parents a997bc26 0db5ccc8
...@@ -2,13 +2,16 @@ import $ from 'jquery'; ...@@ -2,13 +2,16 @@ import $ from 'jquery';
import { rstrip } from './lib/utils/common_utils'; import { rstrip } from './lib/utils/common_utils';
function openConfirmDangerModal($form, text) { function openConfirmDangerModal($form, text) {
const $input = $('.js-confirm-danger-input');
$input.val('');
$('.js-confirm-text').text(text || ''); $('.js-confirm-text').text(text || '');
$('.js-confirm-danger-input').val('');
$('#modal-confirm-danger').modal('show'); $('#modal-confirm-danger').modal('show');
const confirmTextMatch = $('.js-confirm-danger-match').text(); const confirmTextMatch = $('.js-confirm-danger-match').text();
const $submit = $('.js-confirm-danger-submit'); const $submit = $('.js-confirm-danger-submit');
$submit.disable(); $submit.disable();
$input.focus();
$('.js-confirm-danger-input').off('input').on('input', function handleInput() { $('.js-confirm-danger-input').off('input').on('input', function handleInput() {
const confirmText = rstrip($(this).val()); const confirmText = rstrip($(this).val());
......
...@@ -31,9 +31,6 @@ export default { ...@@ -31,9 +31,6 @@ export default {
}; };
}, },
computed: { computed: {
isDiscussionsExpanded() {
return true; // TODO: @fatihacet - Fix this.
},
isCollapsed() { isCollapsed() {
return this.file.collapsed || false; return this.file.collapsed || false;
}, },
...@@ -131,7 +128,6 @@ export default { ...@@ -131,7 +128,6 @@ export default {
:diff-file="file" :diff-file="file"
:collapsible="true" :collapsible="true"
:expanded="!isCollapsed" :expanded="!isCollapsed"
:discussions-expanded="isDiscussionsExpanded"
:add-merge-request-buttons="true" :add-merge-request-buttons="true"
class="js-file-title file-title" class="js-file-title file-title"
@toggleFile="handleToggle" @toggleFile="handleToggle"
......
<script> <script>
import _ from 'underscore'; import _ from 'underscore';
import { mapActions, mapGetters } from 'vuex';
import ClipboardButton from '~/vue_shared/components/clipboard_button.vue'; import ClipboardButton from '~/vue_shared/components/clipboard_button.vue';
import Icon from '~/vue_shared/components/icon.vue'; import Icon from '~/vue_shared/components/icon.vue';
import FileIcon from '~/vue_shared/components/file_icon.vue'; import FileIcon from '~/vue_shared/components/file_icon.vue';
...@@ -38,11 +39,6 @@ export default { ...@@ -38,11 +39,6 @@ export default {
required: false, required: false,
default: true, default: true,
}, },
discussionsExpanded: {
type: Boolean,
required: false,
default: true,
},
currentUser: { currentUser: {
type: Object, type: Object,
required: true, required: true,
...@@ -54,6 +50,10 @@ export default { ...@@ -54,6 +50,10 @@ export default {
}; };
}, },
computed: { computed: {
...mapGetters('diffs', ['diffHasExpandedDiscussions']),
hasExpandedDiscussions() {
return this.diffHasExpandedDiscussions(this.diffFile);
},
icon() { icon() {
if (this.diffFile.submodule) { if (this.diffFile.submodule) {
return 'archive'; return 'archive';
...@@ -88,9 +88,6 @@ export default { ...@@ -88,9 +88,6 @@ export default {
collapseIcon() { collapseIcon() {
return this.expanded ? 'chevron-down' : 'chevron-right'; return this.expanded ? 'chevron-down' : 'chevron-right';
}, },
isDiscussionsExpanded() {
return this.discussionsExpanded && this.expanded;
},
viewFileButtonText() { viewFileButtonText() {
const truncatedContentSha = _.escape(truncateSha(this.diffFile.contentSha)); const truncatedContentSha = _.escape(truncateSha(this.diffFile.contentSha));
return sprintf( return sprintf(
...@@ -113,7 +110,8 @@ export default { ...@@ -113,7 +110,8 @@ export default {
}, },
}, },
methods: { methods: {
handleToggle(e, checkTarget) { ...mapActions('diffs', ['toggleFileDiscussions']),
handleToggleFile(e, checkTarget) {
if ( if (
!checkTarget || !checkTarget ||
e.target === this.$refs.header || e.target === this.$refs.header ||
...@@ -125,6 +123,9 @@ export default { ...@@ -125,6 +123,9 @@ export default {
showForkMessage() { showForkMessage() {
this.$emit('showForkMessage'); this.$emit('showForkMessage');
}, },
handleToggleDiscussions() {
this.toggleFileDiscussions(this.diffFile);
},
}, },
}; };
</script> </script>
...@@ -133,7 +134,7 @@ export default { ...@@ -133,7 +134,7 @@ export default {
<div <div
ref="header" ref="header"
class="js-file-title file-title file-title-flex-parent" class="js-file-title file-title file-title-flex-parent"
@click="handleToggle($event, true)" @click="handleToggleFile($event, true)"
> >
<div class="file-header-content"> <div class="file-header-content">
<icon <icon
...@@ -216,10 +217,11 @@ export default { ...@@ -216,10 +217,11 @@ export default {
v-if="diffFile.blob && diffFile.blob.readableText" v-if="diffFile.blob && diffFile.blob.readableText"
> >
<button <button
:class="{ active: isDiscussionsExpanded }" :class="{ active: hasExpandedDiscussions }"
:title="s__('MergeRequests|Toggle comments for this file')" :title="s__('MergeRequests|Toggle comments for this file')"
class="btn js-toggle-diff-comments" class="js-btn-vue-toggle-comments btn"
type="button" type="button"
@click="handleToggleDiscussions"
> >
<icon name="comment" /> <icon name="comment" />
</button> </button>
......
...@@ -82,5 +82,32 @@ export const expandAllFiles = ({ commit }) => { ...@@ -82,5 +82,32 @@ export const expandAllFiles = ({ commit }) => {
commit(types.EXPAND_ALL_FILES); commit(types.EXPAND_ALL_FILES);
}; };
/**
* Toggles the file discussions after user clicked on the toggle discussions button.
*
* Gets the discussions for the provided diff.
*
* If all discussions are expanded, it will collapse all of them
* If all discussions are collapsed, it will expand all of them
* If some discussions are open and others closed, it will expand the closed ones.
*
* @param {Object} diff
*/
export const toggleFileDiscussions = ({ getters, dispatch }, diff) => {
const discussions = getters.getDiffFileDiscussions(diff);
const shouldCloseAll = getters.diffHasAllExpandedDiscussions(diff);
const shouldExpandAll = getters.diffHasAllCollpasedDiscussions(diff);
discussions.forEach(discussion => {
const data = { discussionId: discussion.id };
if (shouldCloseAll) {
dispatch('collapseDiscussion', data, { root: true });
} else if (shouldExpandAll || (!shouldCloseAll && !shouldExpandAll && !discussion.expanded)) {
dispatch('expandDiscussion', data, { root: true });
}
});
};
// prevent babel-plugin-rewire from generating an invalid default during karma tests // prevent babel-plugin-rewire from generating an invalid default during karma tests
export default () => {}; export default () => {};
import _ from 'underscore';
import { PARALLEL_DIFF_VIEW_TYPE, INLINE_DIFF_VIEW_TYPE } from '../constants'; import { PARALLEL_DIFF_VIEW_TYPE, INLINE_DIFF_VIEW_TYPE } from '../constants';
export const isParallelView = state => state.diffViewType === PARALLEL_DIFF_VIEW_TYPE; export const isParallelView = state => state.diffViewType === PARALLEL_DIFF_VIEW_TYPE;
...@@ -8,5 +9,52 @@ export const areAllFilesCollapsed = state => state.diffFiles.every(file => file. ...@@ -8,5 +9,52 @@ export const areAllFilesCollapsed = state => state.diffFiles.every(file => file.
export const commitId = state => (state.commit && state.commit.id ? state.commit.id : null); export const commitId = state => (state.commit && state.commit.id ? state.commit.id : null);
// prevent babel-plugin-rewire from generating an invalid default during karma tests /**
* Checks if the diff has all discussions expanded
* @param {Object} diff
* @returns {Boolean}
*/
export const diffHasAllExpandedDiscussions = (state, getters) => diff => {
const discussions = getters.getDiffFileDiscussions(diff);
return (discussions.length && discussions.every(discussion => discussion.expanded)) || false;
};
/**
* Checks if the diff has all discussions collpased
* @param {Object} diff
* @returns {Boolean}
*/
export const diffHasAllCollpasedDiscussions = (state, getters) => diff => {
const discussions = getters.getDiffFileDiscussions(diff);
return (discussions.length && discussions.every(discussion => !discussion.expanded)) || false;
};
/**
* Checks if the diff has any open discussions
* @param {Object} diff
* @returns {Boolean}
*/
export const diffHasExpandedDiscussions = (state, getters) => diff => {
const discussions = getters.getDiffFileDiscussions(diff);
return (
(discussions.length && discussions.find(discussion => discussion.expanded) !== undefined) ||
false
);
};
/**
* Returns an array with the discussions of the given diff
* @param {Object} diff
* @returns {Array}
*/
export const getDiffFileDiscussions = (state, getters, rootState, rootGetters) => diff =>
rootGetters.discussions.filter(
discussion =>
discussion.diff_discussion && _.isEqual(discussion.diff_file.file_hash, diff.fileHash),
) || [];
// prevent babel-plugin-rewire from generating an invalid default during karma∂ tests
export default () => {}; export default () => {};
...@@ -15,6 +15,8 @@ let eTagPoll; ...@@ -15,6 +15,8 @@ let eTagPoll;
export const expandDiscussion = ({ commit }, data) => commit(types.EXPAND_DISCUSSION, data); export const expandDiscussion = ({ commit }, data) => commit(types.EXPAND_DISCUSSION, data);
export const collapseDiscussion = ({ commit }, data) => commit(types.COLLAPSE_DISCUSSION, data);
export const setNotesData = ({ commit }, data) => commit(types.SET_NOTES_DATA, data); export const setNotesData = ({ commit }, data) => commit(types.SET_NOTES_DATA, data);
export const setNoteableData = ({ commit }, data) => commit(types.SET_NOTEABLE_DATA, data); export const setNoteableData = ({ commit }, data) => commit(types.SET_NOTEABLE_DATA, data);
......
export const ADD_NEW_NOTE = 'ADD_NEW_NOTE'; export const ADD_NEW_NOTE = 'ADD_NEW_NOTE';
export const ADD_NEW_REPLY_TO_DISCUSSION = 'ADD_NEW_REPLY_TO_DISCUSSION'; export const ADD_NEW_REPLY_TO_DISCUSSION = 'ADD_NEW_REPLY_TO_DISCUSSION';
export const DELETE_NOTE = 'DELETE_NOTE'; export const DELETE_NOTE = 'DELETE_NOTE';
export const EXPAND_DISCUSSION = 'EXPAND_DISCUSSION';
export const REMOVE_PLACEHOLDER_NOTES = 'REMOVE_PLACEHOLDER_NOTES'; export const REMOVE_PLACEHOLDER_NOTES = 'REMOVE_PLACEHOLDER_NOTES';
export const SET_NOTES_DATA = 'SET_NOTES_DATA'; export const SET_NOTES_DATA = 'SET_NOTES_DATA';
export const SET_NOTEABLE_DATA = 'SET_NOTEABLE_DATA'; export const SET_NOTEABLE_DATA = 'SET_NOTEABLE_DATA';
...@@ -11,12 +10,16 @@ export const SET_LAST_FETCHED_AT = 'SET_LAST_FETCHED_AT'; ...@@ -11,12 +10,16 @@ export const SET_LAST_FETCHED_AT = 'SET_LAST_FETCHED_AT';
export const SET_TARGET_NOTE_HASH = 'SET_TARGET_NOTE_HASH'; export const SET_TARGET_NOTE_HASH = 'SET_TARGET_NOTE_HASH';
export const SHOW_PLACEHOLDER_NOTE = 'SHOW_PLACEHOLDER_NOTE'; export const SHOW_PLACEHOLDER_NOTE = 'SHOW_PLACEHOLDER_NOTE';
export const TOGGLE_AWARD = 'TOGGLE_AWARD'; export const TOGGLE_AWARD = 'TOGGLE_AWARD';
export const TOGGLE_DISCUSSION = 'TOGGLE_DISCUSSION';
export const UPDATE_NOTE = 'UPDATE_NOTE'; export const UPDATE_NOTE = 'UPDATE_NOTE';
export const UPDATE_DISCUSSION = 'UPDATE_DISCUSSION'; export const UPDATE_DISCUSSION = 'UPDATE_DISCUSSION';
export const SET_DISCUSSION_DIFF_LINES = 'SET_DISCUSSION_DIFF_LINES'; export const SET_DISCUSSION_DIFF_LINES = 'SET_DISCUSSION_DIFF_LINES';
export const SET_NOTES_FETCHED_STATE = 'SET_NOTES_FETCHED_STATE'; export const SET_NOTES_FETCHED_STATE = 'SET_NOTES_FETCHED_STATE';
// DISCUSSION
export const COLLAPSE_DISCUSSION = 'COLLAPSE_DISCUSSION';
export const EXPAND_DISCUSSION = 'EXPAND_DISCUSSION';
export const TOGGLE_DISCUSSION = 'TOGGLE_DISCUSSION';
// Issue // Issue
export const CLOSE_ISSUE = 'CLOSE_ISSUE'; export const CLOSE_ISSUE = 'CLOSE_ISSUE';
export const REOPEN_ISSUE = 'REOPEN_ISSUE'; export const REOPEN_ISSUE = 'REOPEN_ISSUE';
......
...@@ -58,6 +58,11 @@ export default { ...@@ -58,6 +58,11 @@ export default {
discussion.expanded = true; discussion.expanded = true;
}, },
[types.COLLAPSE_DISCUSSION](state, { discussionId }) {
const discussion = utils.findNoteObjectById(state.discussions, discussionId);
discussion.expanded = false;
},
[types.REMOVE_PLACEHOLDER_NOTES](state) { [types.REMOVE_PLACEHOLDER_NOTES](state) {
const { discussions } = state; const { discussions } = state;
......
...@@ -116,10 +116,8 @@ ...@@ -116,10 +116,8 @@
.modify-merge-commit-link { .modify-merge-commit-link {
padding: 0; padding: 0;
background-color: transparent; background-color: transparent;
border: 0; border: 0;
color: $gl-text-color; color: $gl-text-color;
&:hover, &:hover,
...@@ -501,10 +499,6 @@ ...@@ -501,10 +499,6 @@
} }
} }
.merge-request-details .content-block {
border-bottom: 0;
}
.mr-source-target { .mr-source-target {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
......
.detail-page-description.content-block .detail-page-description
%h2.title %h2.title
= markdown_field(@merge_request, :title) = markdown_field(@merge_request, :title)
......
...@@ -23,6 +23,8 @@ module ChangelogHelpers ...@@ -23,6 +23,8 @@ module ChangelogHelpers
Abort = Class.new(StandardError) Abort = Class.new(StandardError)
Done = Class.new(StandardError) Done = Class.new(StandardError)
MAX_FILENAME_LENGTH = 140 # ecryptfs has a limit of 140 characters
def capture_stdout(cmd) def capture_stdout(cmd)
output = IO.popen(cmd, &:read) output = IO.popen(cmd, &:read)
fail_with "command failed: #{cmd.join(' ')}" unless $?.success? fail_with "command failed: #{cmd.join(' ')}" unless $?.success?
...@@ -142,7 +144,9 @@ class ChangelogEntry ...@@ -142,7 +144,9 @@ class ChangelogEntry
def initialize(options) def initialize(options)
@options = options @options = options
end
def execute
assert_feature_branch! assert_feature_branch!
assert_title! assert_title!
assert_new_file! assert_new_file!
...@@ -221,10 +225,12 @@ class ChangelogEntry ...@@ -221,10 +225,12 @@ class ChangelogEntry
end end
def file_path def file_path
File.join( base_path = File.join(
unreleased_path, unreleased_path,
branch_name.gsub(/[^\w-]/, '-') << '.yml' branch_name.gsub(/[^\w-]/, '-'))
)
# Add padding for .yml extension
base_path[0..MAX_FILENAME_LENGTH - 5] + '.yml'
end end
def unreleased_path def unreleased_path
...@@ -250,7 +256,7 @@ end ...@@ -250,7 +256,7 @@ end
if $0 == __FILE__ if $0 == __FILE__
begin begin
options = ChangelogOptionParser.parse(ARGV) options = ChangelogOptionParser.parse(ARGV)
ChangelogEntry.new(options) ChangelogEntry.new(options).execute
rescue ChangelogHelpers::Abort => ex rescue ChangelogHelpers::Abort => ex
$stderr.puts ex.message $stderr.puts ex.message
exit 1 exit 1
......
---
title: Fixes toggle discussion button not expanding collapsed discussions
merge_request: 20452
author:
type: fixed
---
title: Improve danger confirmation modals by focusing input field
merge_request:
author: Jamie Schembri
type: added
module Gitlab module Gitlab
# Helper methods to do with Kubernetes network services & resources # Helper methods to do with Kubernetes network services & resources
module Kubernetes module Kubernetes
def self.build_header_hash
Hash.new { |h, k| h[k] = [] }
end
# This is the comand that is run to start a terminal session. Kubernetes # This is the comand that is run to start a terminal session. Kubernetes
# expects `command=foo&command=bar, not `command[]=foo&command[]=bar` # expects `command=foo&command=bar, not `command[]=foo&command[]=bar`
EXEC_COMMAND = URI.encode_www_form( EXEC_COMMAND = URI.encode_www_form(
...@@ -37,13 +41,14 @@ module Gitlab ...@@ -37,13 +41,14 @@ module Gitlab
selectors: { pod: pod_name, container: container["name"] }, selectors: { pod: pod_name, container: container["name"] },
url: container_exec_url(api_url, namespace, pod_name, container["name"]), url: container_exec_url(api_url, namespace, pod_name, container["name"]),
subprotocols: ['channel.k8s.io'], subprotocols: ['channel.k8s.io'],
headers: Hash.new { |h, k| h[k] = [] }, headers: ::Gitlab::Kubernetes.build_header_hash,
created_at: created_at created_at: created_at
} }
end end
end end
def add_terminal_auth(terminal, token:, max_session_time:, ca_pem: nil) def add_terminal_auth(terminal, token:, max_session_time:, ca_pem: nil)
terminal[:headers] ||= ::Gitlab::Kubernetes.build_header_hash
terminal[:headers]['Authorization'] << "Bearer #{token}" terminal[:headers]['Authorization'] << "Bearer #{token}"
terminal[:max_session_time] = max_session_time terminal[:max_session_time] = max_session_time
terminal[:ca_pem] = ca_pem if ca_pem.present? terminal[:ca_pem] = ca_pem if ca_pem.present?
......
...@@ -85,6 +85,10 @@ module QA ...@@ -85,6 +85,10 @@ module QA
driver.browser.save_screenshot(path) driver.browser.save_screenshot(path)
end end
Capybara::Screenshot.register_filename_prefix_formatter(:rspec) do |example|
File.join(QA::Runtime::Namespace.name, example.file_path.sub('./qa/specs/features/', ''))
end
Capybara.configure do |config| Capybara.configure do |config|
config.default_driver = :chrome config.default_driver = :chrome
config.javascript_driver = :chrome config.javascript_driver = :chrome
......
...@@ -8,7 +8,7 @@ module QA ...@@ -8,7 +8,7 @@ module QA
end end
def name def name
'qa-test-' + time.strftime('%d-%m-%Y-%H-%M-%S') "qa-test-#{time.strftime('%Y-%m-%d-%Y-%H-%M-%S')}"
end end
def path def path
......
...@@ -3,6 +3,20 @@ require 'spec_helper' ...@@ -3,6 +3,20 @@ require 'spec_helper'
load File.expand_path('../../bin/changelog', __dir__) load File.expand_path('../../bin/changelog', __dir__)
describe 'bin/changelog' do describe 'bin/changelog' do
let(:options) { OpenStruct.new(title: 'Test title', type: 'fixed', dry_run: true) }
describe ChangelogEntry do
it 'truncates the file path' do
entry = described_class.new(options)
allow(entry).to receive(:ee?).and_return(false)
allow(entry).to receive(:branch_name).and_return('long-branch-' * 100)
file_path = entry.send(:file_path)
expect(file_path.length).to eq(140)
end
end
describe ChangelogOptionParser do describe ChangelogOptionParser do
describe '.parse' do describe '.parse' do
it 'parses --amend' do it 'parses --amend' do
......
...@@ -154,6 +154,12 @@ describe 'Group' do ...@@ -154,6 +154,12 @@ describe 'Group' do
end end
end end
it 'focuses confirmation field on remove group' do
click_button('Remove group')
expect(page).to have_selector '#confirm_name_input:focus'
end
it 'removes group' do it 'removes group' do
expect { remove_with_confirm('Remove group', group.path) }.to change {Group.count}.by(-1) expect { remove_with_confirm('Remove group', group.path) }.to change {Group.count}.by(-1)
expect(group.members.all.count).to be_zero expect(group.members.all.count).to be_zero
......
...@@ -31,7 +31,7 @@ describe 'User comments on a diff', :js do ...@@ -31,7 +31,7 @@ describe 'User comments on a diff', :js do
page.within('.files > div:nth-child(3)') do page.within('.files > div:nth-child(3)') do
expect(page).to have_content('Line is wrong') expect(page).to have_content('Line is wrong')
find('.js-toggle-diff-comments').click find('.js-btn-vue-toggle-comments').click
expect(page).not_to have_content('Line is wrong') expect(page).not_to have_content('Line is wrong')
end end
...@@ -64,7 +64,7 @@ describe 'User comments on a diff', :js do ...@@ -64,7 +64,7 @@ describe 'User comments on a diff', :js do
# Hide the comment. # Hide the comment.
page.within('.files > div:nth-child(3)') do page.within('.files > div:nth-child(3)') do
find('.js-toggle-diff-comments').click find('.js-btn-vue-toggle-comments').click
expect(page).not_to have_content('Line is wrong') expect(page).not_to have_content('Line is wrong')
end end
...@@ -77,7 +77,7 @@ describe 'User comments on a diff', :js do ...@@ -77,7 +77,7 @@ describe 'User comments on a diff', :js do
# Show the comment. # Show the comment.
page.within('.files > div:nth-child(3)') do page.within('.files > div:nth-child(3)') do
find('.js-toggle-diff-comments').click find('.js-btn-vue-toggle-comments').click
end end
# Now both the comments should be shown. # Now both the comments should be shown.
......
...@@ -10,7 +10,7 @@ describe 'Projects > Settings > User transfers a project', :js do ...@@ -10,7 +10,7 @@ describe 'Projects > Settings > User transfers a project', :js do
sign_in(user) sign_in(user)
end end
def transfer_project(project, group) def transfer_project(project, group, confirm: true)
visit edit_project_path(project) visit edit_project_path(project)
page.within('.js-project-transfer-form') do page.within('.js-project-transfer-form') do
...@@ -21,6 +21,8 @@ describe 'Projects > Settings > User transfers a project', :js do ...@@ -21,6 +21,8 @@ describe 'Projects > Settings > User transfers a project', :js do
click_button('Transfer project') click_button('Transfer project')
return unless confirm
fill_in 'confirm_name_input', with: project.name fill_in 'confirm_name_input', with: project.name
click_button 'Confirm' click_button 'Confirm'
...@@ -28,6 +30,11 @@ describe 'Projects > Settings > User transfers a project', :js do ...@@ -28,6 +30,11 @@ describe 'Projects > Settings > User transfers a project', :js do
wait_for_requests wait_for_requests
end end
it 'focuses on the confirmation field' do
transfer_project(project, group, confirm: false)
expect(page).to have_selector '#confirm_name_input:focus'
end
it 'allows transferring a project to a group' do it 'allows transferring a project to a group' do
old_path = project_path(project) old_path = project_path(project)
transfer_project(project, group) transfer_project(project, group)
......
...@@ -155,6 +155,12 @@ describe 'Project' do ...@@ -155,6 +155,12 @@ describe 'Project' do
visit edit_project_path(project) visit edit_project_path(project)
end end
it 'focuses on the confirmation field' do
click_button 'Remove project'
expect(page).to have_selector '#confirm_name_input:focus'
end
it 'removes a project' do it 'removes a project' do
expect { remove_with_confirm('Remove project', project.path) }.to change { Project.count }.by(-1) expect { remove_with_confirm('Remove project', project.path) }.to change { Project.count }.by(-1)
expect(page).to have_content "Project '#{project.full_name}' is in the process of being deleted." expect(page).to have_content "Project '#{project.full_name}' is in the process of being deleted."
......
import Vue from 'vue'; import Vue from 'vue';
import Vuex from 'vuex';
import diffsModule from '~/diffs/store/modules';
import notesModule from '~/notes/stores/modules';
import DiffFileHeader from '~/diffs/components/diff_file_header.vue'; import DiffFileHeader from '~/diffs/components/diff_file_header.vue';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils'; import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import mountComponent from 'spec/helpers/vue_mount_component_helper'; import { mountComponentWithStore } from 'spec/helpers/vue_mount_component_helper';
const discussionFixture = 'merge_requests/diff_discussion.json'; const discussionFixture = 'merge_requests/diff_discussion.json';
...@@ -9,6 +12,12 @@ describe('diff_file_header', () => { ...@@ -9,6 +12,12 @@ describe('diff_file_header', () => {
let vm; let vm;
let props; let props;
const Component = Vue.extend(DiffFileHeader); const Component = Vue.extend(DiffFileHeader);
const store = new Vuex.Store({
modules: {
diffs: diffsModule,
notes: notesModule,
},
});
beforeEach(() => { beforeEach(() => {
const diffDiscussionMock = getJSONFixture(discussionFixture)[0]; const diffDiscussionMock = getJSONFixture(discussionFixture)[0];
...@@ -26,13 +35,13 @@ describe('diff_file_header', () => { ...@@ -26,13 +35,13 @@ describe('diff_file_header', () => {
describe('computed', () => { describe('computed', () => {
describe('icon', () => { describe('icon', () => {
beforeEach(() => { beforeEach(() => {
props.diffFile.blob.icon = 'dummy icon'; props.diffFile.blob.icon = 'file-text-o';
}); });
it('returns the blob icon for files', () => { it('returns the blob icon for files', () => {
props.diffFile.submodule = false; props.diffFile.submodule = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.icon).toBe(props.diffFile.blob.icon); expect(vm.icon).toBe(props.diffFile.blob.icon);
}); });
...@@ -40,7 +49,7 @@ describe('diff_file_header', () => { ...@@ -40,7 +49,7 @@ describe('diff_file_header', () => {
it('returns the archive icon for submodules', () => { it('returns the archive icon for submodules', () => {
props.diffFile.submodule = true; props.diffFile.submodule = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.icon).toBe('archive'); expect(vm.icon).toBe('archive');
}); });
...@@ -58,7 +67,7 @@ describe('diff_file_header', () => { ...@@ -58,7 +67,7 @@ describe('diff_file_header', () => {
it('returns the fileHash for files', () => { it('returns the fileHash for files', () => {
props.diffFile.submodule = false; props.diffFile.submodule = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.titleLink).toBe(`#${props.diffFile.fileHash}`); expect(vm.titleLink).toBe(`#${props.diffFile.fileHash}`);
}); });
...@@ -66,7 +75,7 @@ describe('diff_file_header', () => { ...@@ -66,7 +75,7 @@ describe('diff_file_header', () => {
it('returns the submoduleTreeUrl for submodules', () => { it('returns the submoduleTreeUrl for submodules', () => {
props.diffFile.submodule = true; props.diffFile.submodule = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.titleLink).toBe(props.diffFile.submoduleTreeUrl); expect(vm.titleLink).toBe(props.diffFile.submoduleTreeUrl);
}); });
...@@ -77,7 +86,7 @@ describe('diff_file_header', () => { ...@@ -77,7 +86,7 @@ describe('diff_file_header', () => {
submoduleTreeUrl: null, submoduleTreeUrl: null,
}); });
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.titleLink).toBe(props.diffFile.submoduleLink); expect(vm.titleLink).toBe(props.diffFile.submoduleLink);
}); });
...@@ -94,7 +103,7 @@ describe('diff_file_header', () => { ...@@ -94,7 +103,7 @@ describe('diff_file_header', () => {
it('returns the filePath for files', () => { it('returns the filePath for files', () => {
props.diffFile.submodule = false; props.diffFile.submodule = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.filePath).toBe(props.diffFile.filePath); expect(vm.filePath).toBe(props.diffFile.filePath);
}); });
...@@ -102,7 +111,7 @@ describe('diff_file_header', () => { ...@@ -102,7 +111,7 @@ describe('diff_file_header', () => {
it('appends the truncated blob id for submodules', () => { it('appends the truncated blob id for submodules', () => {
props.diffFile.submodule = true; props.diffFile.submodule = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.filePath).toBe( expect(vm.filePath).toBe(
`${props.diffFile.filePath} @ ${props.diffFile.blob.id.substr(0, 8)}`, `${props.diffFile.filePath} @ ${props.diffFile.blob.id.substr(0, 8)}`,
...@@ -114,7 +123,7 @@ describe('diff_file_header', () => { ...@@ -114,7 +123,7 @@ describe('diff_file_header', () => {
it('returns a link tag if fileHash is set', () => { it('returns a link tag if fileHash is set', () => {
props.diffFile.fileHash = 'some hash'; props.diffFile.fileHash = 'some hash';
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.titleTag).toBe('a'); expect(vm.titleTag).toBe('a');
}); });
...@@ -122,7 +131,7 @@ describe('diff_file_header', () => { ...@@ -122,7 +131,7 @@ describe('diff_file_header', () => {
it('returns a span tag if fileHash is not set', () => { it('returns a span tag if fileHash is not set', () => {
props.diffFile.fileHash = null; props.diffFile.fileHash = null;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.titleTag).toBe('span'); expect(vm.titleTag).toBe('span');
}); });
...@@ -137,7 +146,7 @@ describe('diff_file_header', () => { ...@@ -137,7 +146,7 @@ describe('diff_file_header', () => {
}); });
it('returns true if file is stored in LFS', () => { it('returns true if file is stored in LFS', () => {
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.isUsingLfs).toBe(true); expect(vm.isUsingLfs).toBe(true);
}); });
...@@ -145,7 +154,7 @@ describe('diff_file_header', () => { ...@@ -145,7 +154,7 @@ describe('diff_file_header', () => {
it('returns false if file is not stored externally', () => { it('returns false if file is not stored externally', () => {
props.diffFile.storedExternally = false; props.diffFile.storedExternally = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.isUsingLfs).toBe(false); expect(vm.isUsingLfs).toBe(false);
}); });
...@@ -153,7 +162,7 @@ describe('diff_file_header', () => { ...@@ -153,7 +162,7 @@ describe('diff_file_header', () => {
it('returns false if file is not stored in LFS', () => { it('returns false if file is not stored in LFS', () => {
props.diffFile.externalStorage = 'not lfs'; props.diffFile.externalStorage = 'not lfs';
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.isUsingLfs).toBe(false); expect(vm.isUsingLfs).toBe(false);
}); });
...@@ -163,7 +172,7 @@ describe('diff_file_header', () => { ...@@ -163,7 +172,7 @@ describe('diff_file_header', () => {
it('returns chevron-down if the diff is expanded', () => { it('returns chevron-down if the diff is expanded', () => {
props.expanded = true; props.expanded = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.collapseIcon).toBe('chevron-down'); expect(vm.collapseIcon).toBe('chevron-down');
}); });
...@@ -171,49 +180,18 @@ describe('diff_file_header', () => { ...@@ -171,49 +180,18 @@ describe('diff_file_header', () => {
it('returns chevron-right if the diff is collapsed', () => { it('returns chevron-right if the diff is collapsed', () => {
props.expanded = false; props.expanded = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.collapseIcon).toBe('chevron-right'); expect(vm.collapseIcon).toBe('chevron-right');
}); });
}); });
describe('isDiscussionsExpanded', () => {
beforeEach(() => {
Object.assign(props, {
discussionsExpanded: true,
expanded: true,
});
});
it('returns true if diff and discussion are expanded', () => {
vm = mountComponent(Component, props);
expect(vm.isDiscussionsExpanded).toBe(true);
});
it('returns false if discussion is collapsed', () => {
props.discussionsExpanded = false;
vm = mountComponent(Component, props);
expect(vm.isDiscussionsExpanded).toBe(false);
});
it('returns false if diff is collapsed', () => {
props.expanded = false;
vm = mountComponent(Component, props);
expect(vm.isDiscussionsExpanded).toBe(false);
});
});
describe('viewFileButtonText', () => { describe('viewFileButtonText', () => {
it('contains the truncated content SHA', () => { it('contains the truncated content SHA', () => {
const dummySha = 'deebd00f is no SHA'; const dummySha = 'deebd00f is no SHA';
props.diffFile.contentSha = dummySha; props.diffFile.contentSha = dummySha;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.viewFileButtonText).not.toContain(dummySha); expect(vm.viewFileButtonText).not.toContain(dummySha);
expect(vm.viewFileButtonText).toContain(dummySha.substr(0, 8)); expect(vm.viewFileButtonText).toContain(dummySha.substr(0, 8));
...@@ -225,7 +203,7 @@ describe('diff_file_header', () => { ...@@ -225,7 +203,7 @@ describe('diff_file_header', () => {
const dummySha = 'deadabba sings no more'; const dummySha = 'deadabba sings no more';
props.diffFile.diffRefs.baseSha = dummySha; props.diffFile.diffRefs.baseSha = dummySha;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.viewReplacedFileButtonText).not.toContain(dummySha); expect(vm.viewReplacedFileButtonText).not.toContain(dummySha);
expect(vm.viewReplacedFileButtonText).toContain(dummySha.substr(0, 8)); expect(vm.viewReplacedFileButtonText).toContain(dummySha.substr(0, 8));
...@@ -234,25 +212,25 @@ describe('diff_file_header', () => { ...@@ -234,25 +212,25 @@ describe('diff_file_header', () => {
}); });
describe('methods', () => { describe('methods', () => {
describe('handleToggle', () => { describe('handleToggleFile', () => {
beforeEach(() => { beforeEach(() => {
spyOn(vm, '$emit').and.stub(); spyOn(vm, '$emit').and.stub();
}); });
it('emits toggleFile if checkTarget is false', () => { it('emits toggleFile if checkTarget is false', () => {
vm.handleToggle(null, false); vm.handleToggleFile(null, false);
expect(vm.$emit).toHaveBeenCalledWith('toggleFile'); expect(vm.$emit).toHaveBeenCalledWith('toggleFile');
}); });
it('emits toggleFile if checkTarget is true and event target is header', () => { it('emits toggleFile if checkTarget is true and event target is header', () => {
vm.handleToggle({ target: vm.$refs.header }, true); vm.handleToggleFile({ target: vm.$refs.header }, true);
expect(vm.$emit).toHaveBeenCalledWith('toggleFile'); expect(vm.$emit).toHaveBeenCalledWith('toggleFile');
}); });
it('does not emit toggleFile if checkTarget is true and event target is not header', () => { it('does not emit toggleFile if checkTarget is true and event target is not header', () => {
vm.handleToggle({ target: 'not header' }, true); vm.handleToggleFile({ target: 'not header' }, true);
expect(vm.$emit).not.toHaveBeenCalled(); expect(vm.$emit).not.toHaveBeenCalled();
}); });
...@@ -266,7 +244,7 @@ describe('diff_file_header', () => { ...@@ -266,7 +244,7 @@ describe('diff_file_header', () => {
it('is visible if collapsible is true', () => { it('is visible if collapsible is true', () => {
props.collapsible = true; props.collapsible = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(collapseToggle()).not.toBe(null); expect(collapseToggle()).not.toBe(null);
}); });
...@@ -274,14 +252,14 @@ describe('diff_file_header', () => { ...@@ -274,14 +252,14 @@ describe('diff_file_header', () => {
it('is hidden if collapsible is false', () => { it('is hidden if collapsible is false', () => {
props.collapsible = false; props.collapsible = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(collapseToggle()).toBe(null); expect(collapseToggle()).toBe(null);
}); });
}); });
it('displays an file icon in the title', () => { it('displays an file icon in the title', () => {
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.$el.querySelector('svg.js-file-icon use').getAttribute('xlink:href')).toContain( expect(vm.$el.querySelector('svg.js-file-icon use').getAttribute('xlink:href')).toContain(
'ruby', 'ruby',
); );
...@@ -293,7 +271,7 @@ describe('diff_file_header', () => { ...@@ -293,7 +271,7 @@ describe('diff_file_header', () => {
it('displays the path of a added file', () => { it('displays the path of a added file', () => {
props.diffFile.renamedFile = false; props.diffFile.renamedFile = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(filePaths()).toHaveLength(1); expect(filePaths()).toHaveLength(1);
expect(filePaths()[0]).toHaveText(props.diffFile.filePath); expect(filePaths()[0]).toHaveText(props.diffFile.filePath);
...@@ -303,7 +281,7 @@ describe('diff_file_header', () => { ...@@ -303,7 +281,7 @@ describe('diff_file_header', () => {
props.diffFile.renamedFile = false; props.diffFile.renamedFile = false;
props.diffFile.deletedFile = true; props.diffFile.deletedFile = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(filePaths()).toHaveLength(1); expect(filePaths()).toHaveLength(1);
expect(filePaths()[0]).toHaveText(`${props.diffFile.filePath} deleted`); expect(filePaths()[0]).toHaveText(`${props.diffFile.filePath} deleted`);
...@@ -312,7 +290,7 @@ describe('diff_file_header', () => { ...@@ -312,7 +290,7 @@ describe('diff_file_header', () => {
it('displays old and new path if the file was renamed', () => { it('displays old and new path if the file was renamed', () => {
props.diffFile.renamedFile = true; props.diffFile.renamedFile = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(filePaths()).toHaveLength(2); expect(filePaths()).toHaveLength(2);
expect(filePaths()[0]).toHaveText(props.diffFile.oldPath); expect(filePaths()[0]).toHaveText(props.diffFile.oldPath);
...@@ -321,7 +299,7 @@ describe('diff_file_header', () => { ...@@ -321,7 +299,7 @@ describe('diff_file_header', () => {
}); });
it('displays a copy to clipboard button', () => { it('displays a copy to clipboard button', () => {
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
const button = vm.$el.querySelector('.btn-clipboard'); const button = vm.$el.querySelector('.btn-clipboard');
expect(button).not.toBe(null); expect(button).not.toBe(null);
...@@ -332,7 +310,7 @@ describe('diff_file_header', () => { ...@@ -332,7 +310,7 @@ describe('diff_file_header', () => {
it('it displays old and new file mode if it changed', () => { it('it displays old and new file mode if it changed', () => {
props.diffFile.modeChanged = true; props.diffFile.modeChanged = true;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
const { fileMode } = vm.$refs; const { fileMode } = vm.$refs;
expect(fileMode).not.toBe(undefined); expect(fileMode).not.toBe(undefined);
...@@ -343,7 +321,7 @@ describe('diff_file_header', () => { ...@@ -343,7 +321,7 @@ describe('diff_file_header', () => {
it('does not display the file mode if it has not changed', () => { it('does not display the file mode if it has not changed', () => {
props.diffFile.modeChanged = false; props.diffFile.modeChanged = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
const { fileMode } = vm.$refs; const { fileMode } = vm.$refs;
expect(fileMode).toBe(undefined); expect(fileMode).toBe(undefined);
...@@ -359,7 +337,7 @@ describe('diff_file_header', () => { ...@@ -359,7 +337,7 @@ describe('diff_file_header', () => {
externalStorage: 'lfs', externalStorage: 'lfs',
}); });
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(lfsLabel()).not.toBe(null); expect(lfsLabel()).not.toBe(null);
expect(lfsLabel()).toHaveText('LFS'); expect(lfsLabel()).toHaveText('LFS');
...@@ -368,7 +346,7 @@ describe('diff_file_header', () => { ...@@ -368,7 +346,7 @@ describe('diff_file_header', () => {
it('does not display the LFS label for files stored in repository', () => { it('does not display the LFS label for files stored in repository', () => {
props.diffFile.storedExternally = false; props.diffFile.storedExternally = false;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(lfsLabel()).toBe(null); expect(lfsLabel()).toBe(null);
}); });
...@@ -376,7 +354,7 @@ describe('diff_file_header', () => { ...@@ -376,7 +354,7 @@ describe('diff_file_header', () => {
describe('edit button', () => { describe('edit button', () => {
it('should not render edit button if addMergeRequestButtons is not true', () => { it('should not render edit button if addMergeRequestButtons is not true', () => {
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.$el.querySelector('.js-edit-blob')).toEqual(null); expect(vm.$el.querySelector('.js-edit-blob')).toEqual(null);
}); });
...@@ -384,7 +362,7 @@ describe('diff_file_header', () => { ...@@ -384,7 +362,7 @@ describe('diff_file_header', () => {
it('should show edit button when file is editable', () => { it('should show edit button when file is editable', () => {
props.addMergeRequestButtons = true; props.addMergeRequestButtons = true;
props.diffFile.editPath = '/'; props.diffFile.editPath = '/';
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.$el.querySelector('.js-edit-blob')).toContainText('Edit'); expect(vm.$el.querySelector('.js-edit-blob')).toContainText('Edit');
}); });
...@@ -393,7 +371,7 @@ describe('diff_file_header', () => { ...@@ -393,7 +371,7 @@ describe('diff_file_header', () => {
props.addMergeRequestButtons = true; props.addMergeRequestButtons = true;
props.diffFile.deletedFile = true; props.diffFile.deletedFile = true;
props.diffFile.editPath = '/'; props.diffFile.editPath = '/';
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.$el.querySelector('.js-edit-blob')).toEqual(null); expect(vm.$el.querySelector('.js-edit-blob')).toEqual(null);
}); });
...@@ -413,7 +391,7 @@ describe('diff_file_header', () => { ...@@ -413,7 +391,7 @@ describe('diff_file_header', () => {
props.diffFile.externalUrl = url; props.diffFile.externalUrl = url;
props.diffFile.formattedExternalUrl = title; props.diffFile.formattedExternalUrl = title;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.$el.querySelector(`a[href="${url}"]`)).not.toBe(null); expect(vm.$el.querySelector(`a[href="${url}"]`)).not.toBe(null);
expect(vm.$el.querySelector(`a[data-original-title="View on ${title}"]`)).not.toBe(null); expect(vm.$el.querySelector(`a[data-original-title="View on ${title}"]`)).not.toBe(null);
...@@ -423,11 +401,39 @@ describe('diff_file_header', () => { ...@@ -423,11 +401,39 @@ describe('diff_file_header', () => {
props.diffFile.externalUrl = ''; props.diffFile.externalUrl = '';
props.diffFile.formattedExternalUrl = title; props.diffFile.formattedExternalUrl = title;
vm = mountComponent(Component, props); vm = mountComponentWithStore(Component, { props, store });
expect(vm.$el.querySelector(`a[data-original-title="View on ${title}"]`)).toBe(null); expect(vm.$el.querySelector(`a[data-original-title="View on ${title}"]`)).toBe(null);
}); });
}); });
}); });
describe('handles toggle discussions', () => {
it('dispatches toggleFileDiscussions when user clicks on toggle discussions button', () => {
const propsCopy = Object.assign({}, props);
propsCopy.diffFile.submodule = false;
propsCopy.diffFile.blob = {
id: '848ed9407c6730ff16edb3dd24485a0eea24292a',
path: 'lib/base.js',
name: 'base.js',
mode: '100644',
readableText: true,
icon: 'file-text-o',
};
propsCopy.addMergeRequestButtons = true;
propsCopy.diffFile.deletedFile = true;
vm = mountComponentWithStore(Component, {
props: propsCopy,
store,
});
spyOn(vm, 'toggleFileDiscussions');
vm.$el.querySelector('.js-btn-vue-toggle-comments').click();
expect(vm.toggleFileDiscussions).toHaveBeenCalled();
});
});
}); });
}); });
...@@ -191,4 +191,48 @@ describe('DiffsStoreActions', () => { ...@@ -191,4 +191,48 @@ describe('DiffsStoreActions', () => {
); );
}); });
}); });
describe('toggleFileDiscussions', () => {
it('should dispatch collapseDiscussion when all discussions are expanded', () => {
const getters = {
getDiffFileDiscussions: jasmine.createSpy().and.returnValue([{ id: 1 }]),
diffHasAllExpandedDiscussions: jasmine.createSpy().and.returnValue(true),
diffHasAllCollpasedDiscussions: jasmine.createSpy().and.returnValue(false),
};
const dispatch = jasmine.createSpy('dispatch');
actions.toggleFileDiscussions({ getters, dispatch });
expect(dispatch).toHaveBeenCalledWith('collapseDiscussion', { discussionId: 1 }, { root: true });
});
it('should dispatch expandDiscussion when all discussions are collapsed', () => {
const getters = {
getDiffFileDiscussions: jasmine.createSpy().and.returnValue([{ id: 1 }]),
diffHasAllExpandedDiscussions: jasmine.createSpy().and.returnValue(false),
diffHasAllCollpasedDiscussions: jasmine.createSpy().and.returnValue(true),
};
const dispatch = jasmine.createSpy();
actions.toggleFileDiscussions({ getters, dispatch });
expect(dispatch).toHaveBeenCalledWith('expandDiscussion', { discussionId: 1 }, { root: true });
});
it('should dispatch expandDiscussion when some discussions are collapsed and others are expanded for the collapsed discussion', () => {
const getters = {
getDiffFileDiscussions: jasmine.createSpy().and.returnValue([{ expanded: false, id: 1 }]),
diffHasAllExpandedDiscussions: jasmine.createSpy().and.returnValue(false),
diffHasAllCollpasedDiscussions: jasmine.createSpy().and.returnValue(false),
};
const dispatch = jasmine.createSpy();
actions.toggleFileDiscussions({ getters, dispatch });
expect(dispatch).toHaveBeenCalledWith('expandDiscussion', { discussionId: 1 }, { root: true });
});
});
}); });
import * as getters from '~/diffs/store/getters'; import * as getters from '~/diffs/store/getters';
import state from '~/diffs/store/modules/diff_state'; import state from '~/diffs/store/modules/diff_state';
import { PARALLEL_DIFF_VIEW_TYPE, INLINE_DIFF_VIEW_TYPE } from '~/diffs/constants'; import { PARALLEL_DIFF_VIEW_TYPE, INLINE_DIFF_VIEW_TYPE } from '~/diffs/constants';
import discussion from '../mock_data/diff_discussions';
describe('DiffsStoreGetters', () => { describe('Diffs Module Getters', () => {
let localState; let localState;
let discussionMock;
let discussionMock1;
const diffFileMock = {
fileHash: '9732849daca6ae818696d9575f5d1207d1a7f8bb',
};
beforeEach(() => { beforeEach(() => {
localState = state(); localState = state();
discussionMock = Object.assign({}, discussion);
discussionMock.diff_file.file_hash = diffFileMock.fileHash;
discussionMock1 = Object.assign({}, discussion);
discussionMock1.diff_file.file_hash = diffFileMock.fileHash;
}); });
describe('isParallelView', () => { describe('isParallelView', () => {
...@@ -63,4 +75,113 @@ describe('DiffsStoreGetters', () => { ...@@ -63,4 +75,113 @@ describe('DiffsStoreGetters', () => {
expect(getters.commitId(localState)).toEqual(null); expect(getters.commitId(localState)).toEqual(null);
}); });
}); });
describe('diffHasAllExpandedDiscussions', () => {
it('returns true when all discussions are expanded', () => {
expect(
getters.diffHasAllExpandedDiscussions(localState, {
getDiffFileDiscussions: () => [discussionMock, discussionMock],
})(diffFileMock),
).toEqual(true);
});
it('returns false when there are no discussions', () => {
expect(
getters.diffHasAllExpandedDiscussions(localState, {
getDiffFileDiscussions: () => [],
})(diffFileMock),
).toEqual(false);
});
it('returns false when one discussions is collapsed', () => {
discussionMock1.expanded = false;
expect(
getters.diffHasAllExpandedDiscussions(localState, {
getDiffFileDiscussions: () => [discussionMock, discussionMock1],
})(diffFileMock),
).toEqual(false);
});
});
describe('diffHasAllCollpasedDiscussions', () => {
it('returns true when all discussions are collapsed', () => {
discussionMock.diff_file.file_hash = diffFileMock.fileHash;
discussionMock.expanded = false;
expect(
getters.diffHasAllCollpasedDiscussions(localState, {
getDiffFileDiscussions: () => [discussionMock],
})(diffFileMock),
).toEqual(true);
});
it('returns false when there are no discussions', () => {
expect(
getters.diffHasAllCollpasedDiscussions(localState, {
getDiffFileDiscussions: () => [],
})(diffFileMock),
).toEqual(false);
});
it('returns false when one discussions is expanded', () => {
discussionMock1.expanded = false;
expect(
getters.diffHasAllCollpasedDiscussions(localState, {
getDiffFileDiscussions: () => [discussionMock, discussionMock1],
})(diffFileMock),
).toEqual(false);
});
});
describe('diffHasExpandedDiscussions', () => {
it('returns true when one of the discussions is expanded', () => {
discussionMock1.expanded = false;
expect(
getters.diffHasExpandedDiscussions(localState, {
getDiffFileDiscussions: () => [discussionMock, discussionMock],
})(diffFileMock),
).toEqual(true);
});
it('returns false when there are no discussions', () => {
expect(
getters.diffHasExpandedDiscussions(localState, { getDiffFileDiscussions: () => [] })(
diffFileMock,
),
).toEqual(false);
});
it('returns false when no discussion is expanded', () => {
discussionMock.expanded = false;
discussionMock1.expanded = false;
expect(
getters.diffHasExpandedDiscussions(localState, {
getDiffFileDiscussions: () => [discussionMock, discussionMock1],
})(diffFileMock),
).toEqual(false);
});
});
describe('getDiffFileDiscussions', () => {
it('returns an array with discussions when fileHash matches and the discussion belongs to a diff', () => {
discussionMock.diff_file.file_hash = diffFileMock.fileHash;
expect(
getters.getDiffFileDiscussions(localState, {}, {}, { discussions: [discussionMock] })(
diffFileMock,
).length,
).toEqual(1);
});
it('returns an empty array when no discussions are found in the given diff', () => {
expect(
getters.getDiffFileDiscussions(localState, {}, {}, { discussions: [] })(diffFileMock)
.length,
).toEqual(0);
});
});
}); });
...@@ -128,6 +128,19 @@ describe('Actions Notes Store', () => { ...@@ -128,6 +128,19 @@ describe('Actions Notes Store', () => {
}); });
}); });
describe('collapseDiscussion', () => {
it('should commit collapse discussion', done => {
testAction(
actions.collapseDiscussion,
{ discussionId: discussionMock.id },
{ notes: [discussionMock] },
[{ type: 'COLLAPSE_DISCUSSION', payload: { discussionId: discussionMock.id } }],
[],
done,
);
});
});
describe('async methods', () => { describe('async methods', () => {
const interceptor = (request, next) => { const interceptor = (request, next) => {
next( next(
......
...@@ -74,6 +74,20 @@ describe('Notes Store mutations', () => { ...@@ -74,6 +74,20 @@ describe('Notes Store mutations', () => {
}); });
}); });
describe('COLLAPSE_DISCUSSION', () => {
it('should collpase an expanded discussion', () => {
const discussion = Object.assign({}, discussionMock, { expanded: true });
const state = {
discussions: [discussion],
};
mutations.COLLAPSE_DISCUSSION(state, { discussionId: discussion.id });
expect(state.discussions[0].expanded).toEqual(false);
});
});
describe('REMOVE_PLACEHOLDER_NOTES', () => { describe('REMOVE_PLACEHOLDER_NOTES', () => {
it('should remove all placeholder notes in indivudal notes and discussion', () => { it('should remove all placeholder notes in indivudal notes and discussion', () => {
const placeholderNote = Object.assign({}, individualNote, { isPlaceholderNote: true }); const placeholderNote = Object.assign({}, individualNote, { isPlaceholderNote: true });
......
...@@ -70,4 +70,19 @@ describe Gitlab::Kubernetes do ...@@ -70,4 +70,19 @@ describe Gitlab::Kubernetes do
it { is_expected.to eq(YAML.load_file(path)) } it { is_expected.to eq(YAML.load_file(path)) }
end end
end end
describe '#add_terminal_auth' do
it 'adds authentication parameters to a hash' do
terminal = { original: 'value' }
add_terminal_auth(terminal, token: 'foo', max_session_time: 0, ca_pem: 'bar')
expect(terminal).to eq(
original: 'value',
headers: { 'Authorization' => ['Bearer foo'] },
max_session_time: 0,
ca_pem: 'bar'
)
end
end
end end
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