blob: 53ec13e0409d6fb63b412ee03a04ab1e880e41e0 [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import * as zrUtil from 'zrender/src/core/util';
import {retrieveRawValue} from '../../data/helper/dataProvider';
import {formatTpl} from '../../util/format';
import {
DataHost,
DisplayState,
CallbackDataParams,
ColorString,
ZRColor,
OptionDataValue,
SeriesDataType,
ComponentMainType,
ComponentSubType,
DimensionLoose
} from '../../util/types';
import GlobalModel from '../Global';
import { TooltipMarkupBlockFragment } from '../../component/tooltip/tooltipMarkup';
import { makePrintable } from '../../util/log';
const DIMENSION_LABEL_REG = /\{@(.+?)\}/g;
export interface DataFormatMixin extends DataHost {
ecModel: GlobalModel;
mainType: ComponentMainType;
subType: ComponentSubType;
componentIndex: number;
id: string;
name: string;
animatedValue: OptionDataValue[];
}
export class DataFormatMixin {
/**
* Get params for formatter
*/
getDataParams(
dataIndex: number,
dataType?: SeriesDataType
): CallbackDataParams {
const data = this.getData(dataType);
const rawValue = this.getRawValue(dataIndex, dataType);
const rawDataIndex = data.getRawIndex(dataIndex);
const name = data.getName(dataIndex);
const itemOpt = data.getRawDataItem(dataIndex);
const style = data.getItemVisual(dataIndex, 'style');
const color = style && style[data.getItemVisual(dataIndex, 'drawType') || 'fill'] as ZRColor;
const borderColor = style && style.stroke as ColorString;
const mainType = this.mainType;
const isSeries = mainType === 'series';
const userOutput = data.userOutput;
return {
componentType: mainType,
componentSubType: this.subType,
componentIndex: this.componentIndex,
seriesType: isSeries ? this.subType : null,
seriesIndex: (this as any).seriesIndex,
seriesId: isSeries ? this.id : null,
seriesName: isSeries ? this.name : null,
name: name,
dataIndex: rawDataIndex,
data: itemOpt,
dataType: dataType,
value: rawValue,
color: color,
borderColor: borderColor,
dimensionNames: userOutput ? userOutput.dimensionNames : null,
encode: userOutput ? userOutput.encode : null,
// Param name list for mapping `a`, `b`, `c`, `d`, `e`
$vars: ['seriesName', 'name', 'value']
};
}
/**
* Format label
* @param dataIndex
* @param status 'normal' by default
* @param dataType
* @param labelDimIndex Only used in some chart that
* use formatter in different dimensions, like radar.
* @param formatter Formatter given outside.
* @return return null/undefined if no formatter
*/
getFormattedLabel(
dataIndex: number,
status?: DisplayState,
dataType?: SeriesDataType,
labelDimIndex?: number,
formatter?: string | ((params: object) => string),
extendParams?: Partial<CallbackDataParams>
): string {
status = status || 'normal';
const data = this.getData(dataType);
const params = this.getDataParams(dataIndex, dataType);
if (extendParams) {
zrUtil.extend(params, extendParams);
}
if (labelDimIndex != null && zrUtil.isArray(params.value)) {
params.value = params.value[labelDimIndex];
}
if (!formatter) {
const itemModel = data.getItemModel(dataIndex);
// @ts-ignore
formatter = itemModel.get(status === 'normal'
? ['label', 'formatter']
: [status, 'label', 'formatter']
);
}
if (typeof formatter === 'function') {
params.status = status;
params.dimensionIndex = labelDimIndex;
return formatter(params);
}
else if (typeof formatter === 'string') {
const str = formatTpl(formatter, params);
// Support 'aaa{@[3]}bbb{@product}ccc'.
// Do not support '}' in dim name util have to.
return str.replace(DIMENSION_LABEL_REG, function (origin, dimStr: string) {
const len = dimStr.length;
const dimLoose: DimensionLoose = (dimStr.charAt(0) === '[' && dimStr.charAt(len - 1) === ']')
? +dimStr.slice(1, len - 1) // Also support: '[]' => 0
: dimStr;
let val = retrieveRawValue(data, dataIndex, dimLoose) as OptionDataValue;
// Tricky: `extendParams.value` is only used in interpolate case
// (label animation) currently.
if (extendParams && zrUtil.isArray(extendParams.value)) {
const dimInfo = data.getDimensionInfo(dimLoose);
if (dimInfo) {
val = extendParams.value[dimInfo.index];
}
}
return val != null ? val + '' : '';
});
}
}
/**
* Get raw value in option
*/
getRawValue(
idx: number,
dataType?: SeriesDataType
): unknown {
return retrieveRawValue(this.getData(dataType), idx);
}
/**
* Should be implemented.
* @param {number} dataIndex
* @param {boolean} [multipleSeries=false]
* @param {string} [dataType]
*/
formatTooltip(
dataIndex: number,
multipleSeries?: boolean,
dataType?: string
): TooltipFormatResult {
// Empty function
return;
}
};
type TooltipFormatResult =
// If `string`, means `TooltipFormatResultLegacyObject['html']`
string
// | TooltipFormatResultLegacyObject
| TooltipMarkupBlockFragment;
// PENDING: previously we accept this type when calling `formatTooltip`,
// but guess little chance has been used outside. Do we need to backward
// compat it?
// type TooltipFormatResultLegacyObject = {
// // `html` means the markup language text, either in 'html' or 'richText'.
// // The name `html` is not appropriate becuase in 'richText' it is not a HTML
// // string. But still support it for backward compat.
// html: string;
// markers: Dictionary<ColorString>;
// };
/**
* For backward compat, normalize the return from `formatTooltip`.
*/
export function normalizeTooltipFormatResult(
result: TooltipFormatResult
// markersExisting: Dictionary<ColorString>
): {
// If `markupFragment` exists, `markupText` should be ignored.
markupFragment: TooltipMarkupBlockFragment;
// Can be `null`/`undefined`, means no tooltip.
markupText: string;
// Merged with `markersExisting`.
// markers: Dictionary<ColorString>;
} {
let markupText;
// let markers: Dictionary<ColorString>;
let markupFragment: TooltipMarkupBlockFragment;
if (zrUtil.isObject(result)) {
if ((result as TooltipMarkupBlockFragment).type) {
markupFragment = result as TooltipMarkupBlockFragment;
}
else {
if (__DEV__) {
console.warn('The return type of `formatTooltip` is not supported: ' + makePrintable(result));
}
}
// else {
// markupText = (result as TooltipFormatResultLegacyObject).html;
// markers = (result as TooltipFormatResultLegacyObject).markers;
// if (markersExisting) {
// markers = zrUtil.merge(markersExisting, markers);
// }
// }
}
else {
markupText = result;
}
return {
markupText: markupText,
// markers: markers || markersExisting,
markupFragment: markupFragment
};
}