mirror of
https://github.com/grafana/grafana.git
synced 2025-08-01 21:52:37 +08:00

* fix: use fieldConfig.defaults.noValue in barchart and statetimeline * change name of test * remove todo comment * lean on PanelDataErrorView in the case where no fields are present * update StateTimeline and StatusHistory to use PanelDataErrorView * fix test * self-review cleanup
727 lines
19 KiB
TypeScript
727 lines
19 KiB
TypeScript
import {
|
|
DataFrame,
|
|
FALLBACK_COLOR,
|
|
Field,
|
|
FieldColorModeId,
|
|
FieldConfig,
|
|
FieldType,
|
|
formattedValueToString,
|
|
getFieldDisplayName,
|
|
getValueFormat,
|
|
GrafanaTheme2,
|
|
getActiveThreshold,
|
|
Threshold,
|
|
getFieldConfigWithMinMax,
|
|
ThresholdsMode,
|
|
TimeRange,
|
|
cacheFieldDisplayNames,
|
|
outerJoinDataFrames,
|
|
ValueMapping,
|
|
ThresholdsConfig,
|
|
applyNullInsertThreshold,
|
|
nullToValue,
|
|
SpecialValueMatch,
|
|
} from '@grafana/data';
|
|
import { maybeSortFrame, NULL_RETAIN } from '@grafana/data/internal';
|
|
import { t } from '@grafana/i18n';
|
|
import {
|
|
VizLegendOptions,
|
|
AxisPlacement,
|
|
ScaleDirection,
|
|
ScaleOrientation,
|
|
VisibilityMode,
|
|
TimelineValueAlignment,
|
|
HideableFieldConfig,
|
|
MappingType,
|
|
} from '@grafana/schema';
|
|
import { FIXED_UNIT, UPlotConfigBuilder, UPlotConfigPrepFn, VizLegendItem } from '@grafana/ui';
|
|
import { preparePlotData2, getStackingGroups } from '@grafana/ui/internal';
|
|
|
|
import { getConfig, TimelineCoreOptions } from './timeline';
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
interface UPlotConfigOptions {
|
|
frame: DataFrame;
|
|
theme: GrafanaTheme2;
|
|
mode: TimelineMode;
|
|
rowHeight?: number;
|
|
colWidth?: number;
|
|
showValue: VisibilityMode;
|
|
alignValue?: TimelineValueAlignment;
|
|
mergeValues?: boolean;
|
|
getValueColor: (frameIdx: number, fieldIdx: number, value: unknown) => string;
|
|
hoverMulti: boolean;
|
|
axisWidth?: number;
|
|
}
|
|
|
|
/**
|
|
* @internal
|
|
*/
|
|
interface PanelFieldConfig extends HideableFieldConfig {
|
|
fillOpacity?: number;
|
|
lineWidth?: number;
|
|
}
|
|
|
|
export enum TimelineMode {
|
|
Changes = 'changes',
|
|
Samples = 'samples',
|
|
}
|
|
|
|
const defaultConfig: PanelFieldConfig = {
|
|
lineWidth: 0,
|
|
fillOpacity: 80,
|
|
};
|
|
|
|
/** Checks if a mapped value of the specified type exists for the given field */
|
|
export const hasSpecialMappedValue = (field: Field, match: SpecialValueMatch): boolean =>
|
|
field.config.mappings?.some(
|
|
(mapping: ValueMapping): boolean => mapping.type === MappingType.SpecialValue && mapping.options.match === match
|
|
) || false;
|
|
|
|
export const preparePlotConfigBuilder: UPlotConfigPrepFn<UPlotConfigOptions> = ({
|
|
frame,
|
|
theme,
|
|
timeZones,
|
|
getTimeRange,
|
|
mode,
|
|
rowHeight,
|
|
colWidth,
|
|
showValue,
|
|
alignValue,
|
|
mergeValues,
|
|
getValueColor,
|
|
hoverMulti,
|
|
}) => {
|
|
const builder = new UPlotConfigBuilder(timeZones[0]);
|
|
|
|
const xScaleKey = 'x';
|
|
|
|
const isDiscrete = (field: Field) => {
|
|
const mode = field.config?.color?.mode;
|
|
return !(mode && field.display && mode.startsWith('continuous-'));
|
|
};
|
|
const getValueColorFn = (seriesIdx: number, value: unknown) => {
|
|
const field = frame.fields[seriesIdx];
|
|
|
|
if (
|
|
field.state?.origin?.fieldIndex !== undefined &&
|
|
field.state?.origin?.frameIndex !== undefined &&
|
|
getValueColor
|
|
) {
|
|
return getValueColor(field.state?.origin?.frameIndex, field.state?.origin?.fieldIndex, value);
|
|
}
|
|
|
|
return FALLBACK_COLOR;
|
|
};
|
|
|
|
const opts: TimelineCoreOptions = {
|
|
mode: mode!,
|
|
numSeries: frame.fields.length - 1,
|
|
isDiscrete: (seriesIdx) => isDiscrete(frame.fields[seriesIdx]),
|
|
hasMappedNull: (seriesIdx) =>
|
|
hasSpecialMappedValue(frame.fields[seriesIdx], SpecialValueMatch.Null) ||
|
|
hasSpecialMappedValue(frame.fields[seriesIdx], SpecialValueMatch.NullAndNaN),
|
|
hasMappedNaN: (seriesIdx) =>
|
|
hasSpecialMappedValue(frame.fields[seriesIdx], SpecialValueMatch.NaN) ||
|
|
hasSpecialMappedValue(frame.fields[seriesIdx], SpecialValueMatch.NullAndNaN),
|
|
mergeValues,
|
|
rowHeight: rowHeight,
|
|
colWidth: colWidth,
|
|
showValue: showValue!,
|
|
alignValue,
|
|
theme,
|
|
label: (seriesIdx) => getFieldDisplayName(frame.fields[seriesIdx], frame),
|
|
getFieldConfig: (seriesIdx) => frame.fields[seriesIdx].config.custom,
|
|
getValueColor: getValueColorFn,
|
|
getTimeRange,
|
|
// hardcoded formatter for state values
|
|
formatValue: (seriesIdx, value) => formattedValueToString(frame.fields[seriesIdx].display!(value)),
|
|
hoverMulti,
|
|
};
|
|
|
|
const coreConfig = getConfig(opts);
|
|
|
|
builder.addHook('init', coreConfig.init);
|
|
builder.addHook('drawClear', coreConfig.drawClear);
|
|
|
|
builder.setPrepData((frames) => preparePlotData2(frames[0], getStackingGroups(frames[0])));
|
|
|
|
builder.setCursor(coreConfig.cursor);
|
|
|
|
builder.addScale({
|
|
scaleKey: xScaleKey,
|
|
isTime: true,
|
|
orientation: ScaleOrientation.Horizontal,
|
|
direction: ScaleDirection.Right,
|
|
range: coreConfig.xRange,
|
|
});
|
|
|
|
builder.addScale({
|
|
scaleKey: FIXED_UNIT, // y
|
|
isTime: false,
|
|
orientation: ScaleOrientation.Vertical,
|
|
direction: ScaleDirection.Up,
|
|
range: coreConfig.yRange,
|
|
});
|
|
|
|
const xAxisHidden = frame.fields[0].config.custom.axisPlacement === AxisPlacement.Hidden;
|
|
|
|
builder.addAxis({
|
|
show: !xAxisHidden,
|
|
scaleKey: xScaleKey,
|
|
isTime: true,
|
|
splits: coreConfig.xSplits!,
|
|
placement: AxisPlacement.Bottom,
|
|
timeZone: timeZones[0],
|
|
theme,
|
|
});
|
|
|
|
const yCustomConfig = frame.fields[1].config.custom;
|
|
const yAxisWidth = yCustomConfig.axisWidth;
|
|
const yAxisHidden = yCustomConfig.axisPlacement === AxisPlacement.Hidden;
|
|
|
|
builder.addAxis({
|
|
scaleKey: FIXED_UNIT, // y
|
|
isTime: false,
|
|
placement: AxisPlacement.Left,
|
|
splits: coreConfig.ySplits,
|
|
values: yAxisHidden ? (u, splits) => splits.map((v) => null) : coreConfig.yValues,
|
|
grid: { show: false },
|
|
ticks: { show: false },
|
|
gap: yAxisHidden ? 0 : 16,
|
|
size: yAxisHidden ? 0 : yAxisWidth,
|
|
theme,
|
|
});
|
|
|
|
let seriesIndex = 0;
|
|
|
|
for (let i = 0; i < frame.fields.length; i++) {
|
|
if (i === 0) {
|
|
continue;
|
|
}
|
|
|
|
const field = frame.fields[i];
|
|
const config: FieldConfig<PanelFieldConfig> = field.config;
|
|
const customConfig: PanelFieldConfig = {
|
|
...defaultConfig,
|
|
...config.custom,
|
|
};
|
|
|
|
field.state!.seriesIndex = seriesIndex++;
|
|
|
|
// const scaleKey = config.unit || FIXED_UNIT;
|
|
// const colorMode = getFieldColorModeForField(field);
|
|
|
|
builder.addSeries({
|
|
scaleKey: FIXED_UNIT,
|
|
pathBuilder: coreConfig.drawPaths,
|
|
pointsBuilder: coreConfig.drawPoints,
|
|
//colorMode,
|
|
lineWidth: customConfig.lineWidth,
|
|
fillOpacity: customConfig.fillOpacity,
|
|
theme,
|
|
show: !customConfig.hideFrom?.viz,
|
|
thresholds: config.thresholds,
|
|
// The following properties are not used in the uPlot config, but are utilized as transport for legend config
|
|
dataFrameFieldIndex: field.state?.origin,
|
|
});
|
|
}
|
|
|
|
return builder;
|
|
};
|
|
|
|
function getSpanNulls(field: Field) {
|
|
let spanNulls = field.config.custom?.spanNulls;
|
|
|
|
// magic value for join() to leave nulls alone instead of expanding null ranges
|
|
// should be set to -1 when spanNulls = null|undefined|false|0, which is "retain nulls, without expanding"
|
|
// Infinity is not optimal here since it causes spanNulls to be more expensive than simply removing all nulls unconditionally
|
|
return !spanNulls ? -1 : spanNulls === true ? Infinity : spanNulls;
|
|
}
|
|
|
|
/**
|
|
* Merge values by the threshold
|
|
*/
|
|
export function mergeThresholdValues(field: Field, theme: GrafanaTheme2): Field | undefined {
|
|
const thresholds = field.config.thresholds;
|
|
if (field.type !== FieldType.number || !thresholds || !thresholds.steps.length) {
|
|
return undefined;
|
|
}
|
|
|
|
const items = getThresholdItems(field.config, theme);
|
|
if (items.length !== thresholds.steps.length) {
|
|
return undefined; // should not happen
|
|
}
|
|
|
|
const thresholdToText = new Map<Threshold, string>();
|
|
const textToColor = new Map<string, string>();
|
|
for (let i = 0; i < items.length; i++) {
|
|
thresholdToText.set(thresholds.steps[i], items[i].label);
|
|
textToColor.set(items[i].label, items[i].color!);
|
|
}
|
|
|
|
let input = field.values;
|
|
const vals = new Array<String | undefined>(field.values.length);
|
|
if (thresholds.mode === ThresholdsMode.Percentage) {
|
|
const { min, max } = getFieldConfigWithMinMax(field);
|
|
const delta = max! - min!;
|
|
input = input.map((v) => {
|
|
if (v == null) {
|
|
return v;
|
|
}
|
|
return ((v - min!) / delta) * 100;
|
|
});
|
|
}
|
|
|
|
for (let i = 0; i < vals.length; i++) {
|
|
const v = input[i];
|
|
if (v == null) {
|
|
vals[i] = v;
|
|
} else {
|
|
vals[i] = thresholdToText.get(getActiveThreshold(v, thresholds.steps));
|
|
}
|
|
}
|
|
|
|
return {
|
|
...field,
|
|
config: {
|
|
...field.config,
|
|
custom: {
|
|
...field.config.custom,
|
|
spanNulls: getSpanNulls(field),
|
|
},
|
|
},
|
|
type: FieldType.string,
|
|
values: vals,
|
|
display: (value) => ({
|
|
text: String(value),
|
|
color: textToColor.get(String(value)),
|
|
numeric: NaN,
|
|
}),
|
|
};
|
|
}
|
|
|
|
// This will return a set of frames with only graphable values included
|
|
export function prepareTimelineFields(
|
|
series: DataFrame[] | undefined,
|
|
mergeValues: boolean,
|
|
timeRange: TimeRange,
|
|
theme: GrafanaTheme2
|
|
): { frames?: DataFrame[]; warn?: string } {
|
|
// this allows PanelDataErrorView to show the default noValue message
|
|
if (!series?.length) {
|
|
return { warn: '' };
|
|
}
|
|
|
|
cacheFieldDisplayNames(series);
|
|
|
|
let hasTimeseries = false;
|
|
const frames: DataFrame[] = [];
|
|
|
|
for (let frame of series) {
|
|
let startFieldIdx = -1;
|
|
let endFieldIdx = -1;
|
|
|
|
for (let i = 0; i < frame.fields.length; i++) {
|
|
let f = frame.fields[i];
|
|
|
|
if (f.type === FieldType.time) {
|
|
if (startFieldIdx === -1) {
|
|
startFieldIdx = i;
|
|
} else if (endFieldIdx === -1) {
|
|
endFieldIdx = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
let isTimeseries = startFieldIdx !== -1;
|
|
let changed = false;
|
|
frame = maybeSortFrame(frame, startFieldIdx);
|
|
|
|
// if we have a second time field, assume it is state end timestamps
|
|
// and insert nulls into the data at the end timestamps
|
|
if (endFieldIdx !== -1) {
|
|
let startFrame: DataFrame = {
|
|
...frame,
|
|
fields: frame.fields.filter((f, i) => i !== endFieldIdx),
|
|
};
|
|
|
|
let endFrame: DataFrame = {
|
|
length: frame.length,
|
|
fields: [frame.fields[endFieldIdx]],
|
|
};
|
|
|
|
frame = outerJoinDataFrames({
|
|
frames: [startFrame, endFrame],
|
|
keepDisplayNames: true,
|
|
nullMode: () => NULL_RETAIN,
|
|
})!;
|
|
|
|
frame.fields.forEach((f, i) => {
|
|
if (i > 0) {
|
|
let vals = f.values;
|
|
for (let i = 0; i < vals.length; i++) {
|
|
if (vals[i] == null) {
|
|
vals[i] = null;
|
|
}
|
|
}
|
|
}
|
|
});
|
|
|
|
changed = true;
|
|
}
|
|
|
|
let nulledFrame = applyNullInsertThreshold({
|
|
frame,
|
|
refFieldPseudoMin: timeRange.from.valueOf(),
|
|
refFieldPseudoMax: timeRange.to.valueOf(),
|
|
});
|
|
|
|
if (nulledFrame !== frame) {
|
|
changed = true;
|
|
}
|
|
|
|
frame = nullToValue(nulledFrame);
|
|
|
|
const fields: Field[] = [];
|
|
for (let field of frame.fields) {
|
|
switch (field.type) {
|
|
case FieldType.time:
|
|
isTimeseries = true;
|
|
hasTimeseries = true;
|
|
fields.push(field);
|
|
break;
|
|
case FieldType.enum:
|
|
case FieldType.number:
|
|
if (mergeValues && field.config.color?.mode === FieldColorModeId.Thresholds) {
|
|
const f = mergeThresholdValues(field, theme);
|
|
if (f) {
|
|
fields.push(f);
|
|
changed = true;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
case FieldType.boolean:
|
|
case FieldType.string:
|
|
field = {
|
|
...field,
|
|
config: {
|
|
...field.config,
|
|
custom: {
|
|
...field.config.custom,
|
|
spanNulls: getSpanNulls(field),
|
|
},
|
|
},
|
|
};
|
|
changed = true;
|
|
fields.push(field);
|
|
break;
|
|
default:
|
|
changed = true;
|
|
}
|
|
}
|
|
if (isTimeseries && fields.length > 1) {
|
|
hasTimeseries = true;
|
|
if (changed) {
|
|
frames.push({
|
|
...frame,
|
|
fields,
|
|
});
|
|
} else {
|
|
frames.push(frame);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!hasTimeseries) {
|
|
return { warn: t('timeline.missing-field.time', 'Data does not have a time field') };
|
|
}
|
|
if (!frames.length) {
|
|
return { warn: t('timeline.missing-field.all', 'No graphable fields') };
|
|
}
|
|
|
|
return { frames };
|
|
}
|
|
|
|
export function makeFramePerSeries(frames: DataFrame[]) {
|
|
const outFrames: DataFrame[] = [];
|
|
|
|
for (let frame of frames) {
|
|
const timeFields = frame.fields.filter((field) => field.type === FieldType.time);
|
|
|
|
if (timeFields.length > 0) {
|
|
for (let field of frame.fields) {
|
|
if (field.type !== FieldType.time) {
|
|
outFrames.push({ fields: [...timeFields, field], length: frame.length });
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return outFrames;
|
|
}
|
|
|
|
export function getThresholdItems(
|
|
fieldConfig: FieldConfig,
|
|
theme: GrafanaTheme2,
|
|
thresholdItems?: ThresholdsConfig
|
|
): VizLegendItem[] {
|
|
const items: VizLegendItem[] = [];
|
|
const thresholds = thresholdItems ? thresholdItems : fieldConfig.thresholds;
|
|
if (!thresholds || !thresholds.steps.length) {
|
|
return items;
|
|
}
|
|
|
|
const steps = thresholds.steps;
|
|
const getDisplay = getValueFormat(
|
|
thresholds.mode === ThresholdsMode.Percentage ? 'percent' : (fieldConfig.unit ?? '')
|
|
);
|
|
|
|
// `undefined` value for decimals will use `auto`
|
|
const format = (value: number) => formattedValueToString(getDisplay(value, fieldConfig.decimals ?? undefined));
|
|
|
|
for (let i = 0; i < steps.length; i++) {
|
|
let step = steps[i];
|
|
let value = step.value;
|
|
let pre = '';
|
|
let suf = '';
|
|
|
|
if (value === -Infinity && i < steps.length - 1) {
|
|
value = steps[i + 1].value;
|
|
pre = '< ';
|
|
} else {
|
|
suf = '+';
|
|
}
|
|
|
|
items.push({
|
|
label: `${pre}${format(value)}${suf}`,
|
|
color: theme.visualization.getColorByName(step.color),
|
|
yAxis: 1,
|
|
});
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
export function getValueMappingItems(mappings: ValueMapping[], theme: GrafanaTheme2): VizLegendItem[] {
|
|
const items: VizLegendItem[] = [];
|
|
if (!mappings) {
|
|
return items;
|
|
}
|
|
|
|
for (let mapping of mappings) {
|
|
const { options, type } = mapping;
|
|
|
|
if (type === MappingType.ValueToText) {
|
|
for (let [label, value] of Object.entries(options)) {
|
|
const color = value.color;
|
|
items.push({
|
|
label: label,
|
|
color: theme.visualization.getColorByName(color ?? FALLBACK_COLOR),
|
|
yAxis: 1,
|
|
});
|
|
}
|
|
}
|
|
|
|
if (type === MappingType.RangeToText) {
|
|
const { from, result, to } = options;
|
|
const { text, color } = result;
|
|
const label = text ? `[${from} - ${to}] ${text}` : `[${from} - ${to}]`;
|
|
|
|
items.push({
|
|
label: label,
|
|
color: theme.visualization.getColorByName(color ?? FALLBACK_COLOR),
|
|
yAxis: 1,
|
|
});
|
|
}
|
|
|
|
if (type === MappingType.RegexToText) {
|
|
const { pattern, result } = options;
|
|
const { text, color } = result;
|
|
const label = `${text || pattern}`;
|
|
|
|
items.push({
|
|
label: label,
|
|
color: theme.visualization.getColorByName(color ?? FALLBACK_COLOR),
|
|
yAxis: 1,
|
|
});
|
|
}
|
|
|
|
if (type === MappingType.SpecialValue) {
|
|
const { match, result } = options;
|
|
const { text, color } = result;
|
|
const label = `${text || match}`;
|
|
|
|
items.push({
|
|
label: label,
|
|
color: theme.visualization.getColorByName(color ?? FALLBACK_COLOR),
|
|
yAxis: 1,
|
|
});
|
|
}
|
|
}
|
|
|
|
return items;
|
|
}
|
|
|
|
export function prepareTimelineLegendItems(
|
|
frames: DataFrame[] | undefined,
|
|
options: VizLegendOptions,
|
|
theme: GrafanaTheme2
|
|
): VizLegendItem[] | undefined {
|
|
if (!frames || options.showLegend === false) {
|
|
return undefined;
|
|
}
|
|
|
|
return getFieldLegendItem(allNonTimeFields(frames), theme);
|
|
}
|
|
|
|
export function getFieldLegendItem(fields: Field[], theme: GrafanaTheme2): VizLegendItem[] | undefined {
|
|
if (!fields.length) {
|
|
return undefined;
|
|
}
|
|
|
|
const items: VizLegendItem[] = [];
|
|
const fieldConfig = fields[0].config;
|
|
const colorMode = fieldConfig.color?.mode ?? FieldColorModeId.Fixed;
|
|
const thresholds = fieldConfig.thresholds;
|
|
|
|
// If thresholds are enabled show each step in the legend
|
|
// This ignores the hide from legend since the range is valid
|
|
if (colorMode === FieldColorModeId.Thresholds && thresholds?.steps && thresholds.steps.length > 1) {
|
|
return getThresholdItems(fieldConfig, theme);
|
|
}
|
|
|
|
// If thresholds are enabled show each step in the legend
|
|
if (colorMode.startsWith('continuous')) {
|
|
return undefined; // eventually a color bar
|
|
}
|
|
|
|
const stateColors: Map<string, string | undefined> = new Map();
|
|
|
|
fields.forEach((field) => {
|
|
if (!field.config.custom?.hideFrom?.legend) {
|
|
field.values.forEach((v) => {
|
|
let state = field.display!(v);
|
|
if (state.color) {
|
|
stateColors.set(state.text, state.color!);
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
stateColors.forEach((color, label) => {
|
|
if (label.length > 0) {
|
|
items.push({
|
|
label: label!,
|
|
color: theme.visualization.getColorByName(color ?? FALLBACK_COLOR),
|
|
yAxis: 1,
|
|
});
|
|
}
|
|
});
|
|
|
|
return items;
|
|
}
|
|
|
|
function allNonTimeFields(frames: DataFrame[]): Field[] {
|
|
const fields: Field[] = [];
|
|
for (const frame of frames) {
|
|
for (const field of frame.fields) {
|
|
if (field.type !== FieldType.time) {
|
|
fields.push(field);
|
|
}
|
|
}
|
|
}
|
|
return fields;
|
|
}
|
|
|
|
export function findNextStateIndex(field: Field, datapointIdx: number) {
|
|
let end;
|
|
let rightPointer = datapointIdx + 1;
|
|
|
|
if (rightPointer >= field.values.length) {
|
|
return null;
|
|
}
|
|
|
|
const startValue = field.values[datapointIdx];
|
|
|
|
while (end === undefined) {
|
|
if (rightPointer >= field.values.length) {
|
|
return null;
|
|
}
|
|
const rightValue = field.values[rightPointer];
|
|
|
|
if (rightValue === undefined || rightValue === startValue) {
|
|
rightPointer++;
|
|
} else {
|
|
end = rightPointer;
|
|
}
|
|
}
|
|
|
|
return end;
|
|
}
|
|
|
|
/**
|
|
* Returns the precise duration of a time range passed in milliseconds.
|
|
* This function calculates with 30 days month and 365 days year.
|
|
* adapted from https://gist.github.com/remino/1563878
|
|
* @param milliSeconds The duration in milliseconds
|
|
* @returns A formatted string of the duration
|
|
*/
|
|
export function fmtDuration(milliSeconds: number): string {
|
|
if (milliSeconds < 0 || Number.isNaN(milliSeconds)) {
|
|
return '';
|
|
}
|
|
|
|
let yr: number, mo: number, wk: number, d: number, h: number, m: number, s: number, ms: number;
|
|
|
|
s = Math.floor(milliSeconds / 1000);
|
|
m = Math.floor(s / 60);
|
|
s = s % 60;
|
|
h = Math.floor(m / 60);
|
|
m = m % 60;
|
|
d = Math.floor(h / 24);
|
|
h = h % 24;
|
|
|
|
yr = Math.floor(d / 365);
|
|
if (yr > 0) {
|
|
d = d % 365;
|
|
}
|
|
|
|
mo = Math.floor(d / 30);
|
|
if (mo > 0) {
|
|
d = d % 30;
|
|
}
|
|
|
|
wk = Math.floor(d / 7);
|
|
|
|
if (wk > 0) {
|
|
d = d % 7;
|
|
}
|
|
|
|
ms = Math.round((milliSeconds % 1000) * 1000) / 1000;
|
|
|
|
return (
|
|
yr > 0
|
|
? yr + 'y ' + (mo > 0 ? mo + 'mo ' : '') + (wk > 0 ? wk + 'w ' : '') + (d > 0 ? d + 'd ' : '')
|
|
: mo > 0
|
|
? mo + 'mo ' + (wk > 0 ? wk + 'w ' : '') + (d > 0 ? d + 'd ' : '')
|
|
: wk > 0
|
|
? wk + 'w ' + (d > 0 ? d + 'd ' : '')
|
|
: d > 0
|
|
? d + 'd ' + (h > 0 ? h + 'h ' : '')
|
|
: h > 0
|
|
? h + 'h ' + (m > 0 ? m + 'm ' : '')
|
|
: m > 0
|
|
? m + 'm ' + (s > 0 ? s + 's ' : '')
|
|
: s > 0
|
|
? s + 's ' + (ms > 0 ? ms + 'ms ' : '')
|
|
: ms > 0
|
|
? ms + 'ms '
|
|
: '0'
|
|
).trim();
|
|
}
|