123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484 |
- /*
- * 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.
- */
- var _config = require("../../config");
- var __DEV__ = _config.__DEV__;
- var echarts = require("../../echarts");
- var zrUtil = require("zrender/lib/core/util");
- var _symbol = require("../../util/symbol");
- var createSymbol = _symbol.createSymbol;
- var graphic = require("../../util/graphic");
- var _listComponent = require("../helper/listComponent");
- var makeBackground = _listComponent.makeBackground;
- var layoutUtil = require("../../util/layout");
- /*
- * 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.
- */
- var curry = zrUtil.curry;
- var each = zrUtil.each;
- var Group = graphic.Group;
- var _default = echarts.extendComponentView({
- type: 'legend.plain',
- newlineDisabled: false,
- /**
- * @override
- */
- init: function () {
- /**
- * @private
- * @type {module:zrender/container/Group}
- */
- this.group.add(this._contentGroup = new Group());
- /**
- * @private
- * @type {module:zrender/Element}
- */
- this._backgroundEl;
- /**
- * @private
- * @type {module:zrender/container/Group}
- */
- this.group.add(this._selectorGroup = new Group());
- /**
- * If first rendering, `contentGroup.position` is [0, 0], which
- * does not make sense and may cause unexepcted animation if adopted.
- * @private
- * @type {boolean}
- */
- this._isFirstRender = true;
- },
- /**
- * @protected
- */
- getContentGroup: function () {
- return this._contentGroup;
- },
- /**
- * @protected
- */
- getSelectorGroup: function () {
- return this._selectorGroup;
- },
- /**
- * @override
- */
- render: function (legendModel, ecModel, api) {
- var isFirstRender = this._isFirstRender;
- this._isFirstRender = false;
- this.resetInner();
- if (!legendModel.get('show', true)) {
- return;
- }
- var itemAlign = legendModel.get('align');
- var orient = legendModel.get('orient');
- if (!itemAlign || itemAlign === 'auto') {
- itemAlign = legendModel.get('left') === 'right' && orient === 'vertical' ? 'right' : 'left';
- }
- var selector = legendModel.get('selector', true);
- var selectorPosition = legendModel.get('selectorPosition', true);
- if (selector && (!selectorPosition || selectorPosition === 'auto')) {
- selectorPosition = orient === 'horizontal' ? 'end' : 'start';
- }
- this.renderInner(itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition); // Perform layout.
- var positionInfo = legendModel.getBoxLayoutParams();
- var viewportSize = {
- width: api.getWidth(),
- height: api.getHeight()
- };
- var padding = legendModel.get('padding');
- var maxSize = layoutUtil.getLayoutRect(positionInfo, viewportSize, padding);
- var mainRect = this.layoutInner(legendModel, itemAlign, maxSize, isFirstRender, selector, selectorPosition); // Place mainGroup, based on the calculated `mainRect`.
- var layoutRect = layoutUtil.getLayoutRect(zrUtil.defaults({
- width: mainRect.width,
- height: mainRect.height
- }, positionInfo), viewportSize, padding);
- this.group.attr('position', [layoutRect.x - mainRect.x, layoutRect.y - mainRect.y]); // Render background after group is layout.
- this.group.add(this._backgroundEl = makeBackground(mainRect, legendModel));
- },
- /**
- * @protected
- */
- resetInner: function () {
- this.getContentGroup().removeAll();
- this._backgroundEl && this.group.remove(this._backgroundEl);
- this.getSelectorGroup().removeAll();
- },
- /**
- * @protected
- */
- renderInner: function (itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition) {
- var contentGroup = this.getContentGroup();
- var legendDrawnMap = zrUtil.createHashMap();
- var selectMode = legendModel.get('selectedMode');
- var excludeSeriesId = [];
- ecModel.eachRawSeries(function (seriesModel) {
- !seriesModel.get('legendHoverLink') && excludeSeriesId.push(seriesModel.id);
- });
- each(legendModel.getData(), function (itemModel, dataIndex) {
- var name = itemModel.get('name'); // Use empty string or \n as a newline string
- if (!this.newlineDisabled && (name === '' || name === '\n')) {
- contentGroup.add(new Group({
- newline: true
- }));
- return;
- } // Representitive series.
- var seriesModel = ecModel.getSeriesByName(name)[0];
- if (legendDrawnMap.get(name)) {
- // Have been drawed
- return;
- } // Legend to control series.
- if (seriesModel) {
- var data = seriesModel.getData();
- var color = data.getVisual('color');
- var borderColor = data.getVisual('borderColor'); // If color is a callback function
- if (typeof color === 'function') {
- // Use the first data
- color = color(seriesModel.getDataParams(0));
- } // If borderColor is a callback function
- if (typeof borderColor === 'function') {
- // Use the first data
- borderColor = borderColor(seriesModel.getDataParams(0));
- } // Using rect symbol defaultly
- var legendSymbolType = data.getVisual('legendSymbol') || 'roundRect';
- var symbolType = data.getVisual('symbol');
- var itemGroup = this._createItem(name, dataIndex, itemModel, legendModel, legendSymbolType, symbolType, itemAlign, color, borderColor, selectMode);
- itemGroup.on('click', curry(dispatchSelectAction, name, null, api, excludeSeriesId)).on('mouseover', curry(dispatchHighlightAction, seriesModel.name, null, api, excludeSeriesId)).on('mouseout', curry(dispatchDownplayAction, seriesModel.name, null, api, excludeSeriesId));
- legendDrawnMap.set(name, true);
- } else {
- // Legend to control data. In pie and funnel.
- ecModel.eachRawSeries(function (seriesModel) {
- // In case multiple series has same data name
- if (legendDrawnMap.get(name)) {
- return;
- }
- if (seriesModel.legendVisualProvider) {
- var provider = seriesModel.legendVisualProvider;
- if (!provider.containName(name)) {
- return;
- }
- var idx = provider.indexOfName(name);
- var color = provider.getItemVisual(idx, 'color');
- var borderColor = provider.getItemVisual(idx, 'borderColor');
- var legendSymbolType = 'roundRect';
- var itemGroup = this._createItem(name, dataIndex, itemModel, legendModel, legendSymbolType, null, itemAlign, color, borderColor, selectMode); // FIXME: consider different series has items with the same name.
- itemGroup.on('click', curry(dispatchSelectAction, null, name, api, excludeSeriesId)) // Should not specify the series name, consider legend controls
- // more than one pie series.
- .on('mouseover', curry(dispatchHighlightAction, null, name, api, excludeSeriesId)).on('mouseout', curry(dispatchDownplayAction, null, name, api, excludeSeriesId));
- legendDrawnMap.set(name, true);
- }
- }, this);
- }
- }, this);
- if (selector) {
- this._createSelector(selector, legendModel, api, orient, selectorPosition);
- }
- },
- _createSelector: function (selector, legendModel, api, orient, selectorPosition) {
- var selectorGroup = this.getSelectorGroup();
- each(selector, function (selectorItem) {
- createSelectorButton(selectorItem);
- });
- function createSelectorButton(selectorItem) {
- var type = selectorItem.type;
- var labelText = new graphic.Text({
- style: {
- x: 0,
- y: 0,
- align: 'center',
- verticalAlign: 'middle'
- },
- onclick: function () {
- api.dispatchAction({
- type: type === 'all' ? 'legendAllSelect' : 'legendInverseSelect'
- });
- }
- });
- selectorGroup.add(labelText);
- var labelModel = legendModel.getModel('selectorLabel');
- var emphasisLabelModel = legendModel.getModel('emphasis.selectorLabel');
- graphic.setLabelStyle(labelText.style, labelText.hoverStyle = {}, labelModel, emphasisLabelModel, {
- defaultText: selectorItem.title,
- isRectText: false
- });
- graphic.setHoverStyle(labelText);
- }
- },
- _createItem: function (name, dataIndex, itemModel, legendModel, legendSymbolType, symbolType, itemAlign, color, borderColor, selectMode) {
- var itemWidth = legendModel.get('itemWidth');
- var itemHeight = legendModel.get('itemHeight');
- var inactiveColor = legendModel.get('inactiveColor');
- var inactiveBorderColor = legendModel.get('inactiveBorderColor');
- var symbolKeepAspect = legendModel.get('symbolKeepAspect');
- var legendModelItemStyle = legendModel.getModel('itemStyle');
- var isSelected = legendModel.isSelected(name);
- var itemGroup = new Group();
- var textStyleModel = itemModel.getModel('textStyle');
- var itemIcon = itemModel.get('icon');
- var tooltipModel = itemModel.getModel('tooltip');
- var legendGlobalTooltipModel = tooltipModel.parentModel; // Use user given icon first
- legendSymbolType = itemIcon || legendSymbolType;
- var legendSymbol = createSymbol(legendSymbolType, 0, 0, itemWidth, itemHeight, isSelected ? color : inactiveColor, // symbolKeepAspect default true for legend
- symbolKeepAspect == null ? true : symbolKeepAspect);
- itemGroup.add(setSymbolStyle(legendSymbol, legendSymbolType, legendModelItemStyle, borderColor, inactiveBorderColor, isSelected)); // Compose symbols
- // PENDING
- if (!itemIcon && symbolType // At least show one symbol, can't be all none
- && (symbolType !== legendSymbolType || symbolType === 'none')) {
- var size = itemHeight * 0.8;
- if (symbolType === 'none') {
- symbolType = 'circle';
- }
- var legendSymbolCenter = createSymbol(symbolType, (itemWidth - size) / 2, (itemHeight - size) / 2, size, size, isSelected ? color : inactiveColor, // symbolKeepAspect default true for legend
- symbolKeepAspect == null ? true : symbolKeepAspect); // Put symbol in the center
- itemGroup.add(setSymbolStyle(legendSymbolCenter, symbolType, legendModelItemStyle, borderColor, inactiveBorderColor, isSelected));
- }
- var textX = itemAlign === 'left' ? itemWidth + 5 : -5;
- var textAlign = itemAlign;
- var formatter = legendModel.get('formatter');
- var content = name;
- if (typeof formatter === 'string' && formatter) {
- content = formatter.replace('{name}', name != null ? name : '');
- } else if (typeof formatter === 'function') {
- content = formatter(name);
- }
- itemGroup.add(new graphic.Text({
- style: graphic.setTextStyle({}, textStyleModel, {
- text: content,
- x: textX,
- y: itemHeight / 2,
- textFill: isSelected ? textStyleModel.getTextColor() : inactiveColor,
- textAlign: textAlign,
- textVerticalAlign: 'middle'
- })
- })); // Add a invisible rect to increase the area of mouse hover
- var hitRect = new graphic.Rect({
- shape: itemGroup.getBoundingRect(),
- invisible: true,
- tooltip: tooltipModel.get('show') ? zrUtil.extend({
- content: name,
- // Defaul formatter
- formatter: legendGlobalTooltipModel.get('formatter', true) || function () {
- return name;
- },
- formatterParams: {
- componentType: 'legend',
- legendIndex: legendModel.componentIndex,
- name: name,
- $vars: ['name']
- }
- }, tooltipModel.option) : null
- });
- itemGroup.add(hitRect);
- itemGroup.eachChild(function (child) {
- child.silent = true;
- });
- hitRect.silent = !selectMode;
- this.getContentGroup().add(itemGroup);
- graphic.setHoverStyle(itemGroup);
- itemGroup.__legendDataIndex = dataIndex;
- return itemGroup;
- },
- /**
- * @protected
- */
- layoutInner: function (legendModel, itemAlign, maxSize, isFirstRender, selector, selectorPosition) {
- var contentGroup = this.getContentGroup();
- var selectorGroup = this.getSelectorGroup(); // Place items in contentGroup.
- layoutUtil.box(legendModel.get('orient'), contentGroup, legendModel.get('itemGap'), maxSize.width, maxSize.height);
- var contentRect = contentGroup.getBoundingRect();
- var contentPos = [-contentRect.x, -contentRect.y];
- if (selector) {
- // Place buttons in selectorGroup
- layoutUtil.box( // Buttons in selectorGroup always layout horizontally
- 'horizontal', selectorGroup, legendModel.get('selectorItemGap', true));
- var selectorRect = selectorGroup.getBoundingRect();
- var selectorPos = [-selectorRect.x, -selectorRect.y];
- var selectorButtonGap = legendModel.get('selectorButtonGap', true);
- var orientIdx = legendModel.getOrient().index;
- var wh = orientIdx === 0 ? 'width' : 'height';
- var hw = orientIdx === 0 ? 'height' : 'width';
- var yx = orientIdx === 0 ? 'y' : 'x';
- if (selectorPosition === 'end') {
- selectorPos[orientIdx] += contentRect[wh] + selectorButtonGap;
- } else {
- contentPos[orientIdx] += selectorRect[wh] + selectorButtonGap;
- } //Always align selector to content as 'middle'
- selectorPos[1 - orientIdx] += contentRect[hw] / 2 - selectorRect[hw] / 2;
- selectorGroup.attr('position', selectorPos);
- contentGroup.attr('position', contentPos);
- var mainRect = {
- x: 0,
- y: 0
- };
- mainRect[wh] = contentRect[wh] + selectorButtonGap + selectorRect[wh];
- mainRect[hw] = Math.max(contentRect[hw], selectorRect[hw]);
- mainRect[yx] = Math.min(0, selectorRect[yx] + selectorPos[1 - orientIdx]);
- return mainRect;
- } else {
- contentGroup.attr('position', contentPos);
- return this.group.getBoundingRect();
- }
- },
- /**
- * @protected
- */
- remove: function () {
- this.getContentGroup().removeAll();
- this._isFirstRender = true;
- }
- });
- function setSymbolStyle(symbol, symbolType, legendModelItemStyle, borderColor, inactiveBorderColor, isSelected) {
- var itemStyle;
- if (symbolType !== 'line' && symbolType.indexOf('empty') < 0) {
- itemStyle = legendModelItemStyle.getItemStyle();
- symbol.style.stroke = borderColor;
- if (!isSelected) {
- itemStyle.stroke = inactiveBorderColor;
- }
- } else {
- itemStyle = legendModelItemStyle.getItemStyle(['borderWidth', 'borderColor']);
- }
- return symbol.setStyle(itemStyle);
- }
- function dispatchSelectAction(seriesName, dataName, api, excludeSeriesId) {
- // downplay before unselect
- dispatchDownplayAction(seriesName, dataName, api, excludeSeriesId);
- api.dispatchAction({
- type: 'legendToggleSelect',
- name: seriesName != null ? seriesName : dataName
- }); // highlight after select
- dispatchHighlightAction(seriesName, dataName, api, excludeSeriesId);
- }
- function dispatchHighlightAction(seriesName, dataName, api, excludeSeriesId) {
- // If element hover will move to a hoverLayer.
- var el = api.getZr().storage.getDisplayList()[0];
- if (!(el && el.useHoverLayer)) {
- api.dispatchAction({
- type: 'highlight',
- seriesName: seriesName,
- name: dataName,
- excludeSeriesId: excludeSeriesId
- });
- }
- }
- function dispatchDownplayAction(seriesName, dataName, api, excludeSeriesId) {
- // If element hover will move to a hoverLayer.
- var el = api.getZr().storage.getDisplayList()[0];
- if (!(el && el.useHoverLayer)) {
- api.dispatchAction({
- type: 'downplay',
- seriesName: seriesName,
- name: dataName,
- excludeSeriesId: excludeSeriesId
- });
- }
- }
- module.exports = _default;
|