Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: remove duplicate tick labels from axis #577

Merged
merged 32 commits into from
Mar 18, 2020
Merged
Show file tree
Hide file tree
Changes from 15 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
f784b2b
fix: add helper function to remove duplicate tickLabels from axis
rshen91 Mar 3, 2020
e71ac18
test: set utc time for test
rshen91 Mar 3, 2020
f4eafa5
refactor: clean up removeDupeTickLabels
rshen91 Mar 4, 2020
f7a1aa0
refactor: add filter after formatting
rshen91 Mar 4, 2020
a0c8839
feat: adding duplicate ticks story
rshen91 Mar 5, 2020
d6fb645
test: add missing snapshot
rshen91 Mar 6, 2020
f263fbf
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 9, 2020
d5abfd7
fix: add license heading to story file
rshen91 Mar 9, 2020
591c603
docs: correct x axis format for duplicates
rshen91 Mar 9, 2020
dc9a38f
test: update snapshot for x axis
rshen91 Mar 9, 2020
efce9bf
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 9, 2020
60664b9
feat: implement duplicate tick functionality
rshen91 Mar 10, 2020
af7c506
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 10, 2020
935e64a
style: clean up comments
rshen91 Mar 10, 2020
4976be7
test: add unit tests
rshen91 Mar 11, 2020
1b76c69
test: set timezone to clean up tests
rshen91 Mar 11, 2020
5037110
refactor: update naming and remove from scales spec
rshen91 Mar 11, 2020
1d888f7
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 11, 2020
ead2e9e
refactor: add timeZone prop to story
rshen91 Mar 11, 2020
c5ac664
refactor: improve filter() with a set and return allTicks earlier
rshen91 Mar 12, 2020
4e88447
refactor: refactor to reduce() implementation
rshen91 Mar 13, 2020
054c088
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 17, 2020
55bf6ee
refactor: use moment library for time calculations
rshen91 Mar 17, 2020
6ba6751
refactor: change config prop name to show vs enable
rshen91 Mar 17, 2020
9aa1329
fix: fix moment in duplicate ticks
rshen91 Mar 17, 2020
4952e6b
refactor: revert moment changes in brush stories
rshen91 Mar 17, 2020
8a9739c
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 17, 2020
594c6f6
feat: refactor getUniqueValues to be reusable
rshen91 Mar 17, 2020
a63b9eb
Merge remote-tracking branch 'upstream/master' into duplicate-ticks
rshen91 Mar 18, 2020
9ed79f0
refactor: refactored generics in getUniqueValues
rshen91 Mar 18, 2020
506bf0b
style: remove export from computeTickDimensions
rshen91 Mar 18, 2020
d0ff3c8
test: move story to axes stories and update vrt
rshen91 Mar 18, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
9 changes: 7 additions & 2 deletions src/chart_types/xy_chart/state/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -295,9 +295,14 @@ export function computeSeriesGeometries(

// compute how many series are clustered
const { stackedBarsInCluster, totalBarsInCluster } = countBarsInCluster(stacked, nonStacked);

// compute scales
const xScale = computeXScale({ xDomain, totalBarsInCluster, range: [0, width], barsPadding, enableHistogramMode });
const xScale = computeXScale({
xDomain,
totalBarsInCluster,
range: [0, width],
barsPadding,
enableHistogramMode,
});
const yScales = computeYScales({ yDomains: yDomain, range: [height, 0] });

// compute colors
Expand Down
128 changes: 126 additions & 2 deletions src/chart_types/xy_chart/utils/axis_utils.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ import { AxisSpec, DomainRange, AxisStyle } from './specs';
import { Position } from '../../../utils/commons';
import { LIGHT_THEME } from '../../../utils/themes/light_theme';
import { AxisId, GroupId } from '../../../utils/ids';
import { ScaleType } from '../../../scales';
import { ScaleType, Scale } from '../../../scales';
import {
AxisTick,
AxisTicksDimensions,
Expand All @@ -48,13 +48,16 @@ import {
getAxisTickLabelPadding,
isVerticalGrid,
isHorizontalGrid,
getDuplicateTicks,
} from './axis_utils';
import { CanvasTextBBoxCalculator } from '../../../utils/bbox/canvas_text_bbox_calculator';
import { SvgTextBBoxCalculator } from '../../../utils/bbox/svg_text_bbox_calculator';
import { niceTimeFormatter } from '../../../utils/data/formatters';
import { mergeYCustomDomainsByGroupId } from '../state/selectors/merge_y_custom_domains';
import { ChartTypes } from '../..';
import { SpecTypes } from '../../../specs/settings';
import { DateTime } from 'luxon';
import { computeXScale } from './scales';

describe('Axis computational utils', () => {
const mockedRect = {
Expand Down Expand Up @@ -202,7 +205,7 @@ describe('Axis computational utils', () => {
expect(axisDimensions).toEqual(axis1Dims);

const computeScalelessSpec = () => {
computeAxisTicksDimensions(ungroupedAxisSpec, xDomain, [yDomain], 1, bboxCalculator, 0, axes);
computeAxisTicksDimensions(ungroupedAxisSpec, xDomain, [yDomain], 1, bboxCalculator, 0, axes, undefined, false);
};

const ungroupedAxisSpec = { ...verticalAxisSpec, groupId: 'foo' };
Expand Down Expand Up @@ -1436,4 +1439,125 @@ describe('Axis computational utils', () => {

expect(getAxisTickLabelPadding(axisConfigTickLabelPadding, axisSpecStyle)).toEqual(2);
});
test('should show unique tick labels if duplicateTicks is set to false', () => {
const now = DateTime.fromISO('2019-01-11T00:00:00.000')
.setZone('utc+1')
.toMillis();
const oneDay = 1000 * 60 * 60 * 24;
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you should leverage packages to your advantage. Like using duration from moment.

Suggested change
const oneDay = 1000 * 60 * 60 * 24;
const oneDay = moment.duration(1, 'day');

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cool! Pls see commit 55bf6ee for changes 💪

const formatter = niceTimeFormatter([now, now + oneDay * 31]);
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Better yet you could use moment's super easy and declarative API for this using .add and .subtract depending on what type of date niceTimeFormatter is expecting. If you need the value in mills you could use the duration.add.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Changes in commit 55bf6ee thanks!

const axisSpec: AxisSpec = {
id: 'bottom',
position: 'bottom',
duplicateTicks: false,
chartType: 'xy_axis',
specType: 'axis',
groupId: '__global__',
hide: false,
showOverlappingLabels: false,
showOverlappingTicks: false,
tickSize: 10,
tickPadding: 10,
tickLabelRotation: 0,
tickFormat: formatter,
};
const xDomainTime: XDomain = {
type: 'xDomain',
isBandScale: false,
domain: [1547190000000, 1547622000000],
minInterval: 86400000,
scaleType: ScaleType.Time,
};
const scale: Scale = computeXScale({ xDomain: xDomainTime, totalBarsInCluster: 0, range: [0, 603.5] });
// xDomain: { type: 'xDomain', scaleType: 'time', isBandScale: false },
//
// minInterval:
// timeZone: undefined,
// totalBarsInCluster: 0,
// range: [0, 603.5],
// barsPadding: 0.25,
// enableHistogramMode: false,
// ticks: undefined,
// integersOnly: undefined,
// duplicateTicks: false,
// = {
// domain: [],
// barsPadding: 0.25,
// bandwidth: 0,
// type: 'time',
// range: [0, 603.5],
// minInterval: 86400000,
// isInverted: false,
// tickValues: [
// 1547208000000,
// 1547251200000,
// 1547294400000,
// 1547337600000,
// 1547380800000,
// 1547424000000,
// 1547467200000,
// 1547510400000,
// 1547553600000,
// 1547596800000,
// ],
// isSingleValue: () => false,
// totalBarsInCluster: 0,
// bandwidthPadding: 0,
// step: 0,
// timeZone: 'utc',
// isSingleValueHistogram: false,
// };
const offset = 0;
const tickFormatOption = { timeZone: undefined };
expect(getDuplicateTicks(axisSpec, scale, offset, tickFormatOption)).toEqual([
{ value: 1547208000000, label: '2019-01-11', position: 25.145833333333332 },
{ value: 1547294400000, label: '2019-01-12', position: 145.84583333333333 },
{ value: 1547380800000, label: '2019-01-13', position: 266.54583333333335 },
{ value: 1547467200000, label: '2019-01-14', position: 387.24583333333334 },
{ value: 1547553600000, label: '2019-01-15', position: 507.9458333333333 },
]);
});
test('should show duplicate tick labels if duplicateTicks is set to true', () => {
const now = DateTime.fromISO('2019-01-11T00:00:00.000')
.setZone('utc+1')
.toMillis();
const oneDay = 1000 * 60 * 60 * 24;
const formatter = niceTimeFormatter([now, now + oneDay * 31]);
const axisSpec: AxisSpec = {
id: 'bottom',
position: 'bottom',
duplicateTicks: true,
chartType: 'xy_axis',
specType: 'axis',
groupId: '__global__',
hide: false,
showOverlappingLabels: false,
showOverlappingTicks: false,
tickSize: 10,
tickPadding: 10,
tickLabelRotation: 0,
tickFormat: formatter,
};
const xDomainTime: XDomain = {
type: 'xDomain',
isBandScale: false,
domain: [1547190000000, 1547622000000],
minInterval: 86400000,
scaleType: ScaleType.Time,
};
const scale: Scale = computeXScale({ xDomain: xDomainTime, totalBarsInCluster: 0, range: [0, 603.5] });
const offset = 0;
const tickFormatOption = { timeZone: undefined };
expect(getDuplicateTicks(axisSpec, scale, offset, tickFormatOption)).toEqual([
{ value: 1547208000000, label: '2019-01-11', position: 25.145833333333332 },
{ value: 1547251200000, label: '2019-01-11', position: 85.49583333333334 },
{ value: 1547294400000, label: '2019-01-12', position: 145.84583333333333 },
{ value: 1547337600000, label: '2019-01-12', position: 206.19583333333333 },
{ value: 1547380800000, label: '2019-01-13', position: 266.54583333333335 },
{ value: 1547424000000, label: '2019-01-13', position: 326.8958333333333 },
{ value: 1547467200000, label: '2019-01-14', position: 387.24583333333334 },
{ value: 1547510400000, label: '2019-01-14', position: 447.59583333333336 },
{ value: 1547553600000, label: '2019-01-15', position: 507.9458333333333 },
{ value: 1547596800000, label: '2019-01-15', position: 568.2958333333333 },
]);
});
});
39 changes: 33 additions & 6 deletions src/chart_types/xy_chart/utils/axis_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -87,7 +87,6 @@ export function computeAxisTicksDimensions(
if (axisSpec.hide) {
return null;
}

const scale = getScaleForAxisSpec(
axisSpec,
xDomain,
Expand Down Expand Up @@ -157,6 +156,7 @@ export function getScaleForAxisSpec(
range,
ticks: axisSpec.ticks,
integersOnly: axisSpec.integersOnly,
duplicateTicks: axisSpec.duplicateTicks,
});
if (yScales.has(axisSpec.groupId)) {
return yScales.get(axisSpec.groupId)!;
Expand All @@ -171,6 +171,7 @@ export function getScaleForAxisSpec(
enableHistogramMode,
ticks: axisSpec.ticks,
integersOnly: axisSpec.integersOnly,
duplicateTicks: axisSpec.duplicateTicks,
});
}
}
Expand Down Expand Up @@ -225,24 +226,23 @@ export const getMaxBboxDimensions = (
};
};

function computeTickDimensions(
export function computeTickDimensions(
scale: Scale,
tickFormat: TickFormatter,
bboxCalculator: BBoxCalculator,
axisConfig: AxisConfig,
tickLabelPadding: number,
tickLabelRotation = 0,
// duplicateTicks: boolean = false,
tickFormatOptions?: TickFormatterOptions,
) {
const tickValues = scale.ticks();
const tickLabels = tickValues.map((d) => {
return tickFormat(d, tickFormatOptions);
});

const {
tickLabelStyle: { fontFamily, fontSize },
} = axisConfig;

const {
maxLabelBboxWidth,
maxLabelBboxHeight,
Expand All @@ -252,7 +252,6 @@ function computeTickDimensions(
getMaxBboxDimensions(bboxCalculator, fontSize, fontFamily, tickLabelRotation, tickLabelPadding),
{ maxLabelBboxWidth: 0, maxLabelBboxHeight: 0, maxLabelTextWidth: 0, maxLabelTextHeight: 0 },
);

return {
tickValues,
tickLabels,
Expand Down Expand Up @@ -444,14 +443,42 @@ export function getAvailableTicks(

return [firstTick, lastTick];
}
return ticks.map((tick) => {
return getDuplicateTicks(axisSpec, scale, offset, tickFormatOptions);
}

export function getDuplicateTicks(
axisSpec: AxisSpec,
scale: Scale,
offset: number,
tickFormatOptions?: TickFormatterOptions,
) {
const ticks = scale.ticks();
const labels = [...new Set(ticks.map((tick) => axisSpec.tickFormat(tick, tickFormatOptions)))];
const allTicks = ticks.map((tick) => {
return {
value: tick,
label: axisSpec.tickFormat(tick, tickFormatOptions),
position: scale.scale(tick) + offset,
};
});

if (axisSpec.duplicateTicks === false) {
const uniqueTickLabels: { value: any; label: string; position: number }[] = [];
allTicks.filter((value, index) => {
for (let i = 0; i < labels.length; i++) {
if (labels[i] === allTicks[index].label) {
uniqueTickLabels.push(allTicks[index]);
// once uniqueTickLabels has the unique label, remove the label from the labels array so it doesnt create a duplicate
labels.splice(i, 1);
}
}
});
return uniqueTickLabels;
} else {
return allTicks;
}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this code can be simplified a bit, let's chat on that directly

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks a ton for your help with me on this. I added commit 594c6f6 which I think gets done what we discussed, but lmk what you think!

}

export function getVisibleTicks(allTicks: AxisTick[], axisSpec: AxisSpec, axisDim: AxisTicksDimensions): AxisTick[] {
// We sort the ticks by position so that we can incrementally compute previousOccupiedSpace
allTicks.sort((a: AxisTick, b: AxisTick) => a.position - b.position);
Expand Down
18 changes: 15 additions & 3 deletions src/chart_types/xy_chart/utils/scales.ts
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,7 @@ interface XScaleOptions {
enableHistogramMode?: boolean;
ticks?: number;
integersOnly?: boolean;
duplicateTicks?: boolean;
}

/**
Expand All @@ -87,7 +88,16 @@ interface XScaleOptions {
* @param axisLength the length of the x axis
*/
export function computeXScale(options: XScaleOptions): Scale {
const { xDomain, totalBarsInCluster, range, barsPadding, enableHistogramMode, ticks, integersOnly } = options;
const {
xDomain,
totalBarsInCluster,
range,
barsPadding,
enableHistogramMode,
ticks,
integersOnly,
duplicateTicks,
} = options;
const { scaleType, minInterval, domain, isBandScale, timeZone } = xDomain;
const rangeDiff = Math.abs(range[1] - range[0]);
const isInverse = range[1] < range[0];
Expand Down Expand Up @@ -129,7 +139,7 @@ export function computeXScale(options: XScaleOptions): Scale {
} else {
return new ScaleContinuous(
{ type: scaleType, domain, range },
{ bandwidth: 0, minInterval, timeZone, totalBarsInCluster, barsPadding, ticks, integersOnly },
{ bandwidth: 0, minInterval, timeZone, totalBarsInCluster, barsPadding, ticks, integersOnly, duplicateTicks },
);
}
}
Expand All @@ -140,6 +150,7 @@ interface YScaleOptions {
range: [number, number];
ticks?: number;
integersOnly?: boolean;
duplicateTicks?: boolean;
}
/**
* Compute the y scales, one per groupId for the y axis.
Expand All @@ -148,7 +159,7 @@ interface YScaleOptions {
*/
export function computeYScales(options: YScaleOptions): Map<GroupId, Scale> {
const yScales: Map<GroupId, Scale> = new Map();
const { yDomains, range, ticks, integersOnly } = options;
const { yDomains, range, ticks, integersOnly, duplicateTicks } = options;
yDomains.forEach(({ scaleType: type, domain, groupId }) => {
const yScale = new ScaleContinuous(
{
Expand All @@ -159,6 +170,7 @@ export function computeYScales(options: YScaleOptions): Map<GroupId, Scale> {
{
ticks,
integersOnly,
duplicateTicks,
},
);
yScales.set(groupId, yScale);
Expand Down
3 changes: 3 additions & 0 deletions src/chart_types/xy_chart/utils/specs.ts
Original file line number Diff line number Diff line change
Expand Up @@ -525,11 +525,14 @@ export interface AxisSpec extends Spec {
style?: AxisStyle;
/** Show only integar values **/
integersOnly?: boolean;
/** Remove duplicate ticks, default is false*/
duplicateTicks?: boolean;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@rshen91 Maybe we should introduce a breaking change and enabling this by default changing the name to enableDuplicatedTicks or something similar. What do you think?
@nickofthyme ideas?

Copy link
Contributor Author

@rshen91 rshen91 Mar 11, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like the naming a lot better - please see commit 5037110 for naming changes

}

export type TickFormatterOptions = {
timeZone?: string;
};

export type TickFormatter = (value: any, options?: TickFormatterOptions) => string;

export interface AxisStyle {
Expand Down
3 changes: 3 additions & 0 deletions src/scales/scale_continuous.ts
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,8 @@ interface ScaleOptions {
isSingleValueHistogram: boolean;
/** Show only integer values **/
integersOnly?: boolean;
/** Show duplicate tick values default to false */
duplicateTicks?: boolean;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why we need this in the scale option? the getDuplicateTicks doesn't use that but use directly the AxisSpec props

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah thanks for spotting this, I didn't remove this properly after enabling the feature - I'm removing this in commit 5037110

}
const defaultScaleOptions: ScaleOptions = {
bandwidth: 0,
Expand All @@ -149,6 +151,7 @@ const defaultScaleOptions: ScaleOptions = {
ticks: 10,
isSingleValueHistogram: false,
integersOnly: false,
duplicateTicks: false,
};
export class ScaleContinuous implements Scale {
readonly bandwidth: number;
Expand Down
Loading