Commit c5fd4fae authored by Eulyeon Ko's avatar Eulyeon Ko

Create a time tracker wrapper for swimlane sidebar

"board_sidebar_time_tracker.vue" wraps "time_tracker.vue"
(to be used in boards swimlane sidebar.)

Add "showCollapsed" prop to "time_tracker.vue".
Because the swimlanes sidebar is not collapsible,
"time-tracking-collapsed-state" should not be rendered
and "showCollapsed" is used to control this behavior in
"time_trakcer.vue"

Remove rootPath prop from time_tracker spec because
it isn't used in the component.

Remove "padding-bottom" and "border-bottom"
from ".time_tracker" class (these were unused)

Rename css class ".time_tracker" to ".time-tracker"
parent 48389a69
...@@ -90,6 +90,7 @@ export default () => { ...@@ -90,6 +90,7 @@ export default () => {
labelsFetchPath: $boardApp.dataset.labelsFetchPath, labelsFetchPath: $boardApp.dataset.labelsFetchPath,
labelsManagePath: $boardApp.dataset.labelsManagePath, labelsManagePath: $boardApp.dataset.labelsManagePath,
labelsFilterBasePath: $boardApp.dataset.labelsFilterBasePath, labelsFilterBasePath: $boardApp.dataset.labelsFilterBasePath,
timeTrackingLimitToHours: parseBoolean($boardApp.dataset.timeTrackingLimitToHours),
}, },
store, store,
apolloProvider, apolloProvider,
......
...@@ -44,6 +44,21 @@ export default { ...@@ -44,6 +44,21 @@ export default {
default: false, default: false,
required: false, required: false,
}, },
/*
In issue list, "time-tracking-collapsed-state" is always rendered even if the sidebar isn't collapsed.
The actual hiding is controlled with css classes:
Hide "time-tracking-collapsed-state"
if .right-sidebar .right-sidebar-collapsed .sidebar-collapsed-icon
Show "time-tracking-collapsed-state"
if .right-sidebar .right-sidebar-expanded .sidebar-collapsed-icon
In Swimlanes sidebar, we do not use collapsed state at all.
*/
showCollapsed: {
type: Boolean,
default: true,
required: false,
},
}, },
data() { data() {
return { return {
...@@ -93,8 +108,9 @@ export default { ...@@ -93,8 +108,9 @@ export default {
</script> </script>
<template> <template>
<div v-cloak class="time_tracker time-tracking-component-wrap"> <div v-cloak class="time-tracker time-tracking-component-wrap" data-testid="time-tracker">
<time-tracking-collapsed-state <time-tracking-collapsed-state
v-if="showCollapsed"
:show-comparison-state="showComparisonState" :show-comparison-state="showComparisonState"
:show-no-time-tracking-state="showNoTimeTrackingState" :show-no-time-tracking-state="showNoTimeTrackingState"
:show-help-state="showHelpState" :show-help-state="showHelpState"
...@@ -103,7 +119,7 @@ export default { ...@@ -103,7 +119,7 @@ export default {
:time-spent-human-readable="humanTimeSpent" :time-spent-human-readable="humanTimeSpent"
:time-estimate-human-readable="humanTimeEstimate" :time-estimate-human-readable="humanTimeEstimate"
/> />
<div class="title hide-collapsed"> <div class="title hide-collapsed gl-mb-3">
{{ __('Time tracking') }} {{ __('Time tracking') }}
<div <div
v-if="!showHelpState" v-if="!showHelpState"
......
...@@ -808,11 +808,7 @@ ...@@ -808,11 +808,7 @@
} }
} }
.time_tracker { .time-tracker {
padding-bottom: 0;
border-bottom: 0;
.sidebar-collapsed-icon { .sidebar-collapsed-icon {
> .stopwatch-svg { > .stopwatch-svg {
display: inline-block; display: inline-block;
......
...@@ -7,6 +7,7 @@ import IssuableAssignees from '~/sidebar/components/assignees/issuable_assignees ...@@ -7,6 +7,7 @@ import IssuableAssignees from '~/sidebar/components/assignees/issuable_assignees
import IssuableTitle from '~/boards/components/issuable_title.vue'; import IssuableTitle from '~/boards/components/issuable_title.vue';
import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin'; import glFeatureFlagsMixin from '~/vue_shared/mixins/gl_feature_flags_mixin';
import BoardSidebarEpicSelect from './sidebar/board_sidebar_epic_select.vue'; import BoardSidebarEpicSelect from './sidebar/board_sidebar_epic_select.vue';
import BoardSidebarTimeTracker from './sidebar/board_sidebar_time_tracker.vue';
import BoardSidebarWeightInput from './sidebar/board_sidebar_weight_input.vue'; import BoardSidebarWeightInput from './sidebar/board_sidebar_weight_input.vue';
import BoardSidebarLabelsSelect from '~/boards/components/sidebar/board_sidebar_labels_select.vue'; import BoardSidebarLabelsSelect from '~/boards/components/sidebar/board_sidebar_labels_select.vue';
...@@ -17,6 +18,7 @@ export default { ...@@ -17,6 +18,7 @@ export default {
GlDrawer, GlDrawer,
IssuableTitle, IssuableTitle,
BoardSidebarEpicSelect, BoardSidebarEpicSelect,
BoardSidebarTimeTracker,
BoardSidebarWeightInput, BoardSidebarWeightInput,
BoardSidebarLabelsSelect, BoardSidebarLabelsSelect,
}, },
...@@ -48,6 +50,7 @@ export default { ...@@ -48,6 +50,7 @@ export default {
<template> <template>
<issuable-assignees :users="getActiveIssue.assignees" /> <issuable-assignees :users="getActiveIssue.assignees" />
<board-sidebar-epic-select /> <board-sidebar-epic-select />
<board-sidebar-time-tracker class="swimlanes-sidebar-time-tracker" />
<board-sidebar-weight-input v-if="glFeatures.issueWeights" /> <board-sidebar-weight-input v-if="glFeatures.issueWeights" />
<board-sidebar-labels-select /> <board-sidebar-labels-select />
</template> </template>
......
<script>
import { mapGetters } from 'vuex';
import IssuableTimeTracker from '~/sidebar/components/time_tracking/time_tracker.vue';
export default {
inject: ['timeTrackingLimitToHours'],
components: {
IssuableTimeTracker,
},
computed: {
...mapGetters(['getActiveIssue']),
},
};
</script>
<template>
<issuable-time-tracker
:time-estimate="getActiveIssue.timeEstimate"
:time-spent="getActiveIssue.totalTimeSpent"
:human-time-estimate="getActiveIssue.humanTimeEstimate"
:human-time-spent="getActiveIssue.humanTotalTimeSpent"
:limit-to-hours="timeTrackingLimitToHours"
:show-collapsed="false"
/>
</template>
...@@ -7,6 +7,9 @@ fragment IssueNode on Issue { ...@@ -7,6 +7,9 @@ fragment IssueNode on Issue {
referencePath: reference(full: true) referencePath: reference(full: true)
dueDate dueDate
timeEstimate timeEstimate
totalTimeSpent
humanTimeEstimate
humanTotalTimeSpent
weight weight
confidential confidential
webUrl webUrl
......
...@@ -119,3 +119,15 @@ $epic-icons-spacing: 40px; ...@@ -119,3 +119,15 @@ $epic-icons-spacing: 40px;
max-width: calc(100vw - #{$contextual-sidebar-width} - #{$gutter-width} - #{$epic-icons-spacing}); max-width: calc(100vw - #{$contextual-sidebar-width} - #{$gutter-width} - #{$epic-icons-spacing});
} }
} }
.swimlanes-sidebar-time-tracker {
.time-tracking-help-state {
margin: 16px -24px 0;
@include gl-px-6;
@include gl-py-2;
@include gl-border-gray-100;
@include gl-border-t-solid;
@include gl-border-t-1;
@include gl-border-b-0;
}
}
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe 'epics swimlanes sidebar', :js do
let_it_be(:user) { create(:user) }
let_it_be(:group) { create(:group, :public) }
let_it_be(:project) { create(:project, :public, group: group) }
let_it_be(:board) { create(:board, project: project) }
let_it_be(:list) { create(:list, board: board, position: 0) }
let_it_be(:epic1) { create(:epic, group: group) }
let_it_be(:issue1, reload: true) { create(:issue, project: project) }
let_it_be(:epic_issue1, reload: true) { create(:epic_issue, epic: epic1, issue: issue1) }
before do
project.add_maintainer(user)
stub_licensed_features(epics: true, swimlanes: true)
sign_in(user)
visit project_boards_path(project)
wait_for_requests
page.within('.board-swimlanes-toggle-wrapper') do
page.find('.dropdown-toggle').click
page.find('.dropdown-item', text: 'Epic').click
end
wait_for_all_requests
end
context 'time tracking' do
it 'displays time tracking feature with default message' do
click_first_issue_card
page.within('[data-testid="time-tracker"]') do
expect(page).to have_content('Time tracking')
expect(page).to have_content('No estimate or time spent')
end
end
context 'when only spent time is recorded' do
before do
issue1.timelogs.create!(time_spent: 3600, user: user)
click_first_issue_card
end
it 'shows the total time spent only' do
page.within('[data-testid="time-tracker"]') do
expect(page).to have_content('Spent: 1h')
expect(page).not_to have_content('Estimated')
end
end
end
context 'when only estimated time is recorded' do
before do
issue1.update!(time_estimate: 3600)
click_first_issue_card
end
it 'shows the estimated time only' do
page.within('[data-testid="time-tracker"]') do
expect(page).to have_content('Estimated: 1h')
expect(page).not_to have_content('Spent')
end
end
end
context 'when estimated and spent times are available' do
before do
issue1.update!(time_estimate: 3600)
issue1.timelogs.create!(time_spent: 1800, user: user)
click_first_issue_card
end
it 'shows time tracking progress bar' do
page.within('[data-testid="time-tracker"]') do
expect(page).to have_selector('[data-testid="timeTrackingComparisonPane"]')
end
end
it 'shows both estimated and spent time text' do
page.within('[data-testid="time-tracker"]') do
expect(page).to have_content('Spent 30m')
expect(page).to have_content('Est 1h')
end
end
end
context 'when limitedToHours instance option is turned on' do
before do
stub_application_setting(time_tracking_limit_to_hours: true)
# 3600+3600*24 = 1d 1h or 25h
issue1.timelogs.create!(time_spent: 3600 + 3600 * 24, user: user)
click_first_issue_card
end
it 'shows the total time spent only' do
page.within('[data-testid="time-tracker"]') do
expect(page).to have_content('Spent: 25h')
end
end
end
end
def click_first_issue_card
page.within("[data-testid='board-epic-lane-issues']") do
first("[data-testid='board_card']").click
end
end
end
...@@ -19,6 +19,7 @@ describe('ee/BoardContentSidebar', () => { ...@@ -19,6 +19,7 @@ describe('ee/BoardContentSidebar', () => {
store, store,
stubs: { stubs: {
'board-sidebar-epic-select': '<div></div>', 'board-sidebar-epic-select': '<div></div>',
'board-sidebar-time-tracker': '<div></div>',
'board-sidebar-weight-input': '<div></div>', 'board-sidebar-weight-input': '<div></div>',
'board-sidebar-labels-select': '<div></div>', 'board-sidebar-labels-select': '<div></div>',
}, },
......
...@@ -11,6 +11,9 @@ describe('ee/BoardContent', () => { ...@@ -11,6 +11,9 @@ describe('ee/BoardContent', () => {
const createComponent = () => { const createComponent = () => {
wrapper = shallowMount(BoardContent, { wrapper = shallowMount(BoardContent, {
store, store,
provide: {
timeTrackingLimitToHours: false,
},
propsData: { propsData: {
lists: [], lists: [],
canAdminList: false, canAdminList: false,
......
/*
To avoid duplicating tests in time_tracker.spec,
this spec only contains a simple test to check rendering.
A detailed feature spec is used to test time tracking feature
in swimlanes sidebar.
*/
import { shallowMount } from '@vue/test-utils';
import BoardSidebarTimeTracker from 'ee/boards/components/sidebar/board_sidebar_time_tracker.vue';
import IssuableTimeTracker from '~/sidebar/components/time_tracking/time_tracker.vue';
import { createStore } from '~/boards/stores';
describe('BoardSidebarTimeTracker', () => {
let wrapper;
let store;
const createComponent = options => {
wrapper = shallowMount(BoardSidebarTimeTracker, {
store,
...options,
});
};
beforeEach(() => {
store = createStore();
store.state.issues = {
'1': {
timeEstimate: 3600,
totalTimeSpent: 1800,
humanTimeEstimate: '1h',
humanTotalTimeSpent: '30min',
},
};
store.state.activeId = '1';
});
afterEach(() => {
wrapper.destroy();
wrapper = null;
});
it.each([[true], [false]])(
'renders IssuableTimeTracker with correct spent and estimated time (timeTrackingLimitToHours=%s)',
timeTrackingLimitToHours => {
createComponent({ provide: { timeTrackingLimitToHours } });
expect(wrapper.find(IssuableTimeTracker).props()).toEqual({
timeEstimate: 3600,
timeSpent: 1800,
humanTimeEstimate: '1h',
humanTimeSpent: '30min',
limitToHours: timeTrackingLimitToHours,
showCollapsed: false,
});
},
);
});
...@@ -16,7 +16,6 @@ describe('Issuable Time Tracker', () => { ...@@ -16,7 +16,6 @@ describe('Issuable Time Tracker', () => {
humanTimeEstimate: '2h 46m', humanTimeEstimate: '2h 46m',
humanTimeSpent: '1h 23m', humanTimeSpent: '1h 23m',
limitToHours: false, limitToHours: false,
rootPath: '/',
}; };
const mountComponent = ({ props = {} } = {}) => const mountComponent = ({ props = {} } = {}) =>
...@@ -52,6 +51,24 @@ describe('Issuable Time Tracker', () => { ...@@ -52,6 +51,24 @@ describe('Issuable Time Tracker', () => {
}); });
describe('Content panes', () => { describe('Content panes', () => {
describe('Collapsed state', () => {
it('should render "time-tracking-collapsed-state" by default when "showCollapsed" prop is not specified', () => {
wrapper = mountComponent();
expect(findCollapsedState().exists()).toBe(true);
});
it('should not render "time-tracking-collapsed-state" when "showCollapsed" is false', () => {
wrapper = mountComponent({
props: {
showCollapsed: false,
},
});
expect(findCollapsedState().exists()).toBe(false);
});
});
describe('Comparison pane', () => { describe('Comparison pane', () => {
beforeEach(() => { beforeEach(() => {
wrapper = mountComponent({ wrapper = mountComponent({
......
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