123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447 |
- /*
- * 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 zrUtil = require("zrender/lib/core/util");
- var graphic = require("../../util/graphic");
- var layoutUtil = require("../../util/layout");
- var LegendView = require("./LegendView");
- /*
- * 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.
- */
- /**
- * Separate legend and scrollable legend to reduce package size.
- */
- var Group = graphic.Group;
- var WH = ['width', 'height'];
- var XY = ['x', 'y'];
- var ScrollableLegendView = LegendView.extend({
- type: 'legend.scroll',
- newlineDisabled: true,
- init: function () {
- ScrollableLegendView.superCall(this, 'init');
- /**
- * @private
- * @type {number} For `scroll`.
- */
- this._currentIndex = 0;
- /**
- * @private
- * @type {module:zrender/container/Group}
- */
- this.group.add(this._containerGroup = new Group());
- this._containerGroup.add(this.getContentGroup());
- /**
- * @private
- * @type {module:zrender/container/Group}
- */
- this.group.add(this._controllerGroup = new Group());
- /**
- *
- * @private
- */
- this._showController;
- },
- /**
- * @override
- */
- resetInner: function () {
- ScrollableLegendView.superCall(this, 'resetInner');
- this._controllerGroup.removeAll();
- this._containerGroup.removeClipPath();
- this._containerGroup.__rectSize = null;
- },
- /**
- * @override
- */
- renderInner: function (itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition) {
- var me = this; // Render content items.
- ScrollableLegendView.superCall(this, 'renderInner', itemAlign, legendModel, ecModel, api, selector, orient, selectorPosition);
- var controllerGroup = this._controllerGroup; // FIXME: support be 'auto' adapt to size number text length,
- // e.g., '3/12345' should not overlap with the control arrow button.
- var pageIconSize = legendModel.get('pageIconSize', true);
- if (!zrUtil.isArray(pageIconSize)) {
- pageIconSize = [pageIconSize, pageIconSize];
- }
- createPageButton('pagePrev', 0);
- var pageTextStyleModel = legendModel.getModel('pageTextStyle');
- controllerGroup.add(new graphic.Text({
- name: 'pageText',
- style: {
- textFill: pageTextStyleModel.getTextColor(),
- font: pageTextStyleModel.getFont(),
- textVerticalAlign: 'middle',
- textAlign: 'center'
- },
- silent: true
- }));
- createPageButton('pageNext', 1);
- function createPageButton(name, iconIdx) {
- var pageDataIndexName = name + 'DataIndex';
- var icon = graphic.createIcon(legendModel.get('pageIcons', true)[legendModel.getOrient().name][iconIdx], {
- // Buttons will be created in each render, so we do not need
- // to worry about avoiding using legendModel kept in scope.
- onclick: zrUtil.bind(me._pageGo, me, pageDataIndexName, legendModel, api)
- }, {
- x: -pageIconSize[0] / 2,
- y: -pageIconSize[1] / 2,
- width: pageIconSize[0],
- height: pageIconSize[1]
- });
- icon.name = name;
- controllerGroup.add(icon);
- }
- },
- /**
- * @override
- */
- layoutInner: function (legendModel, itemAlign, maxSize, isFirstRender, selector, selectorPosition) {
- var selectorGroup = this.getSelectorGroup();
- var orientIdx = legendModel.getOrient().index;
- var wh = WH[orientIdx];
- var xy = XY[orientIdx];
- var hw = WH[1 - orientIdx];
- var yx = XY[1 - orientIdx];
- selector && layoutUtil.box( // Buttons in selectorGroup always layout horizontally
- 'horizontal', selectorGroup, legendModel.get('selectorItemGap', true));
- var selectorButtonGap = legendModel.get('selectorButtonGap', true);
- var selectorRect = selectorGroup.getBoundingRect();
- var selectorPos = [-selectorRect.x, -selectorRect.y];
- var processMaxSize = zrUtil.clone(maxSize);
- selector && (processMaxSize[wh] = maxSize[wh] - selectorRect[wh] - selectorButtonGap);
- var mainRect = this._layoutContentAndController(legendModel, isFirstRender, processMaxSize, orientIdx, wh, hw, yx);
- if (selector) {
- if (selectorPosition === 'end') {
- selectorPos[orientIdx] += mainRect[wh] + selectorButtonGap;
- } else {
- var offset = selectorRect[wh] + selectorButtonGap;
- selectorPos[orientIdx] -= offset;
- mainRect[xy] -= offset;
- }
- mainRect[wh] += selectorRect[wh] + selectorButtonGap;
- selectorPos[1 - orientIdx] += mainRect[yx] + mainRect[hw] / 2 - selectorRect[hw] / 2;
- mainRect[hw] = Math.max(mainRect[hw], selectorRect[hw]);
- mainRect[yx] = Math.min(mainRect[yx], selectorRect[yx] + selectorPos[1 - orientIdx]);
- selectorGroup.attr('position', selectorPos);
- }
- return mainRect;
- },
- _layoutContentAndController: function (legendModel, isFirstRender, maxSize, orientIdx, wh, hw, yx) {
- var contentGroup = this.getContentGroup();
- var containerGroup = this._containerGroup;
- var controllerGroup = this._controllerGroup; // Place items in contentGroup.
- layoutUtil.box(legendModel.get('orient'), contentGroup, legendModel.get('itemGap'), !orientIdx ? null : maxSize.width, orientIdx ? null : maxSize.height);
- layoutUtil.box( // Buttons in controller are layout always horizontally.
- 'horizontal', controllerGroup, legendModel.get('pageButtonItemGap', true));
- var contentRect = contentGroup.getBoundingRect();
- var controllerRect = controllerGroup.getBoundingRect();
- var showController = this._showController = contentRect[wh] > maxSize[wh];
- var contentPos = [-contentRect.x, -contentRect.y]; // Remain contentPos when scroll animation perfroming.
- // If first rendering, `contentGroup.position` is [0, 0], which
- // does not make sense and may cause unexepcted animation if adopted.
- if (!isFirstRender) {
- contentPos[orientIdx] = contentGroup.position[orientIdx];
- } // Layout container group based on 0.
- var containerPos = [0, 0];
- var controllerPos = [-controllerRect.x, -controllerRect.y];
- var pageButtonGap = zrUtil.retrieve2(legendModel.get('pageButtonGap', true), legendModel.get('itemGap', true)); // Place containerGroup and controllerGroup and contentGroup.
- if (showController) {
- var pageButtonPosition = legendModel.get('pageButtonPosition', true); // controller is on the right / bottom.
- if (pageButtonPosition === 'end') {
- controllerPos[orientIdx] += maxSize[wh] - controllerRect[wh];
- } // controller is on the left / top.
- else {
- containerPos[orientIdx] += controllerRect[wh] + pageButtonGap;
- }
- } // Always align controller to content as 'middle'.
- controllerPos[1 - orientIdx] += contentRect[hw] / 2 - controllerRect[hw] / 2;
- contentGroup.attr('position', contentPos);
- containerGroup.attr('position', containerPos);
- controllerGroup.attr('position', controllerPos); // Calculate `mainRect` and set `clipPath`.
- // mainRect should not be calculated by `this.group.getBoundingRect()`
- // for sake of the overflow.
- var mainRect = {
- x: 0,
- y: 0
- }; // Consider content may be overflow (should be clipped).
- mainRect[wh] = showController ? maxSize[wh] : contentRect[wh];
- mainRect[hw] = Math.max(contentRect[hw], controllerRect[hw]); // `containerRect[yx] + containerPos[1 - orientIdx]` is 0.
- mainRect[yx] = Math.min(0, controllerRect[yx] + controllerPos[1 - orientIdx]);
- containerGroup.__rectSize = maxSize[wh];
- if (showController) {
- var clipShape = {
- x: 0,
- y: 0
- };
- clipShape[wh] = Math.max(maxSize[wh] - controllerRect[wh] - pageButtonGap, 0);
- clipShape[hw] = mainRect[hw];
- containerGroup.setClipPath(new graphic.Rect({
- shape: clipShape
- })); // Consider content may be larger than container, container rect
- // can not be obtained from `containerGroup.getBoundingRect()`.
- containerGroup.__rectSize = clipShape[wh];
- } else {
- // Do not remove or ignore controller. Keep them set as placeholders.
- controllerGroup.eachChild(function (child) {
- child.attr({
- invisible: true,
- silent: true
- });
- });
- } // Content translate animation.
- var pageInfo = this._getPageInfo(legendModel);
- pageInfo.pageIndex != null && graphic.updateProps(contentGroup, {
- position: pageInfo.contentPosition
- }, // When switch from "show controller" to "not show controller", view should be
- // updated immediately without animation, otherwise causes weird effect.
- showController ? legendModel : false);
- this._updatePageInfoView(legendModel, pageInfo);
- return mainRect;
- },
- _pageGo: function (to, legendModel, api) {
- var scrollDataIndex = this._getPageInfo(legendModel)[to];
- scrollDataIndex != null && api.dispatchAction({
- type: 'legendScroll',
- scrollDataIndex: scrollDataIndex,
- legendId: legendModel.id
- });
- },
- _updatePageInfoView: function (legendModel, pageInfo) {
- var controllerGroup = this._controllerGroup;
- zrUtil.each(['pagePrev', 'pageNext'], function (name) {
- var canJump = pageInfo[name + 'DataIndex'] != null;
- var icon = controllerGroup.childOfName(name);
- if (icon) {
- icon.setStyle('fill', canJump ? legendModel.get('pageIconColor', true) : legendModel.get('pageIconInactiveColor', true));
- icon.cursor = canJump ? 'pointer' : 'default';
- }
- });
- var pageText = controllerGroup.childOfName('pageText');
- var pageFormatter = legendModel.get('pageFormatter');
- var pageIndex = pageInfo.pageIndex;
- var current = pageIndex != null ? pageIndex + 1 : 0;
- var total = pageInfo.pageCount;
- pageText && pageFormatter && pageText.setStyle('text', zrUtil.isString(pageFormatter) ? pageFormatter.replace('{current}', current).replace('{total}', total) : pageFormatter({
- current: current,
- total: total
- }));
- },
- /**
- * @param {module:echarts/model/Model} legendModel
- * @return {Object} {
- * contentPosition: Array.<number>, null when data item not found.
- * pageIndex: number, null when data item not found.
- * pageCount: number, always be a number, can be 0.
- * pagePrevDataIndex: number, null when no previous page.
- * pageNextDataIndex: number, null when no next page.
- * }
- */
- _getPageInfo: function (legendModel) {
- var scrollDataIndex = legendModel.get('scrollDataIndex', true);
- var contentGroup = this.getContentGroup();
- var containerRectSize = this._containerGroup.__rectSize;
- var orientIdx = legendModel.getOrient().index;
- var wh = WH[orientIdx];
- var xy = XY[orientIdx];
- var targetItemIndex = this._findTargetItemIndex(scrollDataIndex);
- var children = contentGroup.children();
- var targetItem = children[targetItemIndex];
- var itemCount = children.length;
- var pCount = !itemCount ? 0 : 1;
- var result = {
- contentPosition: contentGroup.position.slice(),
- pageCount: pCount,
- pageIndex: pCount - 1,
- pagePrevDataIndex: null,
- pageNextDataIndex: null
- };
- if (!targetItem) {
- return result;
- }
- var targetItemInfo = getItemInfo(targetItem);
- result.contentPosition[orientIdx] = -targetItemInfo.s; // Strategy:
- // (1) Always align based on the left/top most item.
- // (2) It is user-friendly that the last item shown in the
- // current window is shown at the begining of next window.
- // Otherwise if half of the last item is cut by the window,
- // it will have no chance to display entirely.
- // (3) Consider that item size probably be different, we
- // have calculate pageIndex by size rather than item index,
- // and we can not get page index directly by division.
- // (4) The window is to narrow to contain more than
- // one item, we should make sure that the page can be fliped.
- for (var i = targetItemIndex + 1, winStartItemInfo = targetItemInfo, winEndItemInfo = targetItemInfo, currItemInfo = null; i <= itemCount; ++i) {
- currItemInfo = getItemInfo(children[i]);
- if ( // Half of the last item is out of the window.
- !currItemInfo && winEndItemInfo.e > winStartItemInfo.s + containerRectSize || // If the current item does not intersect with the window, the new page
- // can be started at the current item or the last item.
- currItemInfo && !intersect(currItemInfo, winStartItemInfo.s)) {
- if (winEndItemInfo.i > winStartItemInfo.i) {
- winStartItemInfo = winEndItemInfo;
- } else {
- // e.g., when page size is smaller than item size.
- winStartItemInfo = currItemInfo;
- }
- if (winStartItemInfo) {
- if (result.pageNextDataIndex == null) {
- result.pageNextDataIndex = winStartItemInfo.i;
- }
- ++result.pageCount;
- }
- }
- winEndItemInfo = currItemInfo;
- }
- for (var i = targetItemIndex - 1, winStartItemInfo = targetItemInfo, winEndItemInfo = targetItemInfo, currItemInfo = null; i >= -1; --i) {
- currItemInfo = getItemInfo(children[i]);
- if ( // If the the end item does not intersect with the window started
- // from the current item, a page can be settled.
- (!currItemInfo || !intersect(winEndItemInfo, currItemInfo.s)) && // e.g., when page size is smaller than item size.
- winStartItemInfo.i < winEndItemInfo.i) {
- winEndItemInfo = winStartItemInfo;
- if (result.pagePrevDataIndex == null) {
- result.pagePrevDataIndex = winStartItemInfo.i;
- }
- ++result.pageCount;
- ++result.pageIndex;
- }
- winStartItemInfo = currItemInfo;
- }
- return result;
- function getItemInfo(el) {
- if (el) {
- var itemRect = el.getBoundingRect();
- var start = itemRect[xy] + el.position[orientIdx];
- return {
- s: start,
- e: start + itemRect[wh],
- i: el.__legendDataIndex
- };
- }
- }
- function intersect(itemInfo, winStart) {
- return itemInfo.e >= winStart && itemInfo.s <= winStart + containerRectSize;
- }
- },
- _findTargetItemIndex: function (targetDataIndex) {
- if (!this._showController) {
- return 0;
- }
- var index;
- var contentGroup = this.getContentGroup();
- var defaultIndex;
- contentGroup.eachChild(function (child, idx) {
- var legendDataIdx = child.__legendDataIndex; // FIXME
- // If the given targetDataIndex (from model) is illegal,
- // we use defaultIndex. But the index on the legend model and
- // action payload is still illegal. That case will not be
- // changed until some scenario requires.
- if (defaultIndex == null && legendDataIdx != null) {
- defaultIndex = idx;
- }
- if (legendDataIdx === targetDataIndex) {
- index = idx;
- }
- });
- return index != null ? index : defaultIndex;
- }
- });
- var _default = ScrollableLegendView;
- module.exports = _default;
|