Commit edcd45b5 authored by GitLab Bot's avatar GitLab Bot

Automatic merge of gitlab-org/gitlab-ce master

parents 5d21fa1d 751c8c8e
<script> <script>
import { mapGetters } from 'vuex'; import { mapGetters } from 'vuex';
import draftCommentsMixin from 'ee_else_ce/diffs/mixins/draft_comments';
import inlineDiffTableRow from './inline_diff_table_row.vue'; import inlineDiffTableRow from './inline_diff_table_row.vue';
import inlineDiffCommentRow from './inline_diff_comment_row.vue'; import inlineDiffCommentRow from './inline_diff_comment_row.vue';
...@@ -12,6 +13,7 @@ export default { ...@@ -12,6 +13,7 @@ export default {
inlineDiffTableRow, inlineDiffTableRow,
InlineDraftCommentRow, InlineDraftCommentRow,
}, },
mixins: [draftCommentsMixin],
props: { props: {
diffFile: { diffFile: {
type: Object, type: Object,
......
export default {
computed: {
shouldRenderDraftRow: () => () => false,
draftForLine: () => () => ({}),
},
};
...@@ -3,7 +3,7 @@ import { mapActions } from 'vuex'; ...@@ -3,7 +3,7 @@ import { mapActions } from 'vuex';
import Translate from '../vue_shared/translate'; import Translate from '../vue_shared/translate';
import ImportProjectsTable from './components/import_projects_table.vue'; import ImportProjectsTable from './components/import_projects_table.vue';
import { parseBoolean } from '../lib/utils/common_utils'; import { parseBoolean } from '../lib/utils/common_utils';
import store from './store'; import createStore from './store';
Vue.use(Translate); Vue.use(Translate);
...@@ -20,6 +20,7 @@ export default function mountImportProjectsTable(mountElement) { ...@@ -20,6 +20,7 @@ export default function mountImportProjectsTable(mountElement) {
ciCdOnly, ciCdOnly,
} = mountElement.dataset; } = mountElement.dataset;
const store = createStore();
return new Vue({ return new Vue({
el: mountElement, el: mountElement,
store, store,
......
...@@ -7,9 +7,10 @@ import mutations from './mutations'; ...@@ -7,9 +7,10 @@ import mutations from './mutations';
Vue.use(Vuex); Vue.use(Vuex);
export default new Vuex.Store({ export default () =>
state: state(), new Vuex.Store({
actions, state: state(),
mutations, actions,
getters, mutations,
}); getters,
});
import $ from 'jquery';
import Vue from 'vue'; import Vue from 'vue';
import { mapActions, mapState, mapGetters } from 'vuex'; import store from 'ee_else_ce/mr_notes/stores';
import initNotesApp from './init_notes';
import initDiffsApp from '../diffs'; import initDiffsApp from '../diffs';
import notesApp from '../notes/components/notes_app.vue';
import discussionCounter from '../notes/components/discussion_counter.vue'; import discussionCounter from '../notes/components/discussion_counter.vue';
import initDiscussionFilters from '../notes/discussion_filters'; import initDiscussionFilters from '../notes/discussion_filters';
import store from './stores';
import MergeRequest from '../merge_request'; import MergeRequest from '../merge_request';
import { resetServiceWorkersPublicPath } from '../lib/utils/webpack'; import { resetServiceWorkersPublicPath } from '../lib/utils/webpack';
...@@ -18,68 +16,7 @@ export default function initMrNotes() { ...@@ -18,68 +16,7 @@ export default function initMrNotes() {
action: mrShowNode.dataset.mrAction, action: mrShowNode.dataset.mrAction,
}); });
// eslint-disable-next-line no-new initNotesApp();
new Vue({
el: '#js-vue-mr-discussions',
name: 'MergeRequestDiscussions',
components: {
notesApp,
},
store,
data() {
const notesDataset = document.getElementById('js-vue-mr-discussions').dataset;
const noteableData = JSON.parse(notesDataset.noteableData);
noteableData.noteableType = notesDataset.noteableType;
noteableData.targetType = notesDataset.targetType;
return {
noteableData,
currentUserData: JSON.parse(notesDataset.currentUserData),
notesData: JSON.parse(notesDataset.notesData),
helpPagePath: notesDataset.helpPagePath,
};
},
computed: {
...mapGetters(['discussionTabCounter']),
...mapState({
activeTab: state => state.page.activeTab,
}),
},
watch: {
discussionTabCounter() {
this.updateDiscussionTabCounter();
},
},
created() {
this.setActiveTab(window.mrTabs.getCurrentAction());
},
mounted() {
this.notesCountBadge = $('.issuable-details').find('.notes-tab .badge');
$(document).on('visibilitychange', this.updateDiscussionTabCounter);
window.mrTabs.eventHub.$on('MergeRequestTabChange', this.setActiveTab);
},
beforeDestroy() {
$(document).off('visibilitychange', this.updateDiscussionTabCounter);
window.mrTabs.eventHub.$off('MergeRequestTabChange', this.setActiveTab);
},
methods: {
...mapActions(['setActiveTab']),
updateDiscussionTabCounter() {
this.notesCountBadge.text(this.discussionTabCounter);
},
},
render(createElement) {
return createElement('notes-app', {
props: {
noteableData: this.noteableData,
notesData: this.notesData,
userData: this.currentUserData,
shouldShow: this.activeTab === 'show',
helpPagePath: this.helpPagePath,
},
});
},
});
// eslint-disable-next-line no-new // eslint-disable-next-line no-new
new Vue({ new Vue({
......
import $ from 'jquery';
import Vue from 'vue';
import { mapActions, mapState, mapGetters } from 'vuex';
import store from 'ee_else_ce/mr_notes/stores';
import notesApp from '../notes/components/notes_app.vue';
export default () => {
// eslint-disable-next-line no-new
new Vue({
el: '#js-vue-mr-discussions',
name: 'MergeRequestDiscussions',
components: {
notesApp,
},
store,
data() {
const notesDataset = document.getElementById('js-vue-mr-discussions').dataset;
const noteableData = JSON.parse(notesDataset.noteableData);
noteableData.noteableType = notesDataset.noteableType;
noteableData.targetType = notesDataset.targetType;
return {
noteableData,
currentUserData: JSON.parse(notesDataset.currentUserData),
notesData: JSON.parse(notesDataset.notesData),
helpPagePath: notesDataset.helpPagePath,
};
},
computed: {
...mapGetters(['discussionTabCounter']),
...mapState({
activeTab: state => state.page.activeTab,
}),
},
watch: {
discussionTabCounter() {
this.updateDiscussionTabCounter();
},
},
created() {
this.setActiveTab(window.mrTabs.getCurrentAction());
},
mounted() {
this.notesCountBadge = $('.issuable-details').find('.notes-tab .badge');
$(document).on('visibilitychange', this.updateDiscussionTabCounter);
window.mrTabs.eventHub.$on('MergeRequestTabChange', this.setActiveTab);
},
beforeDestroy() {
$(document).off('visibilitychange', this.updateDiscussionTabCounter);
window.mrTabs.eventHub.$off('MergeRequestTabChange', this.setActiveTab);
},
methods: {
...mapActions(['setActiveTab']),
updateDiscussionTabCounter() {
this.notesCountBadge.text(this.discussionTabCounter);
},
},
render(createElement) {
return createElement('notes-app', {
props: {
noteableData: this.noteableData,
notesData: this.notesData,
userData: this.currentUserData,
shouldShow: this.activeTab === 'show',
helpPagePath: this.helpPagePath,
},
});
},
});
};
...@@ -93,23 +93,22 @@ function UsersSelect(currentUser, els, options = {}) { ...@@ -93,23 +93,22 @@ function UsersSelect(currentUser, els, options = {}) {
} }
// Save current selected user to the DOM // Save current selected user to the DOM
const input = document.createElement('input'); const currentUserInfo = $dropdown.data('currentUserInfo') || {};
input.type = 'hidden'; const currentUser = _this.currentUser || {};
input.name = $dropdown.data('fieldName'); const fieldName = $dropdown.data('fieldName');
const userName = currentUserInfo.name;
const currentUserInfo = $dropdown.data('currentUserInfo'); const userId = currentUserInfo.id || currentUser.id;
if (currentUserInfo) { const inputHtmlString = _.template(`
input.value = currentUserInfo.id; <input type="hidden" name="<%- fieldName %>"
input.dataset.meta = _.escape(currentUserInfo.name); data-meta="<%- userName %>"
} else if (_this.currentUser) { value="<%- userId %>" />
input.value = _this.currentUser.id; `)({ fieldName, userName, userId });
}
if ($selectbox) { if ($selectbox) {
$dropdown.parent().before(input); $dropdown.parent().before(inputHtmlString);
} else { } else {
$dropdown.after(input); $dropdown.after(inputHtmlString);
} }
}; };
......
---
title: Fix username escaping when using assign to me for issues
merge_request: 24673
author:
type: fixed
...@@ -93,4 +93,22 @@ describe "User creates issue" do ...@@ -93,4 +93,22 @@ describe "User creates issue" do
end end
end end
end end
context "when signed in as user with special characters in their name" do
let(:user_special) { create(:user, name: "Jon O'Shea") }
before do
project.add_developer(user_special)
sign_in(user_special)
visit(new_project_issue_path(project))
end
it "will correctly escape user names with an apostrophe when clicking 'Assign to me'", :js do
first('.assign-to-me-link').click
expect(page).to have_content(user_special.name)
expect(page.find('input[name="issue[assignee_ids][]"]', visible: false)['data-meta']).to eq(user_special.name)
end
end
end end
import Vue from 'vue'; import Vue from 'vue';
import DiffContentComponent from '~/diffs/components/diff_content.vue'; import DiffContentComponent from '~/diffs/components/diff_content.vue';
import { createStore } from '~/mr_notes/stores'; import { createStore } from 'ee_else_ce/mr_notes/stores';
import { mountComponentWithStore } from 'spec/helpers/vue_mount_component_helper'; import { mountComponentWithStore } from 'spec/helpers/vue_mount_component_helper';
import { GREEN_BOX_IMAGE_URL, RED_BOX_IMAGE_URL } from 'spec/test_constants'; import { GREEN_BOX_IMAGE_URL, RED_BOX_IMAGE_URL } from 'spec/test_constants';
import '~/behaviors/markdown/render_gfm'; import '~/behaviors/markdown/render_gfm';
......
import Vue from 'vue'; import Vue from 'vue';
import DiffFileComponent from '~/diffs/components/diff_file.vue'; import DiffFileComponent from '~/diffs/components/diff_file.vue';
import { diffViewerModes, diffViewerErrors } from '~/ide/constants'; import { diffViewerModes, diffViewerErrors } from '~/ide/constants';
import store from '~/mr_notes/stores'; import store from 'ee_else_ce/mr_notes/stores';
import { createComponentWithStore } from 'spec/helpers/vue_mount_component_helper'; import { createComponentWithStore } from 'spec/helpers/vue_mount_component_helper';
import diffFileMockData from '../mock_data/diff_file'; import diffFileMockData from '../mock_data/diff_file';
......
import Vue from 'vue'; import Vue from 'vue';
import '~/behaviors/markdown/render_gfm'; import '~/behaviors/markdown/render_gfm';
import InlineDiffView from '~/diffs/components/inline_diff_view.vue'; import InlineDiffView from '~/diffs/components/inline_diff_view.vue';
import store from '~/mr_notes/stores'; import store from 'ee_else_ce/mr_notes/stores';
import { createComponentWithStore } from 'spec/helpers/vue_mount_component_helper'; import { createComponentWithStore } from 'spec/helpers/vue_mount_component_helper';
import diffFileMockData from '../mock_data/diff_file'; import diffFileMockData from '../mock_data/diff_file';
import discussionsMockData from '../mock_data/diff_discussions'; import discussionsMockData from '../mock_data/diff_discussions';
......
import Vue from 'vue'; import Vue from 'vue';
import ParallelDiffView from '~/diffs/components/parallel_diff_view.vue'; import ParallelDiffView from '~/diffs/components/parallel_diff_view.vue';
import store from '~/mr_notes/stores'; import store from 'ee_else_ce/mr_notes/stores';
import * as constants from '~/diffs/constants'; import * as constants from '~/diffs/constants';
import { createComponentWithStore } from 'spec/helpers/vue_mount_component_helper'; import { createComponentWithStore } from 'spec/helpers/vue_mount_component_helper';
import diffFileMockData from '../mock_data/diff_file'; import diffFileMockData from '../mock_data/diff_file';
...@@ -18,6 +18,10 @@ describe('ParallelDiffView', () => { ...@@ -18,6 +18,10 @@ describe('ParallelDiffView', () => {
}).$mount(); }).$mount();
}); });
afterEach(() => {
component.$destroy();
});
describe('assigned', () => { describe('assigned', () => {
describe('diffLines', () => { describe('diffLines', () => {
it('should normalize lines for empty cells', () => { it('should normalize lines for empty cells', () => {
......
import Vue from 'vue'; import Vue from 'vue';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import store from '~/import_projects/store'; import createStore from '~/import_projects/store';
import importProjectsTable from '~/import_projects/components/import_projects_table.vue'; import importProjectsTable from '~/import_projects/components/import_projects_table.vue';
import STATUS_MAP from '~/import_projects/constants'; import STATUS_MAP from '~/import_projects/constants';
import setTimeoutPromise from '../../helpers/set_timeout_promise_helper'; import setTimeoutPromise from '../../helpers/set_timeout_promise_helper';
...@@ -9,6 +9,7 @@ import setTimeoutPromise from '../../helpers/set_timeout_promise_helper'; ...@@ -9,6 +9,7 @@ import setTimeoutPromise from '../../helpers/set_timeout_promise_helper';
describe('ImportProjectsTable', () => { describe('ImportProjectsTable', () => {
let vm; let vm;
let mock; let mock;
let store;
const reposPath = '/repos-path'; const reposPath = '/repos-path';
const jobsPath = '/jobs-path'; const jobsPath = '/jobs-path';
const providerTitle = 'THE PROVIDER'; const providerTitle = 'THE PROVIDER';
...@@ -31,12 +32,13 @@ describe('ImportProjectsTable', () => { ...@@ -31,12 +32,13 @@ describe('ImportProjectsTable', () => {
}, },
}).$mount(); }).$mount();
component.$store.dispatch('stopJobsPolling'); store.dispatch('stopJobsPolling');
return component; return component;
} }
beforeEach(() => { beforeEach(() => {
store = createStore();
store.dispatch('setInitialData', { reposPath }); store.dispatch('setInitialData', { reposPath });
mock = new MockAdapter(axios); mock = new MockAdapter(axios);
}); });
...@@ -167,7 +169,7 @@ describe('ImportProjectsTable', () => { ...@@ -167,7 +169,7 @@ describe('ImportProjectsTable', () => {
expect(vm.$el.querySelector(`.ic-status_${statusObject.icon}`)).not.toBeNull(); expect(vm.$el.querySelector(`.ic-status_${statusObject.icon}`)).not.toBeNull();
mock.onGet(jobsPath).replyOnce(200, updatedProjects); mock.onGet(jobsPath).replyOnce(200, updatedProjects);
return vm.$store.dispatch('restartJobsPolling'); return store.dispatch('restartJobsPolling');
}) })
.then(() => setTimeoutPromise()) .then(() => setTimeoutPromise())
.then(() => { .then(() => {
......
import Vue from 'vue'; import Vue from 'vue';
import store from '~/import_projects/store'; import createStore from '~/import_projects/store';
import importedProjectTableRow from '~/import_projects/components/imported_project_table_row.vue'; import importedProjectTableRow from '~/import_projects/components/imported_project_table_row.vue';
import STATUS_MAP from '~/import_projects/constants'; import STATUS_MAP from '~/import_projects/constants';
...@@ -16,6 +16,7 @@ describe('ImportedProjectTableRow', () => { ...@@ -16,6 +16,7 @@ describe('ImportedProjectTableRow', () => {
function createComponent() { function createComponent() {
const ImportedProjectTableRow = Vue.extend(importedProjectTableRow); const ImportedProjectTableRow = Vue.extend(importedProjectTableRow);
const store = createStore();
return new ImportedProjectTableRow({ return new ImportedProjectTableRow({
store, store,
propsData: { propsData: {
......
import Vue from 'vue'; import Vue from 'vue';
import MockAdapter from 'axios-mock-adapter'; import MockAdapter from 'axios-mock-adapter';
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
import store from '~/import_projects/store'; import createStore from '~/import_projects/store';
import providerRepoTableRow from '~/import_projects/components/provider_repo_table_row.vue'; import providerRepoTableRow from '~/import_projects/components/provider_repo_table_row.vue';
import STATUS_MAP, { STATUSES } from '~/import_projects/constants'; import STATUS_MAP, { STATUSES } from '~/import_projects/constants';
import setTimeoutPromise from '../../helpers/set_timeout_promise_helper'; import setTimeoutPromise from '../../helpers/set_timeout_promise_helper';
describe('ProviderRepoTableRow', () => { describe('ProviderRepoTableRow', () => {
let store;
let vm; let vm;
const repo = { const repo = {
id: 10, id: 10,
...@@ -28,6 +29,10 @@ describe('ProviderRepoTableRow', () => { ...@@ -28,6 +29,10 @@ describe('ProviderRepoTableRow', () => {
}).$mount(); }).$mount();
} }
beforeEach(() => {
store = createStore();
});
afterEach(() => { afterEach(() => {
vm.$destroy(); vm.$destroy();
}); });
......
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