Commit a2955b97 authored by Filipa Lacerda's avatar Filipa Lacerda

Adds loading state to linked pipeline

Renders a loading spinner for the isLoading
state.
Updates tests
parent f1c10b7f
<script> <script>
import ciStatus from '~/vue_shared/components/ci_icon.vue'; import { GlLoadingIcon, GlTooltipDirective, GlLink } from '@gitlab/ui';
import tooltip from '~/vue_shared/directives/tooltip'; import CiStatus from '~/vue_shared/components/ci_icon.vue';
export default { export default {
directives: { directives: {
tooltip, GlTooltip: GlTooltipDirective,
}, },
components: { components: {
ciStatus, CiStatus,
GlLoadingIcon,
GlLink,
}, },
props: { props: {
pipelineId: { pipelineId: {
...@@ -26,6 +28,11 @@ export default { ...@@ -26,6 +28,11 @@ export default {
type: String, type: String,
required: true, required: true,
}, },
isLoading: {
type: Boolean,
required: false,
default: false,
},
}, },
computed: { computed: {
tooltipText() { tooltipText() {
...@@ -39,20 +46,20 @@ export default { ...@@ -39,20 +46,20 @@ export default {
<li class="linked-pipeline build"> <li class="linked-pipeline build">
<div class="curve"></div> <div class="curve"></div>
<div> <div>
<a <gl-link
v-tooltip v-gl-tooltip
:href="pipelinePath" :href="pipelinePath"
:title="tooltipText" :title="tooltipText"
class="linked-pipeline-content" class="js-linked-pipeline-content linked-pipeline-content"
data-container="body"
> >
<span class="linked-pipeline-status ci-status-text"> <span class="js-linked-pipeline-status ci-status-text">
<ci-status :status="pipelineStatus" /> <gl-loading-icon v-if="isLoading" class="js-linked-pipeline-loading" />
<ci-status v-else :status="pipelineStatus" class="js-linked-pipeline-status" />
</span> </span>
<span class="linked-pipeline-project-name">{{ projectName }}</span> <span class="linked-pipeline-project-name">{{ projectName }}</span>
<span class="project-name-pipeline-id-separator">&#8226;</span> <span class="project-name-pipeline-id-separator">&#8226;</span>
<span class="linked-pipeline-id">#{{ pipelineId }}</span> <span class="js-linked-pipeline-id">#{{ pipelineId }}</span>
</a> </gl-link>
</div> </div>
</li> </li>
</template> </template>
import Vue from 'vue'; import Vue from 'vue';
import LinkedPipelineComponent from 'ee/pipelines/components/graph/linked_pipeline.vue'; import LinkedPipelineComponent from 'ee/pipelines/components/graph/linked_pipeline.vue';
import mountComponent from 'spec/helpers/vue_mount_component_helper';
import mockData from './linked_pipelines_mock_data'; import mockData from './linked_pipelines_mock_data';
const LinkedPipeline = Vue.extend(LinkedPipelineComponent);
const mockPipeline = mockData.triggered[0]; const mockPipeline = mockData.triggered[0];
describe('Linked pipeline', function() { describe('Linked pipeline', () => {
beforeEach(() => { const Component = Vue.extend(LinkedPipelineComponent);
this.propsData = { let vm;
afterEach(() => {
vm.$destroy();
});
describe('while is loading', () => {
const props = {
pipelineId: mockPipeline.id, pipelineId: mockPipeline.id,
pipelinePath: mockPipeline.path, pipelinePath: mockPipeline.path,
pipelineStatus: mockPipeline.details.status, pipelineStatus: mockPipeline.details.status,
projectName: mockPipeline.project.name, projectName: mockPipeline.project.name,
isLoading: true,
}; };
this.linkedPipeline = new LinkedPipeline({ beforeEach(() => {
propsData: this.propsData, vm = mountComponent(Component, props);
}).$mount();
}); });
it('should return a defined Vue component', () => { it('renders loading icon', () => {
expect(this.linkedPipeline).toBeDefined(); expect(vm.$el.querySelector('.js-linked-pipeline-loading')).not.toBeNull();
});
});
describe('when it is not loading', () => {
const props = {
pipelineId: mockPipeline.id,
pipelinePath: mockPipeline.path,
pipelineStatus: mockPipeline.details.status,
projectName: mockPipeline.project.name,
isLoading: false,
};
beforeEach(() => {
vm = mountComponent(Component, props);
}); });
it('should render a list item as the containing element', () => { it('should render a list item as the containing element', () => {
expect(this.linkedPipeline.$el.tagName).toBe('LI'); expect(vm.$el.tagName).toBe('LI');
}); });
it('should render a link', () => { it('should render a link', () => {
const linkElement = this.linkedPipeline.$el.querySelector('.linked-pipeline-content'); const linkElement = vm.$el.querySelector('.js-linked-pipeline-content');
expect(linkElement).not.toBeNull(); expect(linkElement).not.toBeNull();
}); });
it('should link to the correct path', () => { it('should link to the correct path', () => {
const linkElement = this.linkedPipeline.$el.querySelector('.linked-pipeline-content'); const linkElement = vm.$el.querySelector('.js-linked-pipeline-content');
expect(linkElement.getAttribute('href')).toBe(this.propsData.pipelinePath); expect(linkElement.getAttribute('href')).toBe(props.pipelinePath);
}); });
it('should render the project name', () => { it('should render the project name', () => {
const projectNameElement = this.linkedPipeline.$el.querySelector( const projectNameElement = vm.$el.querySelector('.linked-pipeline-project-name');
'.linked-pipeline-project-name',
);
expect(projectNameElement.innerText).toContain(this.propsData.projectName); expect(projectNameElement.innerText).toContain(props.projectName);
}); });
it('should render an svg within the status container', () => { it('should render an svg within the status container', () => {
const pipelineStatusElement = this.linkedPipeline.$el.querySelector('.linked-pipeline-status'); const pipelineStatusElement = vm.$el.querySelector('.js-linked-pipeline-status');
expect(pipelineStatusElement.querySelector('svg')).not.toBeNull(); expect(pipelineStatusElement.querySelector('svg')).not.toBeNull();
}); });
it('should render the pipeline status icon svg', () => { it('should render the pipeline status icon svg', () => {
const pipelineStatusElement = this.linkedPipeline.$el.querySelector('.linked-pipeline-status'); const pipelineStatusElement = vm.$el.querySelector('.js-linked-pipeline-status');
expect(pipelineStatusElement.querySelector('.ci-status-icon-running')).not.toBeNull(); expect(pipelineStatusElement.querySelector('.ci-status-icon-running')).not.toBeNull();
expect(pipelineStatusElement.innerHTML).toContain('<svg'); expect(pipelineStatusElement.innerHTML).toContain('<svg');
}); });
it('should render the correct pipeline status icon style selector', () => { it('should render the correct pipeline status icon style selector', () => {
const pipelineStatusElement = this.linkedPipeline.$el.querySelector('.linked-pipeline-status'); const pipelineStatusElement = vm.$el.querySelector('.js-linked-pipeline-status');
expect(pipelineStatusElement.firstChild.classList.contains('ci-status-icon-running')).toBe( expect(pipelineStatusElement.firstChild.classList.contains('ci-status-icon-running')).toBe(
true, true,
...@@ -69,28 +88,26 @@ describe('Linked pipeline', function() { ...@@ -69,28 +88,26 @@ describe('Linked pipeline', function() {
}); });
it('should have a ci-status child component', () => { it('should have a ci-status child component', () => {
const ciStatusComponent = this.linkedPipeline.$children[0]; expect(vm.$el.querySelector('.js-linked-pipeline-status')).not.toBeNull();
expect(ciStatusComponent).toBeDefined();
expect(ciStatusComponent.$el.classList.contains('ci-status-icon')).toBe(true);
}); });
it('should render the pipeline id', () => { it('should render the pipeline id', () => {
const pipelineIdElement = this.linkedPipeline.$el.querySelector('.linked-pipeline-id'); const pipelineIdElement = vm.$el.querySelector('.js-linked-pipeline-id');
expect(pipelineIdElement.innerText).toContain(`#${this.propsData.pipelineId}`); expect(pipelineIdElement.innerText).toContain(`#${props.pipelineId}`);
}); });
it('should correctly compute the tooltip text', () => { it('should correctly compute the tooltip text', () => {
expect(this.linkedPipeline.tooltipText).toContain(mockPipeline.project.name); expect(vm.tooltipText).toContain(mockPipeline.project.name);
expect(this.linkedPipeline.tooltipText).toContain(mockPipeline.details.status.label); expect(vm.tooltipText).toContain(mockPipeline.details.status.label);
}); });
it('should render the tooltip text as the title attribute', () => { it('should render the tooltip text as the title attribute', () => {
const tooltipRef = this.linkedPipeline.$el.querySelector('.linked-pipeline-content'); const tooltipRef = vm.$el.querySelector('.js-linked-pipeline-content');
const titleAttr = tooltipRef.getAttribute('data-original-title'); const titleAttr = tooltipRef.getAttribute('data-original-title');
expect(titleAttr).toContain(mockPipeline.project.name); expect(titleAttr).toContain(mockPipeline.project.name);
expect(titleAttr).toContain(mockPipeline.details.status.label); expect(titleAttr).toContain(mockPipeline.details.status.label);
}); });
});
}); });
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