Commit d844bca2 authored by Michael Lunøe's avatar Michael Lunøe Committed by Miguel Rincon

Fix(chart skeleton loader): adjust size

This adjusts the size of the chart skeleton loader
so it fits the size of the charts a bit better
making for a better user experience
parent cc2070a9
export const DEFAULT_RX = 0.4; export const DEFAULT_RX = 0.4;
export const DEFAULT_BAR_WIDTH = 6; export const DEFAULT_BAR_WIDTH = 4;
export const DEFAULT_LABEL_WIDTH = 4; export const DEFAULT_LABEL_WIDTH = 3;
export const DEFAULT_LABEL_HEIGHT = 5; export const DEFAULT_LABEL_HEIGHT = 3;
export const BAR_HEIGHTS = [5, 7, 9, 14, 21, 35, 50, 80]; export const BAR_HEIGHTS = [5, 7, 9, 14, 21, 35, 50, 80];
export const GRID_YS = [30, 60, 90]; export const GRID_YS = [30, 60, 90];
...@@ -61,35 +61,37 @@ export default { ...@@ -61,35 +61,37 @@ export default {
}; };
</script> </script>
<template> <template>
<gl-skeleton-loader :unique-key="uniqueKey"> <div class="gl-px-8">
<rect <gl-skeleton-loader :unique-key="uniqueKey" class="gl-p-8">
v-for="(y, index) in $options.GRID_YS" <rect
:key="`grid-${index}`" v-for="(y, index) in $options.GRID_YS"
data-testid="skeleton-chart-grid" :key="`grid-${index}`"
x="0" data-testid="skeleton-chart-grid"
:y="`${y}%`" x="0"
width="100%" :y="`${y}%`"
height="1px" width="100%"
/> height="1px"
<rect />
v-for="(height, index) in $options.BAR_HEIGHTS" <rect
:key="`bar-${index}`" v-for="(height, index) in $options.BAR_HEIGHTS"
data-testid="skeleton-chart-bar" :key="`bar-${index}`"
:x="`${getBarXPosition(index)}%`" data-testid="skeleton-chart-bar"
:y="`${90 - height}%`" :x="`${getBarXPosition(index)}%`"
:width="`${barWidth}%`" :y="`${90 - height}%`"
:height="`${height}%`" :width="`${barWidth}%`"
:rx="`${rx}%`" :height="`${height}%`"
/> :rx="`${rx}%`"
<rect />
v-for="(height, index) in $options.BAR_HEIGHTS" <rect
:key="`label-${index}`" v-for="(height, index) in $options.BAR_HEIGHTS"
data-testid="skeleton-chart-label" :key="`label-${index}`"
:x="`${labelCentering + getBarXPosition(index)}%`" data-testid="skeleton-chart-label"
:y="`${100 - labelHeight}%`" :x="`${labelCentering + getBarXPosition(index)}%`"
:width="`${labelWidth}%`" :y="`${100 - labelHeight}%`"
:height="`${labelHeight}%`" :width="`${labelWidth}%`"
:rx="`${rx}%`" :height="`${labelHeight}%`"
/> :rx="`${rx}%`"
</gl-skeleton-loader> />
</gl-skeleton-loader>
</div>
</template> </template>
...@@ -127,11 +127,11 @@ export default { ...@@ -127,11 +127,11 @@ export default {
:error="error" :error="error"
/> />
</div> </div>
<resizable-chart-container v-else-if="loaded" class="insights-chart"> <resizable-chart-container v-else class="insights-chart">
<h5 class="text-center">{{ title }}</h5> <h5 class="text-center">{{ title }}</h5>
<p v-if="description" class="text-center">{{ description }}</p> <p v-if="description" class="text-center">{{ description }}</p>
<gl-column-chart <gl-column-chart
v-if="isColumnChart" v-if="loaded && isColumnChart"
v-bind="$attrs" v-bind="$attrs"
:height="$options.height" :height="$options.height"
:data="data.datasets" :data="data.datasets"
...@@ -142,7 +142,7 @@ export default { ...@@ -142,7 +142,7 @@ export default {
@created="onChartCreated" @created="onChartCreated"
/> />
<gl-stacked-column-chart <gl-stacked-column-chart
v-else-if="isStackedColumnChart" v-else-if="loaded && isStackedColumnChart"
v-bind="$attrs" v-bind="$attrs"
:height="$options.height" :height="$options.height"
:data="data.datasets" :data="data.datasets"
...@@ -155,13 +155,13 @@ export default { ...@@ -155,13 +155,13 @@ export default {
@created="onChartCreated" @created="onChartCreated"
/> />
<gl-line-chart <gl-line-chart
v-else-if="isLineChart" v-else-if="loaded && isLineChart"
v-bind="$attrs" v-bind="$attrs"
:height="$options.height" :height="$options.height"
:data="data.datasets" :data="data.datasets"
:option="chartOptions" :option="chartOptions"
@created="onChartCreated" @created="onChartCreated"
/> />
<chart-skeleton-loader v-else />
</resizable-chart-container> </resizable-chart-container>
<chart-skeleton-loader v-else />
</template> </template>
---
title: Fix the size of chart placeholder on Analytics pages for Merge Requests, Insights and Value stream, so it matches the actual charts.
the size of the real chart shown afterwards
merge_request: 41904
author:
type: fixed
...@@ -10,8 +10,17 @@ import { ...@@ -10,8 +10,17 @@ import {
import InsightsChart from 'ee/insights/components/insights_chart.vue'; import InsightsChart from 'ee/insights/components/insights_chart.vue';
import InsightsChartError from 'ee/insights/components/insights_chart_error.vue'; import InsightsChartError from 'ee/insights/components/insights_chart_error.vue';
import { CHART_TYPES } from 'ee/insights/constants'; import { CHART_TYPES } from 'ee/insights/constants';
import ResizableChartContainer from '~/vue_shared/components/resizable_chart/resizable_chart_container.vue';
import ChartSkeletonLoader from '~/vue_shared/components/resizable_chart/skeleton_loader.vue'; import ChartSkeletonLoader from '~/vue_shared/components/resizable_chart/skeleton_loader.vue';
const DEFAULT_PROPS = {
loaded: false,
type: chartInfo.type,
title: chartInfo.title,
data: null,
error: '',
};
describe('Insights chart component', () => { describe('Insights chart component', () => {
let wrapper; let wrapper;
...@@ -26,66 +35,32 @@ describe('Insights chart component', () => { ...@@ -26,66 +35,32 @@ describe('Insights chart component', () => {
}); });
describe('when chart is loading', () => { describe('when chart is loading', () => {
it('displays the chart loader', () => { it('displays the chart loader in the container', () => {
wrapper = factory({ wrapper = factory(DEFAULT_PROPS);
loaded: false,
type: CHART_TYPES.BAR,
title: chartInfo.title,
data: null,
error: '',
});
expect(wrapper.find(ChartSkeletonLoader).exists()).toBe(true); expect(wrapper.find(ChartSkeletonLoader).exists()).toBe(true);
expect(wrapper.find(ResizableChartContainer).exists()).toBe(true);
}); });
}); });
describe('when chart is loaded', () => { describe.each`
it('displays a bar chart', () => { type | component | name | data
wrapper = factory({ ${CHART_TYPES.BAR} | ${GlColumnChart} | ${'GlColumnChart'} | ${barChartData}
loaded: true, ${CHART_TYPES.LINE} | ${GlLineChart} | ${'GlLineChart'} | ${lineChartData}
type: CHART_TYPES.BAR, ${CHART_TYPES.STACKED_BAR} | ${GlStackedColumnChart} | ${'GlStackedColumnChart'} | ${stackedBarChartData}
title: chartInfo.title, ${CHART_TYPES.PIE} | ${GlColumnChart} | ${'GlColumnChart'} | ${barChartData}
data: barChartData, `('when chart is loaded', ({ type, component, name, data }) => {
error: '', it(`when ${type} is passed: displays the a ${name}-chart in container and not the loader`, () => {
});
expect(wrapper.find(GlColumnChart).exists()).toBe(true);
});
it('displays a line chart', () => {
wrapper = factory({
loaded: true,
type: CHART_TYPES.LINE,
title: chartInfo.title,
data: lineChartData,
error: '',
});
expect(wrapper.find(GlLineChart).exists()).toBe(true);
});
it('displays a stacked bar chart', () => {
wrapper = factory({
loaded: true,
type: CHART_TYPES.STACKED_BAR,
title: chartInfo.title,
data: stackedBarChartData,
error: '',
});
expect(wrapper.find(GlStackedColumnChart).exists()).toBe(true);
});
it('displays a bar chart when a pie chart is requested', () => {
wrapper = factory({ wrapper = factory({
...DEFAULT_PROPS,
loaded: true, loaded: true,
type: CHART_TYPES.PIE, type,
title: chartInfo.title, data,
data: barChartData,
error: '',
}); });
expect(wrapper.find(GlColumnChart).exists()).toBe(true); expect(wrapper.find(ChartSkeletonLoader).exists()).toBe(false);
expect(wrapper.find(ResizableChartContainer).exists()).toBe(true);
expect(wrapper.find(component).exists()).toBe(true);
}); });
}); });
...@@ -94,15 +69,15 @@ describe('Insights chart component', () => { ...@@ -94,15 +69,15 @@ describe('Insights chart component', () => {
beforeEach(() => { beforeEach(() => {
wrapper = factory({ wrapper = factory({
loaded: false, ...DEFAULT_PROPS,
type: chartInfo.type,
title: chartInfo.title,
data: {}, data: {},
error, error,
}); });
}); });
it('displays info about the error', () => { it('displays info about the error', () => {
expect(wrapper.find(ChartSkeletonLoader).exists()).toBe(false);
expect(wrapper.find(ResizableChartContainer).exists()).toBe(false);
expect(wrapper.find(InsightsChartError).exists()).toBe(true); expect(wrapper.find(InsightsChartError).exists()).toBe(true);
}); });
}); });
......
// Jest Snapshot v1, https://goo.gl/fbAQLP // Jest Snapshot v1, https://goo.gl/fbAQLP
exports[`Resizable Skeleton Loader default setup renders the bars, labels, and grid with correct position, size, and rx percentages 1`] = ` exports[`Resizable Skeleton Loader default setup renders the bars, labels, and grid with correct position, size, and rx percentages 1`] = `
<svg <div
class="gl-skeleton-loader" class="gl-px-8"
preserveAspectRatio="xMidYMid meet"
version="1.1"
viewBox="0 0 400 130"
> >
<rect <svg
clip-path="url(#null-idClip)" class="gl-skeleton-loader"
height="130" preserveAspectRatio="xMidYMid meet"
style="fill: url(#null-idGradient);" version="1.1"
width="400" viewBox="0 0 400 130"
x="0" >
y="0" <rect
/> clip-path="url(#null-idClip)"
<defs> height="130"
<clippath style="fill: url(#null-idGradient);"
id="null-idClip" width="400"
> x="0"
<rect y="0"
data-testid="skeleton-chart-grid" />
height="1px" <defs>
width="100%" <clippath
x="0" id="null-idClip"
y="30%"
/>
<rect
data-testid="skeleton-chart-grid"
height="1px"
width="100%"
x="0"
y="60%"
/>
<rect
data-testid="skeleton-chart-grid"
height="1px"
width="100%"
x="0"
y="90%"
/>
<rect
data-testid="skeleton-chart-bar"
height="5%"
rx="0.4%"
width="6%"
x="5.875%"
y="85%"
/>
<rect
data-testid="skeleton-chart-bar"
height="7%"
rx="0.4%"
width="6%"
x="17.625%"
y="83%"
/>
<rect
data-testid="skeleton-chart-bar"
height="9%"
rx="0.4%"
width="6%"
x="29.375%"
y="81%"
/>
<rect
data-testid="skeleton-chart-bar"
height="14%"
rx="0.4%"
width="6%"
x="41.125%"
y="76%"
/>
<rect
data-testid="skeleton-chart-bar"
height="21%"
rx="0.4%"
width="6%"
x="52.875%"
y="69%"
/>
<rect
data-testid="skeleton-chart-bar"
height="35%"
rx="0.4%"
width="6%"
x="64.625%"
y="55%"
/>
<rect
data-testid="skeleton-chart-bar"
height="50%"
rx="0.4%"
width="6%"
x="76.375%"
y="40%"
/>
<rect
data-testid="skeleton-chart-bar"
height="80%"
rx="0.4%"
width="6%"
x="88.125%"
y="10%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="6.875%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="18.625%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="30.375%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="42.125%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="53.875%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="65.625%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="77.375%"
y="95%"
/>
<rect
data-testid="skeleton-chart-label"
height="5%"
rx="0.4%"
width="4%"
x="89.125%"
y="95%"
/>
</clippath>
<lineargradient
id="null-idGradient"
>
<stop
class="primary-stop"
offset="0%"
> >
<animate <rect
attributeName="offset" data-testid="skeleton-chart-grid"
dur="1s" height="1px"
repeatCount="indefinite" width="100%"
values="-2; 1" x="0"
/> y="30%"
</stop> />
<stop <rect
class="secondary-stop" data-testid="skeleton-chart-grid"
offset="50%" height="1px"
> width="100%"
<animate x="0"
attributeName="offset" y="60%"
dur="1s" />
repeatCount="indefinite" <rect
values="-1.5; 1.5" data-testid="skeleton-chart-grid"
/> height="1px"
</stop> width="100%"
<stop x="0"
class="primary-stop" y="90%"
offset="100%" />
<rect
data-testid="skeleton-chart-bar"
height="5%"
rx="0.4%"
width="4%"
x="6%"
y="85%"
/>
<rect
data-testid="skeleton-chart-bar"
height="7%"
rx="0.4%"
width="4%"
x="18%"
y="83%"
/>
<rect
data-testid="skeleton-chart-bar"
height="9%"
rx="0.4%"
width="4%"
x="30%"
y="81%"
/>
<rect
data-testid="skeleton-chart-bar"
height="14%"
rx="0.4%"
width="4%"
x="42%"
y="76%"
/>
<rect
data-testid="skeleton-chart-bar"
height="21%"
rx="0.4%"
width="4%"
x="54%"
y="69%"
/>
<rect
data-testid="skeleton-chart-bar"
height="35%"
rx="0.4%"
width="4%"
x="66%"
y="55%"
/>
<rect
data-testid="skeleton-chart-bar"
height="50%"
rx="0.4%"
width="4%"
x="78%"
y="40%"
/>
<rect
data-testid="skeleton-chart-bar"
height="80%"
rx="0.4%"
width="4%"
x="90%"
y="10%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="6.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="18.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="30.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="42.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="54.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="66.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="78.5%"
y="97%"
/>
<rect
data-testid="skeleton-chart-label"
height="3%"
rx="0.4%"
width="3%"
x="90.5%"
y="97%"
/>
</clippath>
<lineargradient
id="null-idGradient"
> >
<animate <stop
attributeName="offset" class="primary-stop"
dur="1s" offset="0%"
repeatCount="indefinite" >
values="-1; 2" <animate
/> attributeName="offset"
</stop> dur="1s"
</lineargradient> repeatCount="indefinite"
</defs> values="-2; 1"
</svg> />
</stop>
<stop
class="secondary-stop"
offset="50%"
>
<animate
attributeName="offset"
dur="1s"
repeatCount="indefinite"
values="-1.5; 1.5"
/>
</stop>
<stop
class="primary-stop"
offset="100%"
>
<animate
attributeName="offset"
dur="1s"
repeatCount="indefinite"
values="-1; 2"
/>
</stop>
</lineargradient>
</defs>
</svg>
</div>
`; `;
exports[`Resizable Skeleton Loader with custom settings renders the correct position, and size percentages for bars and labels with different settings 1`] = ` exports[`Resizable Skeleton Loader with custom settings renders the correct position, and size percentages for bars and labels with different settings 1`] = `
<svg <div
class="gl-skeleton-loader" class="gl-px-8"
preserveAspectRatio="xMidYMid meet"
version="1.1"
viewBox="0 0 400 130"
> >
<rect <svg
clip-path="url(#-idClip)" class="gl-skeleton-loader"
height="130" preserveAspectRatio="xMidYMid meet"
style="fill: url(#-idGradient);" version="1.1"
width="400" viewBox="0 0 400 130"
x="0" >
y="0" <rect
/> clip-path="url(#-idClip)"
<defs> height="130"
<clippath style="fill: url(#-idGradient);"
id="-idClip" width="400"
> x="0"
<rect y="0"
data-testid="skeleton-chart-grid" />
height="1px" <defs>
width="100%" <clippath
x="0" id="-idClip"
y="30%"
/>
<rect
data-testid="skeleton-chart-grid"
height="1px"
width="100%"
x="0"
y="60%"
/>
<rect
data-testid="skeleton-chart-grid"
height="1px"
width="100%"
x="0"
y="90%"
/>
<rect
data-testid="skeleton-chart-bar"
height="5%"
rx="0.6%"
width="3%"
x="6.0625%"
y="85%"
/>
<rect
data-testid="skeleton-chart-bar"
height="7%"
rx="0.6%"
width="3%"
x="18.1875%"
y="83%"
/>
<rect
data-testid="skeleton-chart-bar"
height="9%"
rx="0.6%"
width="3%"
x="30.3125%"
y="81%"
/>
<rect
data-testid="skeleton-chart-bar"
height="14%"
rx="0.6%"
width="3%"
x="42.4375%"
y="76%"
/>
<rect
data-testid="skeleton-chart-bar"
height="21%"
rx="0.6%"
width="3%"
x="54.5625%"
y="69%"
/>
<rect
data-testid="skeleton-chart-bar"
height="35%"
rx="0.6%"
width="3%"
x="66.6875%"
y="55%"
/>
<rect
data-testid="skeleton-chart-bar"
height="50%"
rx="0.6%"
width="3%"
x="78.8125%"
y="40%"
/>
<rect
data-testid="skeleton-chart-bar"
height="80%"
rx="0.6%"
width="3%"
x="90.9375%"
y="10%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="4.0625%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="16.1875%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="28.3125%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="40.4375%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="52.5625%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="64.6875%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="76.8125%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="88.9375%"
y="98%"
/>
</clippath>
<lineargradient
id="-idGradient"
>
<stop
class="primary-stop"
offset="0%"
> >
<animate <rect
attributeName="offset" data-testid="skeleton-chart-grid"
dur="1s" height="1px"
repeatCount="indefinite" width="100%"
values="-2; 1" x="0"
/> y="30%"
</stop> />
<stop <rect
class="secondary-stop" data-testid="skeleton-chart-grid"
offset="50%" height="1px"
> width="100%"
<animate x="0"
attributeName="offset" y="60%"
dur="1s" />
repeatCount="indefinite" <rect
values="-1.5; 1.5" data-testid="skeleton-chart-grid"
/> height="1px"
</stop> width="100%"
<stop x="0"
class="primary-stop" y="90%"
offset="100%" />
<rect
data-testid="skeleton-chart-bar"
height="5%"
rx="0.6%"
width="3%"
x="6.0625%"
y="85%"
/>
<rect
data-testid="skeleton-chart-bar"
height="7%"
rx="0.6%"
width="3%"
x="18.1875%"
y="83%"
/>
<rect
data-testid="skeleton-chart-bar"
height="9%"
rx="0.6%"
width="3%"
x="30.3125%"
y="81%"
/>
<rect
data-testid="skeleton-chart-bar"
height="14%"
rx="0.6%"
width="3%"
x="42.4375%"
y="76%"
/>
<rect
data-testid="skeleton-chart-bar"
height="21%"
rx="0.6%"
width="3%"
x="54.5625%"
y="69%"
/>
<rect
data-testid="skeleton-chart-bar"
height="35%"
rx="0.6%"
width="3%"
x="66.6875%"
y="55%"
/>
<rect
data-testid="skeleton-chart-bar"
height="50%"
rx="0.6%"
width="3%"
x="78.8125%"
y="40%"
/>
<rect
data-testid="skeleton-chart-bar"
height="80%"
rx="0.6%"
width="3%"
x="90.9375%"
y="10%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="4.0625%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="16.1875%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="28.3125%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="40.4375%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="52.5625%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="64.6875%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="76.8125%"
y="98%"
/>
<rect
data-testid="skeleton-chart-label"
height="2%"
rx="0.6%"
width="7%"
x="88.9375%"
y="98%"
/>
</clippath>
<lineargradient
id="-idGradient"
> >
<animate <stop
attributeName="offset" class="primary-stop"
dur="1s" offset="0%"
repeatCount="indefinite" >
values="-1; 2" <animate
/> attributeName="offset"
</stop> dur="1s"
</lineargradient> repeatCount="indefinite"
</defs> values="-2; 1"
</svg> />
</stop>
<stop
class="secondary-stop"
offset="50%"
>
<animate
attributeName="offset"
dur="1s"
repeatCount="indefinite"
values="-1.5; 1.5"
/>
</stop>
<stop
class="primary-stop"
offset="100%"
>
<animate
attributeName="offset"
dur="1s"
repeatCount="indefinite"
values="-1; 2"
/>
</stop>
</lineargradient>
</defs>
</svg>
</div>
`; `;
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