未验证 提交 9963aa4f 编写于 作者: S sushuang 提交者: GitHub

Merge pull request #12775 from apache/custom-series-enhance

Custom series enhancement (for-next)
此差异已折叠。
......@@ -42,11 +42,18 @@ class RadiusAxisView extends AxisView {
axisPointerClass = 'PolarAxisPointer';
private _axisGroup: graphic.Group;
render(radiusAxisModel: RadiusAxisModel, ecModel: GlobalModel) {
this.group.removeAll();
if (!radiusAxisModel.get('show')) {
return;
}
const oldAxisGroup = this._axisGroup;
const newAxisGroup = this._axisGroup = new graphic.Group();
this.group.add(newAxisGroup);
const radiusAxis = radiusAxisModel.axis;
const polar = radiusAxis.polar;
const angleAxis = polar.getAngleAxis();
......@@ -58,7 +65,9 @@ class RadiusAxisView extends AxisView {
const layout = layoutAxis(polar, radiusAxisModel, axisAngle);
const axisBuilder = new AxisBuilder(radiusAxisModel, layout);
zrUtil.each(axisBuilderAttrs, axisBuilder.add, axisBuilder);
this.group.add(axisBuilder.getGroup());
newAxisGroup.add(axisBuilder.getGroup());
graphic.groupTransition(oldAxisGroup, newAxisGroup, radiusAxisModel);
zrUtil.each(selfBuilderAttrs, function (name) {
if (radiusAxisModel.get([name, 'show']) && !radiusAxis.scale.isBlank()) {
......
......@@ -185,7 +185,7 @@ class PiecewiseModel extends VisualMapModel<PiecewiseVisualMapOption> {
const isCategory = this.isCategory();
zrUtil.each(option.pieces, function (piece) {
zrUtil.each(visualTypes, function (visualType) {
zrUtil.each(visualTypes, function (visualType: BuiltinVisualProperty) {
if (piece.hasOwnProperty(visualType)) {
visualTypesInPieces[visualType] = 1;
}
......
......@@ -26,6 +26,7 @@ import { BoundingRect } from '../util/graphic';
import { MatrixArray } from 'zrender/src/core/matrix';
import ComponentModel from '../model/Component';
import { RectLike } from 'zrender/src/core/BoundingRect';
import { PrepareCustomInfo } from '../chart/custom';
export interface CoordinateSystemCreator {
......@@ -151,6 +152,8 @@ export interface CoordinateSystem {
// Currently only Cartesian2D implements it.
// But if other coordinate systems implement it, should follow this signature.
getAxesByScale?: (scaleType: string) => Axis[];
prepareCustoms?: PrepareCustomInfo;
}
/**
......
......@@ -24,6 +24,7 @@ import RadiusAxis from './RadiusAxis';
function dataToCoordSize(this: Polar, dataSize: number[], dataItem: number[]) {
// dataItem is necessary in log axis.
dataItem = dataItem || [0, 0];
return zrUtil.map(['Radius', 'Angle'], function (dim, dimIdx) {
const getterName = 'get' + dim + 'Axis' as 'getAngleAxis'| 'getRadiusAxis';
// TODO: TYPE Check Angle Axis
......
......@@ -1802,6 +1802,9 @@ class ECharts extends Eventful {
};
updateZ = function (model: ComponentModel, view: ComponentView | ChartView): void {
if (model.preventAutoZ) {
return;
}
const z = model.get('z');
const zlevel = model.get('zlevel');
// Set z and zlevel
......@@ -1816,6 +1819,7 @@ class ECharts extends Eventful {
textContent.z = el.z;
textContent.zlevel = el.zlevel;
// lift z2 of text content
// TODO if el.emphasis.z2 is spcefied, what about textContent.
textContent.z2 = el.z2 + 1;
}
}
......
......@@ -127,6 +127,11 @@ class ComponentModel<Opt extends ComponentOption = ComponentOption> extends Mode
*/
static layoutMode: ComponentLayoutMode | ComponentLayoutMode['type'];
/**
* Prevent from auto set z, zlevel, z2 by the framework.
*/
preventAutoZ: boolean;
// Injectable properties:
__viewId: string;
......
......@@ -44,7 +44,7 @@ type ItemStyleKeys = 'fill'
| 'shadowOffsetY'
| 'shadowColor';
type ItemStyleProps = Pick<PathStyleProps, ItemStyleKeys>;
export type ItemStyleProps = Pick<PathStyleProps, ItemStyleKeys>;
class ItemStyleMixin {
......
......@@ -59,6 +59,7 @@ import {
DataModel,
ECEventData,
ZRStyleProps,
TextCommonOption,
SeriesOption,
ParsedValue,
CallbackDataParams
......@@ -73,13 +74,13 @@ import {
trim,
isArrayLike,
map,
defaults
defaults,
isObject
} from 'zrender/src/core/util';
import * as numberUtil from './number';
import SeriesModel from '../model/Series';
import {OnframeCallback, interpolateNumber} from 'zrender/src/animation/Animator';
import {interpolateNumber} from 'zrender/src/animation/Animator';
import List from '../data/List';
import DataFormatMixin from '../model/mixin/dataFormat';
const mathMax = Math.max;
......@@ -89,13 +90,6 @@ const EMPTY_OBJ = {};
export const Z2_EMPHASIS_LIFT = 10;
// key: label model property nane, value: style property name.
export const CACHED_LABEL_STYLE_PROPERTIES = {
color: 'textFill',
textBorderColor: 'textStroke',
textBorderWidth: 'textStrokeWidth'
};
const EMPHASIS = 'emphasis';
const NORMAL = 'normal';
......@@ -134,8 +128,6 @@ type TextCommonParams = {
forceRich?: boolean
getTextPosition?: (textStyleModel: Model, isEmphasis?: boolean) => string | string[] | number[]
defaultOutsidePosition?: LabelOption['position']
textStyle?: ZRStyleProps
......@@ -398,12 +390,14 @@ function singleEnterEmphasis(el: Element) {
if (!hasFillOrStroke(emphasisStyle.stroke)) {
disp.style.stroke = liftColor(currentStroke);
}
disp.z2 += Z2_EMPHASIS_LIFT;
const z2EmphasisLift = (disp as ECElement).z2EmphasisLift;
disp.z2 += z2EmphasisLift != null ? z2EmphasisLift : Z2_EMPHASIS_LIFT;
}
const textContent = el.getTextContent();
if (textContent) {
textContent.z2 += Z2_EMPHASIS_LIFT;
const z2EmphasisLift = (textContent as ECElement).z2EmphasisLift;
textContent.z2 += z2EmphasisLift != null ? z2EmphasisLift : Z2_EMPHASIS_LIFT;
}
// TODO hover layer
}
......@@ -793,6 +787,7 @@ export function createTextConfig(
opt: TextCommonParams,
isEmphasis: boolean
) {
opt = opt || {};
const textConfig: ElementTextConfig = {};
let labelPosition;
let labelRotate = textStyleModel.getShallow('rotate');
......@@ -801,16 +796,11 @@ export function createTextConfig(
);
const labelOffset = textStyleModel.getShallow('offset');
if (opt.getTextPosition) {
labelPosition = opt.getTextPosition(textStyleModel, isEmphasis);
}
else {
labelPosition = textStyleModel.getShallow('position')
|| (isEmphasis ? null : 'inside');
// 'outside' is not a valid zr textPostion value, but used
// in bar series, and magric type should be considered.
labelPosition === 'outside' && (labelPosition = opt.defaultOutsidePosition || 'top');
}
labelPosition = textStyleModel.getShallow('position')
|| (isEmphasis ? null : 'inside');
// 'outside' is not a valid zr textPostion value, but used
// in bar series, and magric type should be considered.
labelPosition === 'outside' && (labelPosition = opt.defaultOutsidePosition || 'top');
if (labelPosition != null) {
textConfig.position = labelPosition;
......@@ -1054,7 +1044,10 @@ function setTokenTextStyle(
}
}
export function getFont(opt: LabelOption, ecModel: GlobalModel) {
export function getFont(
opt: Pick<TextCommonOption, 'fontStyle' | 'fontWeight' | 'fontSize' | 'fontFamily'>,
ecModel: GlobalModel
) {
const gTextStyleModel = ecModel && ecModel.getModel('textStyle');
return trim([
// FIXME in node-canvas fontWeight is before fontStyle
......@@ -1065,6 +1058,13 @@ export function getFont(opt: LabelOption, ecModel: GlobalModel) {
].join(' '));
}
type AnimateOrSetPropsOption = {
dataIndex?: number;
cb?: () => void;
during?: (percent: number) => void;
isFrom?: boolean;
};
function animateOrSetProps<Props>(
isUpdate: boolean,
el: Element<Props>,
......@@ -1072,15 +1072,22 @@ function animateOrSetProps<Props>(
animatableModel?: Model<AnimationOptionMixin> & {
getAnimationDelayParams?: (el: Element<Props>, dataIndex: number) => AnimationDelayCallbackParam
},
dataIndex?: number | (() => void),
cb?: () => void,
during?: (percent: number) => void
dataIndex?: AnimateOrSetPropsOption['dataIndex'] | AnimateOrSetPropsOption['cb'] | AnimateOrSetPropsOption,
cb?: AnimateOrSetPropsOption['cb'] | AnimateOrSetPropsOption['during'],
during?: AnimateOrSetPropsOption['during']
) {
let isFrom = false;
if (typeof dataIndex === 'function') {
during = cb;
cb = dataIndex;
dataIndex = null;
}
else if (isObject(dataIndex)) {
cb = dataIndex.cb;
during = dataIndex.during;
isFrom = dataIndex.isFrom;
dataIndex = dataIndex.dataIndex;
}
// Do not check 'animation' property directly here. Consider this case:
// animation model is an `itemModel`, whose does not have `isAnimationEnabled`
// but its parent model (`seriesModel`) does.
......@@ -1109,20 +1116,31 @@ function animateOrSetProps<Props>(
}
duration > 0
? el.animateTo(props, {
duration,
delay: animationDelay || 0,
easing: animationEasing,
done: cb,
force: !!cb || !!during,
during: during
})
: (el.stopAnimation(), el.attr(props), cb && cb());
? (
isFrom
? el.animateFrom(props, {
duration,
delay: animationDelay || 0,
easing: animationEasing,
done: cb,
force: !!cb || !!during,
during: during
})
: el.animateTo(props, {
duration,
delay: animationDelay || 0,
easing: animationEasing,
done: cb,
force: !!cb || !!during,
during: during
})
)
: (el.stopAnimation(), el.attr(props), cb && (cb as AnimateOrSetPropsOption['cb'])());
}
else {
el.stopAnimation();
el.attr(props);
cb && cb();
!isFrom && el.attr(props);
cb && (cb as AnimateOrSetPropsOption['cb'])();
}
}
......@@ -1147,9 +1165,9 @@ function updateProps<Props>(
props: Props,
// TODO: TYPE AnimatableModel
animatableModel?: Model<AnimationOptionMixin>,
dataIndex?: number | (() => void),
cb?: () => void,
during?: () => void
dataIndex?: AnimateOrSetPropsOption['dataIndex'] | AnimateOrSetPropsOption['cb'] | AnimateOrSetPropsOption,
cb?: AnimateOrSetPropsOption['cb'] | AnimateOrSetPropsOption['during'],
during?: AnimateOrSetPropsOption['during']
) {
animateOrSetProps(true, el, props, animatableModel, dataIndex, cb, during);
}
......@@ -1168,9 +1186,9 @@ export function initProps<Props>(
el: Element<Props>,
props: Props,
animatableModel?: Model<AnimationOptionMixin>,
dataIndex?: number | (() => void),
cb?: () => void,
during?: () => void
dataIndex?: AnimateOrSetPropsOption['dataIndex'] | AnimateOrSetPropsOption['cb'] | AnimateOrSetPropsOption,
cb?: AnimateOrSetPropsOption['cb'] | AnimateOrSetPropsOption['during'],
during?: AnimateOrSetPropsOption['during']
) {
animateOrSetProps(false, el, props, animatableModel, dataIndex, cb, during);
}
......
/*
* 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 { Dictionary, ZRStyleProps } from './types';
import { ElementTextConfig } from 'zrender/src/Element';
import { TextStyleProps, TextStylePropsPart, TextProps } from 'zrender/src/graphic/Text';
import { each, hasOwn } from 'zrender/src/core/util';
import { __DEV__ } from '../config';
import { ItemStyleProps } from '../model/mixin/itemStyle';
export interface LegacyStyleProps {
legacy?: boolean
}
const deprecatedLogs = {} as Dictionary<boolean>;
/**
* Whether need to call `convertEC4CompatibleStyle`.
*/
export function isEC4CompatibleStyle(
style: ZRStyleProps & LegacyStyleProps,
elType: string,
hasOwnTextContentOption: boolean,
hasOwnTextConfig: boolean
): boolean {
// Since echarts5, `RectText` is separated from its host element and style.text
// does not exist any more. The compat work brings some extra burden on performance.
// So we provide:
// `legacy: true` force make compat.
// `legacy: false`, force do not compat.
// `legacy` not set: auto detect wheter legacy.
// But in this case we do not compat (difficult to detect and rare case):
// Becuse custom series and graphic component support "merge", users may firstly
// only set `textStrokeWidth` style or secondly only set `text`.
return style && (
style.legacy
|| (
style.legacy !== false
&& !hasOwnTextContentOption
&& !hasOwnTextConfig
&& elType !== 'tspan'
// Difficult to detect whether legacy for a "text" el.
&& (elType === 'text' || hasOwn(style, 'text'))
)
);
}
/**
* `EC4CompatibleStyle` is style that might be in echarts4 format or echarts5 format.
* @param hostStyle The properties might be modified.
* @return If be text el, `textContentStyle` and `textConfig` will not be retured.
* Otherwise a `textContentStyle` and `textConfig` will be created, whose props area
* retried from the `hostStyle`.
*/
export function convertFromEC4CompatibleStyle(hostStyle: ZRStyleProps, elType: string, isNormal: boolean): {
textContent: TextProps & {type: string},
textConfig: ElementTextConfig
} {
const srcStyle = hostStyle as Dictionary<any>;
let textConfig: ElementTextConfig;
let textContent: TextProps & {type: string};
let textContentStyle: TextStyleProps;
if (elType === 'text') {
textContentStyle = srcStyle;
}
else {
textContentStyle = {};
hasOwn(srcStyle, 'text') && (textContentStyle.text = srcStyle.text);
hasOwn(srcStyle, 'rich') && (textContentStyle.rich = srcStyle.rich);
hasOwn(srcStyle, 'textFill') && (textContentStyle.fill = srcStyle.textFill);
hasOwn(srcStyle, 'textStroke') && (textContentStyle.stroke = srcStyle.textStroke);
textContent = {
type: 'text',
style: textContentStyle,
// ec4 do not support rectText trigger.
// And when text postion is different in normal and emphasis
// => hover text trigger emphasis;
// => text position changed, leave mouse pointer immediately;
// That might cause state incorrect.
silent: true
};
textConfig = {};
const hasOwnPos = hasOwn(srcStyle, 'textPosition');
if (isNormal) {
textConfig.position = hasOwnPos ? srcStyle.textPosition : 'inside';
}
else {
hasOwnPos && (textConfig.position = srcStyle.textPosition);
}
hasOwn(srcStyle, 'textPosition') && (textConfig.position = srcStyle.textPosition);
hasOwn(srcStyle, 'textOffset') && (textConfig.offset = srcStyle.textOffset);
hasOwn(srcStyle, 'textRotation') && (textConfig.rotation = srcStyle.textRotation);
hasOwn(srcStyle, 'textDistance') && (textConfig.distance = srcStyle.textDistance);
}
convertEC4CompatibleRichItem(textContentStyle, hostStyle);
each(textContentStyle.rich, function (richItem) {
convertEC4CompatibleRichItem(richItem as TextStyleProps, richItem);
});
return {
textConfig: textConfig,
textContent: textContent
};
}
/**
* The result will be set to `out`.
*/
function convertEC4CompatibleRichItem(out: TextStylePropsPart, richItem: Dictionary<any>): void {
if (!richItem) {
return;
}
// (1) For simplicity, make textXXX properties (deprecated since ec5) has
// higher priority. For example, consider in ec4 `borderColor: 5, textBorderColor: 10`
// on a rect means `borderColor: 4` on the rect and `borderColor: 10` on an attached
// richText in ec5.
// (2) `out === richItem` if and only if `out` is text el or rich item.
// So we can overwite existing props in `out` since textXXX has higher priority.
richItem.font = richItem.textFont || richItem.font;
hasOwn(richItem, 'textStrokeWidth') && (out.lineWidth = richItem.textStrokeWidth);
hasOwn(richItem, 'textAlign') && (out.align = richItem.textAlign);
hasOwn(richItem, 'textVerticalAlign') && (out.verticalAlign = richItem.textVerticalAlign);
hasOwn(richItem, 'textLineHeight') && (out.lineHeight = richItem.textLineHeight);
hasOwn(richItem, 'textWidth') && (out.width = richItem.textWidth);
hasOwn(richItem, 'textHeight') && (out.height = richItem.textHeight);
hasOwn(richItem, 'textBackgroundColor') && (out.backgroundColor = richItem.textBackgroundColor);
hasOwn(richItem, 'textPadding') && (out.padding = richItem.textPadding);
hasOwn(richItem, 'textBorderColor') && (out.borderColor = richItem.textBorderColor);
hasOwn(richItem, 'textBorderWidth') && (out.borderWidth = richItem.textBorderWidth);
hasOwn(richItem, 'textBorderRadius') && (out.borderRadius = richItem.textBorderRadius);
hasOwn(richItem, 'textBoxShadowColor') && (out.shadowColor = richItem.textBoxShadowColor);
hasOwn(richItem, 'textBoxShadowBlur') && (out.shadowBlur = richItem.textBoxShadowBlur);
hasOwn(richItem, 'textBoxShadowOffsetX') && (out.shadowOffsetX = richItem.textBoxShadowOffsetX);
hasOwn(richItem, 'textBoxShadowOffsetY') && (out.shadowOffsetY = richItem.textBoxShadowOffsetY);
}
/**
* Convert to pure echarts4 format style.
* `itemStyle` will be modified, added with ec4 style properties from
* `textStyle` and `textConfig`.
*
* [Caveat]: For simplicity, `insideRollback` in ec4 does not compat, where
* `styleEmphasis: {textFill: 'red'}` will remove the normal auto added stroke.
*/
export function convertToEC4StyleForCustomSerise(
itemStl: ItemStyleProps,
txStl: TextStyleProps,
txCfg: ElementTextConfig
): ZRStyleProps {
const out = itemStl as Dictionary<unknown>;
// See `custom.ts`, a trick to set extra `textPosition` firstly.
out.textPosition = out.textPosition || txCfg.position || 'inside';
txCfg.offset != null && (out.textOffset = txCfg.offset);
txCfg.rotation != null && (out.textRotation = txCfg.rotation);
txCfg.distance != null && (out.textDistance = txCfg.distance);
const isInside = (out.textPosition as string).indexOf('inside') >= 0;
const hostFill = itemStl.fill || '#000';
convertToEC4RichItem(out, txStl);
const textFillNotSet = out.textFill == null;
if (isInside) {
if (textFillNotSet) {
out.textFill = txCfg.insideFill || '#fff';
!out.textStroke && txCfg.insideStroke && (out.textStroke = txCfg.insideStroke);
!out.textStroke && (out.textStroke = hostFill);
out.textStrokeWidth == null && (out.textStrokeWidth = 2);
}
}
else {
if (textFillNotSet) {
out.textFill = txCfg.outsideFill || hostFill;
}
!out.textStroke && txCfg.outsideStroke && (out.textStroke = txCfg.outsideStroke);
}
out.text = txStl.text;
out.rich = txStl.rich;
each(txStl.rich, function (richItem) {
convertToEC4RichItem(richItem as Dictionary<unknown>, richItem);
});
return out;
}
function convertToEC4RichItem(out: Dictionary<unknown>, richItem: TextStylePropsPart) {
if (!richItem) {
return;
}
hasOwn(richItem, 'fill') && (out.textFill = richItem.fill);
hasOwn(richItem, 'stroke') && (out.textStroke = richItem.fill);
hasOwn(richItem, 'lineWidth') && (out.textStrokeWidth = richItem.lineWidth);
hasOwn(richItem, 'font') && (out.textStrokeWidth = richItem.font);
hasOwn(richItem, 'fontStyle') && (out.fontStyle = richItem.fontStyle);
hasOwn(richItem, 'fontWeight') && (out.fontWeight = richItem.fontWeight);
hasOwn(richItem, 'fontSize') && (out.fontSize = richItem.fontSize);
hasOwn(richItem, 'fontFamily') && (out.fontFamily = richItem.fontFamily);
hasOwn(richItem, 'align') && (out.textAlign = richItem.align);
hasOwn(richItem, 'verticalAlign') && (out.textVerticalAlign = richItem.verticalAlign);
hasOwn(richItem, 'lineHeight') && (out.textLineHeight = richItem.lineHeight);
hasOwn(richItem, 'width') && (out.textWidth = richItem.width);
hasOwn(richItem, 'height') && (out.textHeight = richItem.height);
hasOwn(richItem, 'backgroundColor') && (out.textBackgroundColor = richItem.backgroundColor);
hasOwn(richItem, 'padding') && (out.textPadding = richItem.padding);
hasOwn(richItem, 'borderColor') && (out.textBorderColor = richItem.borderColor);
hasOwn(richItem, 'borderWidth') && (out.textBorderWidth = richItem.borderWidth);
hasOwn(richItem, 'borderRadius') && (out.textBorderRadius = richItem.borderRadius);
hasOwn(richItem, 'shadowColor') && (out.textBoxShadowColor = richItem.shadowColor);
hasOwn(richItem, 'shadowBlur') && (out.textBoxShadowBlur = richItem.shadowBlur);
hasOwn(richItem, 'shadowOffsetX') && (out.textBoxShadowOffsetX = richItem.shadowOffsetX);
hasOwn(richItem, 'shadowOffsetY') && (out.textBoxShadowOffsetY = richItem.shadowOffsetY);
hasOwn(richItem, 'textShadowColor') && (out.textShadowColor = richItem.textShadowColor);
hasOwn(richItem, 'textShadowBlur') && (out.textShadowBlur = richItem.textShadowBlur);
hasOwn(richItem, 'textShadowOffsetX') && (out.textShadowOffsetX = richItem.textShadowOffsetX);
hasOwn(richItem, 'textShadowOffsetY') && (out.textShadowOffsetY = richItem.textShadowOffsetY);
}
export function warnDeprecated(deprecated: string, insteadApproach: string): void {
if (__DEV__) {
const key = deprecated + '^_^' + insteadApproach;
if (!deprecatedLogs[key]) {
console.warn(`DEPRECATED: "${deprecated}" has been deprecated. ${insteadApproach}`);
deprecatedLogs[key] = true;
}
}
}
......@@ -106,6 +106,7 @@ export interface ECElement extends Element {
};
highDownSilentOnTouch?: boolean;
onStateChange?: (fromState: 'normal' | 'emphasis', toState: 'normal' | 'emphasis') => void;
z2EmphasisLift?: number;
}
export interface DataHost {
......@@ -426,6 +427,7 @@ export type ModelOption = any;
export type ThemeOption = Dictionary<any>;
export type DisplayState = 'normal' | 'emphasis';
export type DisplayStateNonNormal = 'emphasis';
export type DisplayStateHostOption = {
emphasis?: Dictionary<any>,
[key: string]: any
......
<!DOCTYPE html>
<!--
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.
-->
<meta charset="utf-8">
<style>
text {
font: 10px sans-serif;
text-anchor: middle;
}
.node--hover circle {
stroke: #000;
stroke-width: 1.2px;
}
#main {
width: 1000px;
height: 500px;
}
</style>
<div id="main" style="width:960px; height:960px;"></div>
<svg width="960" height="960"><g transform="translate(1,1)"></g></svg>
<script src="https://d3js.org/d3.v4.min.js"></script>
<script src="../dist/echarts.js"></script>
<script>
var stratify = d3.stratify()
.parentId(function(d) {
return d.id.substring(0, d.id.lastIndexOf("."));
});
d3.csv("data/flare.csv", function(error, rawData) {
if (error) throw error;
var root = stratify(rawData)
.sum(function(d) {
return d.value;
})
.sort(function(a, b) {
return b.value - a.value;
});
var maxDepth = 0;
var seriesData = root.descendants().map(function (node) {
maxDepth = Math.max(maxDepth, node.depth);
return [
node.value,
node.depth,
node.id
];
});
var chart = echarts.init(document.getElementById('main'));
function renderItem(params, api) {
var context = params.context;
if (!context.layout) {
d3.pack()
.size([api.getWidth() - 2, api.getHeight() - 2])
.padding(3)(root);
context.layout = {};
root.descendants().forEach(function (node) {
context.layout[node.id] = {
x: node.x,
y: node.y,
r: node.r,
isLeaf: !node.children || !node.children.length
};
});
}
var nodePath = api.value(2);
var itemLayout = context.layout[nodePath];
var nodeName = '';
var textFont = api.font({
fontSize: 12,
fontFamily: 'Arial'
});
if (itemLayout.isLeaf && itemLayout.r > 10) {
nodeName = nodePath.slice(nodePath.lastIndexOf('.') + 1).split(/(?=[A-Z][^A-Z])/g).join('\n');
nodeName = echarts.format.truncateText(nodeName, itemLayout.r, textFont, '.');
}
return {
type: 'circle',
shape: {
cx: itemLayout.x,
cy: itemLayout.y,
r: itemLayout.r
},
z2: api.value(1) * 2,
style: api.style({
text: nodeName,
textFont: textFont,
textPosition: 'inside'
}),
styleEmphasis: api.style({
text: nodeName,
textPosition: 'inside',
textFont: textFont,
stroke: 'rgba(0,0,0,0.5)',
lineWidth: 3
})
};
}
var option = {
xAxis: {
axisLine: {show: false},
axisTick: {show: false},
axisLabel: {show: false},
splitLine: {show: false}
},
yAxis: {
axisLine: {show: false},
axisTick: {show: false},
axisLabel: {show: false},
splitLine: {show: false}
},
tooltip: {},
visualMap: {
show: false,
min: 0,
max: maxDepth,
dimension: 1,
inRange: {
color: ['#006edd', '#e0ffff']
}
},
series: {
type: 'custom',
renderItem: renderItem,
encode: {
tooltip: 0,
itemName: 2
},
data: seriesData
}
};
chart.setOption(option);
});
</script>
......@@ -45,6 +45,7 @@ text {
<svg width="960" height="960"><g transform="translate(1,1)"></g></svg>
<script src="https://d3js.org/d3.v4.min.js"></script>
<script src="../dist/echarts.js"></script>
<script src="./lib/testHelper.js"></script>
<script>
var stratify = d3.stratify()
......@@ -105,6 +106,8 @@ d3.csv("data/flare.csv", function(error, rawData) {
nodeName = nodePath.slice(nodePath.lastIndexOf('.') + 1).split(/(?=[A-Z][^A-Z])/g).join('\n');
nodeName = echarts.format.truncateText(nodeName, itemLayout.r, textFont, '.');
}
var z2 = api.value(1) * 2;
// console.log(api.style());
return {
type: 'circle',
......@@ -113,19 +116,35 @@ d3.csv("data/flare.csv", function(error, rawData) {
cy: itemLayout.y,
r: itemLayout.r
},
z2: api.value(1) * 2,
style: api.style({
text: nodeName,
textFont: textFont,
textPosition: 'inside'
}),
styleEmphasis: api.style({
z2: z2,
textContent: {
type: 'text',
style: {
text: nodeName,
// fill: 'blue'
},
emphasis: {
style: {
fontSize: 16,
// fill: 'red'
}
}
},
textConfig: {
position: 'inside'
},
style: {
fill: api.visual('color'),
text: nodeName,
textPosition: 'inside',
textFont: textFont,
stroke: 'rgba(0,0,0,0.5)',
lineWidth: 3
})
font: textFont,
},
emphasis: {
style: {
font: textFont,
stroke: 'rgba(0,0,0,0.5)',
lineWidth: 3
}
}
};
}
......@@ -165,6 +184,13 @@ d3.csv("data/flare.csv", function(error, rawData) {
chart.setOption(option);
// testHelper.printElements(chart, {
// attr: ['z', 'z2', 'style.text', 'style.fill', 'style.stroke'],
// filter: function (el) {
// return el.style && el.style.text;
// }
// });
});
</script>
......@@ -35,10 +35,10 @@ under the License.
</style>
<div id="main0"></div>
<div id="main2"></div>
<div id="main3"></div>
<!-- <div id="main1"></div> -->
<div id="main-eventful"></div>
<div id="main-clip-by-system"></div>
<div id="main-clip-by-self"></div>
<div id="main-SVG-Path"></div>
<script>
......@@ -85,6 +85,7 @@ under the License.
style: {
fill: 'red',
text: 'dataIndex: ' + params.dataIndex,
textFill: '#000',
textStroke: '#fff',
textStrokeWidth: 1
}
......@@ -107,7 +108,7 @@ under the License.
}]
};
var chart = testHelper.create(echarts, 'main0', {
var chart = testHelper.create(echarts, 'main-eventful', {
title: [
'Eventful: ',
'Only this el trigger events: **red circle** and **red rect** of **dataIndex: 1**',
......@@ -138,8 +139,6 @@ under the License.
'echarts'/*, 'map/js/china' */
], function (echarts) {
// deprecated: this case would be wrong.
var option = {
xAxis: {
min: 90,
......@@ -151,13 +150,11 @@ under the License.
max: 500,
scale: true
},
dataZoom: [{
type: 'inside',
filterMode: 'none'
}, {
type: 'slider',
filterMode: 'none'
}],
dataZoom: [
{type: 'inside', filterMode: 'none'},
{type: 'slider', filterMode: 'none'},
{type: 'slider', filterMode: 'none', orient: 'vertical'},
],
series: [{
type: 'custom',
renderItem: function (params, api) {
......@@ -173,24 +170,19 @@ under the License.
[90, 50]
]
},
clip: {
x: params.coordSys.x,
y: params.coordSys.y,
width: params.coordSys.width,
height: params.coordSys.height
},
style: {
fill: 'green'
}
}
},
clip: true,
data: [[100, 300]]
}]
};
var chart = testHelper.create(echarts, 'main2', {
var chart = testHelper.create(echarts, 'main-clip-by-system', {
title: [
'The shape should be **clipped** by the grid. (TODO)',
'The shape should be **clipped** by the grid (by series.clip).',
],
option: option
});
......@@ -204,13 +196,82 @@ under the License.
<script>
require([
'echarts'/*, 'map/js/china' */
], function (echarts) {
var option = {
xAxis: {
min: 90,
max: 120,
scale: true
},
yAxis: {
min: 50,
max: 500,
scale: true
},
dataZoom: [
{type: 'inside', filterMode: 'none'},
{type: 'slider', filterMode: 'none'},
{type: 'slider', filterMode: 'none', orient: 'vertical'},
],
series: [{
type: 'custom',
renderItem: function (params, api) {
return {
type: 'group',
children: [{
type: 'polygon',
position: api.coord([api.value(0), api.value(1)]),
shape: {
points: [
[0, 0],
[50, -50],
[90, -50],
[140, 0],
[90, 50]
]
},
style: {
fill: 'blue'
}
}],
clipPath: {
type: 'rect',
shape: {
x: params.coordSys.x,
y: params.coordSys.y,
width: params.coordSys.width,
height: params.coordSys.height
}
}
}
},
data: [[100, 300]]
}]
};
var chart = testHelper.create(echarts, 'main-clip-by-self', {
title: [
'The shape should be **clipped** by the grid (by custom clipPath).',
],
option: option
});
});
</script>
<script>
require(['echarts'], function (echarts) {
// deprecated: this case would be wrong.
var data = [];
......@@ -224,23 +285,6 @@ under the License.
tooltip: {
},
dataZoom: [{
type: 'slider',
xAxisIndex: 0,
filterMode: 'weakFilter',
height: 20,
bottom: 0,
start: -26,
end: 26,
showDetail: false
}, {
type: 'inside',
xAxisIndex: 0,
filterMode: 'weakFilter',
start: -26,
end: 26,
zoomOnMouseWheel: false,
moveOnMouseMove: true
}, {
type: 'slider',
yAxisIndex: 0,
zoomLock: true,
......@@ -350,7 +394,7 @@ under the License.
};
}
var chart = testHelper.create(echarts, 'main3', {
var chart = testHelper.create(echarts, 'main-SVG-Path', {
title: [
'Test SVG path data auto fit to rect: ',
'the y axis label (made by custom series) should be center',
......@@ -365,5 +409,9 @@ under the License.
</body>
</html>
\ No newline at end of file
此差异已折叠。
window.BAR_ROUND_GRADIENT_TEXTURE = '';
\ No newline at end of file
此差异已折叠。
此差异已折叠。
......@@ -680,7 +680,8 @@ under the License.
// silent: true,
label: {
show: true,
silent: true,
// silent: true,
position: 'top'
},
itemStyle: {
color: 'green',
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册