Commit 1ca3a787 authored by Kushal Pandya's avatar Kushal Pandya Committed by Jarka Kadlecová

[ci skip] Enable todo toggle action on Epic sidebar

parent a88097b1
...@@ -30,18 +30,17 @@ export default { ...@@ -30,18 +30,17 @@ export default {
required: false, required: false,
default: true, default: true,
}, },
isActionActive: {
type: Boolean,
required: false,
default: false,
},
collapsed: { collapsed: {
type: Boolean, type: Boolean,
required: false, required: false,
default: false, default: false,
}, },
}, },
data() {
return {
isButtonTypeTodo: this.isTodo,
isActionActive: false,
};
},
computed: { computed: {
buttonClasses() { buttonClasses() {
return this.collapsed ? return this.collapsed ?
...@@ -49,13 +48,18 @@ export default { ...@@ -49,13 +48,18 @@ export default {
'btn btn-default btn-todo issuable-header-btn float-right'; 'btn btn-default btn-todo issuable-header-btn float-right';
}, },
buttonLabel() { buttonLabel() {
return this.isButtonTypeTodo ? MARK_TEXT : TODO_TEXT; return this.isTodo ? MARK_TEXT : TODO_TEXT;
}, },
collapsedButtonIconClasses() { collapsedButtonIconClasses() {
return this.isButtonTypeTodo ? 'todo-undone' : ''; return this.isTodo ? 'todo-undone' : '';
}, },
collapsedButtonIcon() { collapsedButtonIcon() {
return this.isButtonTypeTodo ? 'check-circle' : 'plus-square'; return this.isTodo ? 'check-circle' : 'plus-square';
},
},
methods: {
handleButtonClick() {
this.$emit('toggleTodo');
}, },
}, },
}; };
...@@ -73,6 +77,7 @@ export default { ...@@ -73,6 +77,7 @@ export default {
data-container="body" data-container="body"
data-placement="left" data-placement="left"
data-boundary="viewport" data-boundary="viewport"
@click="handleButtonClick"
> >
<icon <icon
v-if="collapsed" v-if="collapsed"
......
...@@ -119,6 +119,15 @@ ...@@ -119,6 +119,15 @@
type: String, type: String,
required: true, required: true,
}, },
todoPath: {
type: String,
required: true,
},
todoDeletePath: {
type: String,
required: false,
default: '',
},
labelsWebUrl: { labelsWebUrl: {
type: String, type: String,
required: true, required: true,
...@@ -191,6 +200,8 @@ ...@@ -191,6 +200,8 @@
:update-path="updateEndpoint" :update-path="updateEndpoint"
:labels-path="labelsPath" :labels-path="labelsPath"
:toggle-subscription-path="toggleSubscriptionPath" :toggle-subscription-path="toggleSubscriptionPath"
:todo-path="todoPath"
:todo-delete-path="todoDeletePath"
:labels-web-url="labelsWebUrl" :labels-web-url="labelsWebUrl"
:epics-web-url="epicsWebUrl" :epics-web-url="epicsWebUrl"
/> />
......
import Vue from 'vue'; import Vue from 'vue';
import { convertObjectPropsToCamelCase } from '~/lib/utils/common_utils';
import EpicShowApp from './components/epic_show_app.vue'; import EpicShowApp from './components/epic_show_app.vue';
export default () => { export default () => {
const el = document.querySelector('#epic-show-app'); const el = document.querySelector('#epic-show-app');
const metaData = JSON.parse(el.dataset.meta); const metaData = convertObjectPropsToCamelCase(JSON.parse(el.dataset.meta));
const initialData = JSON.parse(el.dataset.initial); const initialData = JSON.parse(el.dataset.initial);
const props = Object.assign({}, initialData, metaData, el.dataset); const props = Object.assign({}, initialData, metaData, el.dataset);
// Convert backend casing to match frontend style guide
props.epicId = props.epic_id;
props.todoExists = props.todo_exists;
props.startDate = props.start_date;
props.endDate = props.end_date;
return new Vue({ return new Vue({
el, el,
components: { components: {
......
<script> <script>
/* eslint-disable vue/require-default-prop */ /* eslint-disable vue/require-default-prop */
import $ from 'jquery';
import _ from 'underscore'; import _ from 'underscore';
import Cookies from 'js-cookie'; import Cookies from 'js-cookie';
import Flash from '~/flash'; import Flash from '~/flash';
...@@ -82,6 +83,14 @@ export default { ...@@ -82,6 +83,14 @@ export default {
type: String, type: String,
required: true, required: true,
}, },
todoPath: {
type: String,
required: true,
},
todoDeletePath: {
type: String,
required: true,
},
labelsWebUrl: { labelsWebUrl: {
type: String, type: String,
required: true, required: true,
...@@ -96,6 +105,8 @@ export default { ...@@ -96,6 +105,8 @@ export default {
startDate: this.initialStartDate, startDate: this.initialStartDate,
endDate: this.initialEndDate, endDate: this.initialEndDate,
subscribed: this.initialSubscribed, subscribed: this.initialSubscribed,
todoExists: this.initialTodoExists,
todoDeletePath: this.todoDeletePath,
}); });
return { return {
...@@ -107,7 +118,12 @@ export default { ...@@ -107,7 +118,12 @@ export default {
savingStartDate: false, savingStartDate: false,
savingEndDate: false, savingEndDate: false,
savingSubscription: false, savingSubscription: false,
service: new SidebarService(this.endpoint, this.toggleSubscriptionPath), savingTodoAction: false,
service: new SidebarService({
endpoint: this.endpoint,
subscriptionEndpoint: this.subscriptionEndpoint,
todoPath: this.todoPath,
}),
epicContext: { epicContext: {
labels: this.initialLabels, labels: this.initialLabels,
}, },
...@@ -202,6 +218,37 @@ export default { ...@@ -202,6 +218,37 @@ export default {
} }
}); });
}, },
handleToggleTodo() {
this.savingTodoAction = true;
if (!this.store.todoExists) {
this.service
.addTodo(this.epicId)
.then(res => res.data)
.then(data => {
this.savingTodoAction = false;
this.store.setTodoDeletePath(data.delete_path);
this.store.setTodoExists(!this.store.todoExists);
$(document).trigger('todo:toggle', data.count);
})
.catch(() => {
this.savingTodoAction = false;
Flash(__('There was an error adding a todo.'));
});
} else {
this.service
.deleteTodo(this.store.todoDeletePath)
.then(res => res.data)
.then(data => {
this.savingTodoAction = false;
this.store.setTodoExists(!this.store.todoExists);
$(document).trigger('todo:toggle', data.count);
})
.catch(() => {
this.savingTodoAction = false;
Flash(__('There was an error deleting the todo.'));
});
}
},
}, },
}; };
</script> </script>
...@@ -226,8 +273,10 @@ export default { ...@@ -226,8 +273,10 @@ export default {
v-if="!collapsed" v-if="!collapsed"
:collapsed="collapsed" :collapsed="collapsed"
:issuable-id="epicId" :issuable-id="epicId"
:is-todo="initialTodoExists" :is-todo="store.todoExists"
:is-action-active="savingTodoAction"
issuable-type="epic" issuable-type="epic"
@toggleTodo="handleToggleTodo"
/> />
</div> </div>
<div <div
...@@ -237,8 +286,10 @@ export default { ...@@ -237,8 +286,10 @@ export default {
<sidebar-todo <sidebar-todo
:collapsed="collapsed" :collapsed="collapsed"
:issuable-id="epicId" :issuable-id="epicId"
:is-todo="initialTodoExists" :is-todo="store.todoExists"
:is-action-active="savingTodoAction"
issuable-type="epic" issuable-type="epic"
@toggleTodo="handleToggleTodo"
/> />
</div> </div>
<sidebar-date-picker <sidebar-date-picker
......
import axios from '~/lib/utils/axios_utils'; import axios from '~/lib/utils/axios_utils';
export default class SidebarService { export default class SidebarService {
constructor(endpoint, subscriptionEndpoint) { constructor({ endpoint, subscriptionEndpoint, todoPath }) {
this.endpoint = endpoint; this.endpoint = endpoint;
this.subscriptionEndpoint = subscriptionEndpoint; this.subscriptionEndpoint = subscriptionEndpoint;
this.todoPath = todoPath;
} }
updateStartDate(startDate) { updateStartDate(startDate) {
...@@ -17,4 +18,16 @@ export default class SidebarService { ...@@ -17,4 +18,16 @@ export default class SidebarService {
toggleSubscribed() { toggleSubscribed() {
return axios.post(this.subscriptionEndpoint); return axios.post(this.subscriptionEndpoint);
} }
addTodo(epicId) {
return axios.post(this.todoPath, {
issuable_id: epicId,
issuable_type: 'epic',
});
}
// eslint-disable-next-line class-methods-use-this
deleteTodo(todoDeletePath) {
return axios.delete(todoDeletePath);
}
} }
import { parsePikadayDate } from '~/lib/utils/datefix'; import { parsePikadayDate } from '~/lib/utils/datefix';
export default class SidebarStore { export default class SidebarStore {
constructor({ startDate, endDate, subscribed }) { constructor({ startDate, endDate, subscribed, todoExists, todoDeletePath }) {
this.startDate = startDate; this.startDate = startDate;
this.endDate = endDate; this.endDate = endDate;
this.subscribed = subscribed; this.subscribed = subscribed;
this.todoExists = todoExists;
this.todoDeletePath = todoDeletePath;
} }
get startDateTime() { get startDateTime() {
...@@ -18,4 +20,12 @@ export default class SidebarStore { ...@@ -18,4 +20,12 @@ export default class SidebarStore {
setSubscribed(subscribed) { setSubscribed(subscribed) {
this.subscribed = subscribed; this.subscribed = subscribed;
} }
setTodoExists(todoExists) {
this.todoExists = todoExists;
}
setTodoDeletePath(deletePath) {
this.todoDeletePath = deletePath;
}
} }
...@@ -14,6 +14,7 @@ module EpicsHelper ...@@ -14,6 +14,7 @@ module EpicsHelper
src: opts[:author_icon] src: opts[:author_icon]
}, },
todo_exists: todo.present?, todo_exists: todo.present?,
todo_path: group_todos_path(group),
start_date: epic.start_date, start_date: epic.start_date,
end_date: epic.end_date end_date: epic.end_date
} }
......
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