TooltipView.js 23 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716
  1. var echarts = require("../../echarts");
  2. var zrUtil = require("zrender/lib/core/util");
  3. var env = require("zrender/lib/core/env");
  4. var TooltipContent = require("./TooltipContent");
  5. var formatUtil = require("../../util/format");
  6. var numberUtil = require("../../util/number");
  7. var graphic = require("../../util/graphic");
  8. var findPointFromSeries = require("../axisPointer/findPointFromSeries");
  9. var layoutUtil = require("../../util/layout");
  10. var Model = require("../../model/Model");
  11. var globalListener = require("../axisPointer/globalListener");
  12. var axisHelper = require("../../coord/axisHelper");
  13. var axisPointerViewHelper = require("../axisPointer/viewHelper");
  14. var bind = zrUtil.bind;
  15. var each = zrUtil.each;
  16. var parsePercent = numberUtil.parsePercent;
  17. var proxyRect = new graphic.Rect({
  18. shape: {
  19. x: -1,
  20. y: -1,
  21. width: 2,
  22. height: 2
  23. }
  24. });
  25. var _default = echarts.extendComponentView({
  26. type: 'tooltip',
  27. init: function (ecModel, api) {
  28. if (env.node) {
  29. return;
  30. }
  31. var tooltipContent = new TooltipContent(api.getDom(), api);
  32. this._tooltipContent = tooltipContent;
  33. },
  34. render: function (tooltipModel, ecModel, api) {
  35. if (env.node) {
  36. return;
  37. } // Reset
  38. this.group.removeAll();
  39. /**
  40. * @private
  41. * @type {module:echarts/component/tooltip/TooltipModel}
  42. */
  43. this._tooltipModel = tooltipModel;
  44. /**
  45. * @private
  46. * @type {module:echarts/model/Global}
  47. */
  48. this._ecModel = ecModel;
  49. /**
  50. * @private
  51. * @type {module:echarts/ExtensionAPI}
  52. */
  53. this._api = api;
  54. /**
  55. * Should be cleaned when render.
  56. * @private
  57. * @type {Array.<Array.<Object>>}
  58. */
  59. this._lastDataByCoordSys = null;
  60. /**
  61. * @private
  62. * @type {boolean}
  63. */
  64. this._alwaysShowContent = tooltipModel.get('alwaysShowContent');
  65. var tooltipContent = this._tooltipContent;
  66. tooltipContent.update();
  67. tooltipContent.setEnterable(tooltipModel.get('enterable'));
  68. this._initGlobalListener();
  69. this._keepShow();
  70. },
  71. _initGlobalListener: function () {
  72. var tooltipModel = this._tooltipModel;
  73. var triggerOn = tooltipModel.get('triggerOn');
  74. globalListener.register('itemTooltip', this._api, bind(function (currTrigger, e, dispatchAction) {
  75. // If 'none', it is not controlled by mouse totally.
  76. if (triggerOn !== 'none') {
  77. if (triggerOn.indexOf(currTrigger) >= 0) {
  78. this._tryShow(e, dispatchAction);
  79. } else if (currTrigger === 'leave') {
  80. this._hide(dispatchAction);
  81. }
  82. }
  83. }, this));
  84. },
  85. _keepShow: function () {
  86. var tooltipModel = this._tooltipModel;
  87. var ecModel = this._ecModel;
  88. var api = this._api; // Try to keep the tooltip show when refreshing
  89. if (this._lastX != null && this._lastY != null // When user is willing to control tooltip totally using API,
  90. // self.manuallyShowTip({x, y}) might cause tooltip hide,
  91. // which is not expected.
  92. && tooltipModel.get('triggerOn') !== 'none') {
  93. var self = this;
  94. clearTimeout(this._refreshUpdateTimeout);
  95. this._refreshUpdateTimeout = setTimeout(function () {
  96. // Show tip next tick after other charts are rendered
  97. // In case highlight action has wrong result
  98. // FIXME
  99. self.manuallyShowTip(tooltipModel, ecModel, api, {
  100. x: self._lastX,
  101. y: self._lastY
  102. });
  103. });
  104. }
  105. },
  106. /**
  107. * Show tip manually by
  108. * dispatchAction({
  109. * type: 'showTip',
  110. * x: 10,
  111. * y: 10
  112. * });
  113. * Or
  114. * dispatchAction({
  115. * type: 'showTip',
  116. * seriesIndex: 0,
  117. * dataIndex or dataIndexInside or name
  118. * });
  119. *
  120. * TODO Batch
  121. */
  122. manuallyShowTip: function (tooltipModel, ecModel, api, payload) {
  123. if (payload.from === this.uid || env.node) {
  124. return;
  125. }
  126. var dispatchAction = makeDispatchAction(payload, api); // Reset ticket
  127. this._ticket = ''; // When triggered from axisPointer.
  128. var dataByCoordSys = payload.dataByCoordSys;
  129. if (payload.tooltip && payload.x != null && payload.y != null) {
  130. var el = proxyRect;
  131. el.position = [payload.x, payload.y];
  132. el.update();
  133. el.tooltip = payload.tooltip; // Manually show tooltip while view is not using zrender elements.
  134. this._tryShow({
  135. offsetX: payload.x,
  136. offsetY: payload.y,
  137. target: el
  138. }, dispatchAction);
  139. } else if (dataByCoordSys) {
  140. this._tryShow({
  141. offsetX: payload.x,
  142. offsetY: payload.y,
  143. position: payload.position,
  144. event: {},
  145. dataByCoordSys: payload.dataByCoordSys,
  146. tooltipOption: payload.tooltipOption
  147. }, dispatchAction);
  148. } else if (payload.seriesIndex != null) {
  149. if (this._manuallyAxisShowTip(tooltipModel, ecModel, api, payload)) {
  150. return;
  151. }
  152. var pointInfo = findPointFromSeries(payload, ecModel);
  153. var cx = pointInfo.point[0];
  154. var cy = pointInfo.point[1];
  155. if (cx != null && cy != null) {
  156. this._tryShow({
  157. offsetX: cx,
  158. offsetY: cy,
  159. position: payload.position,
  160. target: pointInfo.el,
  161. event: {}
  162. }, dispatchAction);
  163. }
  164. } else if (payload.x != null && payload.y != null) {
  165. // FIXME
  166. // should wrap dispatchAction like `axisPointer/globalListener` ?
  167. api.dispatchAction({
  168. type: 'updateAxisPointer',
  169. x: payload.x,
  170. y: payload.y
  171. });
  172. this._tryShow({
  173. offsetX: payload.x,
  174. offsetY: payload.y,
  175. position: payload.position,
  176. target: api.getZr().findHover(payload.x, payload.y).target,
  177. event: {}
  178. }, dispatchAction);
  179. }
  180. },
  181. manuallyHideTip: function (tooltipModel, ecModel, api, payload) {
  182. var tooltipContent = this._tooltipContent;
  183. if (!this._alwaysShowContent) {
  184. tooltipContent.hideLater(this._tooltipModel.get('hideDelay'));
  185. }
  186. this._lastX = this._lastY = null;
  187. if (payload.from !== this.uid) {
  188. this._hide(makeDispatchAction(payload, api));
  189. }
  190. },
  191. // Be compatible with previous design, that is, when tooltip.type is 'axis' and
  192. // dispatchAction 'showTip' with seriesIndex and dataIndex will trigger axis pointer
  193. // and tooltip.
  194. _manuallyAxisShowTip: function (tooltipModel, ecModel, api, payload) {
  195. var seriesIndex = payload.seriesIndex;
  196. var dataIndex = payload.dataIndex;
  197. var coordSysAxesInfo = ecModel.getComponent('axisPointer').coordSysAxesInfo;
  198. if (seriesIndex == null || dataIndex == null || coordSysAxesInfo == null) {
  199. return;
  200. }
  201. var seriesModel = ecModel.getSeriesByIndex(seriesIndex);
  202. if (!seriesModel) {
  203. return;
  204. }
  205. var data = seriesModel.getData();
  206. var tooltipModel = buildTooltipModel([data.getItemModel(dataIndex), seriesModel, (seriesModel.coordinateSystem || {}).model, tooltipModel]);
  207. if (tooltipModel.get('trigger') !== 'axis') {
  208. return;
  209. }
  210. api.dispatchAction({
  211. type: 'updateAxisPointer',
  212. seriesIndex: seriesIndex,
  213. dataIndex: dataIndex,
  214. position: payload.position
  215. });
  216. return true;
  217. },
  218. _tryShow: function (e, dispatchAction) {
  219. var el = e.target;
  220. var tooltipModel = this._tooltipModel;
  221. if (!tooltipModel) {
  222. return;
  223. } // Save mouse x, mouse y. So we can try to keep showing the tip if chart is refreshed
  224. this._lastX = e.offsetX;
  225. this._lastY = e.offsetY;
  226. var dataByCoordSys = e.dataByCoordSys;
  227. if (dataByCoordSys && dataByCoordSys.length) {
  228. this._showAxisTooltip(dataByCoordSys, e);
  229. } // Always show item tooltip if mouse is on the element with dataIndex
  230. else if (el && el.dataIndex != null) {
  231. this._lastDataByCoordSys = null;
  232. this._showSeriesItemTooltip(e, el, dispatchAction);
  233. } // Tooltip provided directly. Like legend.
  234. else if (el && el.tooltip) {
  235. this._lastDataByCoordSys = null;
  236. this._showComponentItemTooltip(e, el, dispatchAction);
  237. } else {
  238. this._lastDataByCoordSys = null;
  239. this._hide(dispatchAction);
  240. }
  241. },
  242. _showOrMove: function (tooltipModel, cb) {
  243. // showDelay is used in this case: tooltip.enterable is set
  244. // as true. User intent to move mouse into tooltip and click
  245. // something. `showDelay` makes it easyer to enter the content
  246. // but tooltip do not move immediately.
  247. var delay = tooltipModel.get('showDelay');
  248. cb = zrUtil.bind(cb, this);
  249. clearTimeout(this._showTimout);
  250. delay > 0 ? this._showTimout = setTimeout(cb, delay) : cb();
  251. },
  252. _showAxisTooltip: function (dataByCoordSys, e) {
  253. var ecModel = this._ecModel;
  254. var globalTooltipModel = this._tooltipModel;
  255. var point = [e.offsetX, e.offsetY];
  256. var singleDefaultHTML = [];
  257. var singleParamsList = [];
  258. var singleTooltipModel = buildTooltipModel([e.tooltipOption, globalTooltipModel]);
  259. each(dataByCoordSys, function (itemCoordSys) {
  260. // var coordParamList = [];
  261. // var coordDefaultHTML = [];
  262. // var coordTooltipModel = buildTooltipModel([
  263. // e.tooltipOption,
  264. // itemCoordSys.tooltipOption,
  265. // ecModel.getComponent(itemCoordSys.coordSysMainType, itemCoordSys.coordSysIndex),
  266. // globalTooltipModel
  267. // ]);
  268. // var displayMode = coordTooltipModel.get('displayMode');
  269. // var paramsList = displayMode === 'single' ? singleParamsList : [];
  270. each(itemCoordSys.dataByAxis, function (item) {
  271. var axisModel = ecModel.getComponent(item.axisDim + 'Axis', item.axisIndex);
  272. var axisValue = item.value;
  273. var seriesDefaultHTML = [];
  274. if (!axisModel || axisValue == null) {
  275. return;
  276. }
  277. var valueLabel = axisPointerViewHelper.getValueLabel(axisValue, axisModel.axis, ecModel, item.seriesDataIndices, item.valueLabelOpt);
  278. zrUtil.each(item.seriesDataIndices, function (idxItem) {
  279. var series = ecModel.getSeriesByIndex(idxItem.seriesIndex);
  280. var dataIndex = idxItem.dataIndexInside;
  281. var dataParams = series && series.getDataParams(dataIndex);
  282. dataParams.axisDim = item.axisDim;
  283. dataParams.axisIndex = item.axisIndex;
  284. dataParams.axisType = item.axisType;
  285. dataParams.axisId = item.axisId;
  286. dataParams.axisValue = axisHelper.getAxisRawValue(axisModel.axis, axisValue);
  287. dataParams.axisValueLabel = valueLabel;
  288. if (dataParams) {
  289. singleParamsList.push(dataParams);
  290. seriesDefaultHTML.push(series.formatTooltip(dataIndex, true));
  291. }
  292. }); // Default tooltip content
  293. // FIXME
  294. // (1) shold be the first data which has name?
  295. // (2) themeRiver, firstDataIndex is array, and first line is unnecessary.
  296. var firstLine = valueLabel;
  297. singleDefaultHTML.push((firstLine ? formatUtil.encodeHTML(firstLine) + '<br />' : '') + seriesDefaultHTML.join('<br />'));
  298. });
  299. }, this); // In most case, the second axis is shown upper than the first one.
  300. singleDefaultHTML.reverse();
  301. singleDefaultHTML = singleDefaultHTML.join('<br /><br />');
  302. var positionExpr = e.position;
  303. this._showOrMove(singleTooltipModel, function () {
  304. if (this._updateContentNotChangedOnAxis(dataByCoordSys)) {
  305. this._updatePosition(singleTooltipModel, positionExpr, point[0], point[1], this._tooltipContent, singleParamsList);
  306. } else {
  307. this._showTooltipContent(singleTooltipModel, singleDefaultHTML, singleParamsList, Math.random(), point[0], point[1], positionExpr);
  308. }
  309. }); // Do not trigger events here, because this branch only be entered
  310. // from dispatchAction.
  311. },
  312. _showSeriesItemTooltip: function (e, el, dispatchAction) {
  313. var ecModel = this._ecModel; // Use dataModel in element if possible
  314. // Used when mouseover on a element like markPoint or edge
  315. // In which case, the data is not main data in series.
  316. var seriesIndex = el.seriesIndex;
  317. var seriesModel = ecModel.getSeriesByIndex(seriesIndex); // For example, graph link.
  318. var dataModel = el.dataModel || seriesModel;
  319. var dataIndex = el.dataIndex;
  320. var dataType = el.dataType;
  321. var data = dataModel.getData();
  322. var tooltipModel = buildTooltipModel([data.getItemModel(dataIndex), dataModel, seriesModel && (seriesModel.coordinateSystem || {}).model, this._tooltipModel]);
  323. var tooltipTrigger = tooltipModel.get('trigger');
  324. if (tooltipTrigger != null && tooltipTrigger !== 'item') {
  325. return;
  326. }
  327. var params = dataModel.getDataParams(dataIndex, dataType);
  328. var defaultHtml = dataModel.formatTooltip(dataIndex, false, dataType);
  329. var asyncTicket = 'item_' + dataModel.name + '_' + dataIndex;
  330. this._showOrMove(tooltipModel, function () {
  331. this._showTooltipContent(tooltipModel, defaultHtml, params, asyncTicket, e.offsetX, e.offsetY, e.position, e.target);
  332. }); // FIXME
  333. // duplicated showtip if manuallyShowTip is called from dispatchAction.
  334. dispatchAction({
  335. type: 'showTip',
  336. dataIndexInside: dataIndex,
  337. dataIndex: data.getRawIndex(dataIndex),
  338. seriesIndex: seriesIndex,
  339. from: this.uid
  340. });
  341. },
  342. _showComponentItemTooltip: function (e, el, dispatchAction) {
  343. var tooltipOpt = el.tooltip;
  344. if (typeof tooltipOpt === 'string') {
  345. var content = tooltipOpt;
  346. tooltipOpt = {
  347. content: content,
  348. // Fixed formatter
  349. formatter: content
  350. };
  351. }
  352. var subTooltipModel = new Model(tooltipOpt, this._tooltipModel, this._ecModel);
  353. var defaultHtml = subTooltipModel.get('content');
  354. var asyncTicket = Math.random(); // Do not check whether `trigger` is 'none' here, because `trigger`
  355. // only works on cooridinate system. In fact, we have not found case
  356. // that requires setting `trigger` nothing on component yet.
  357. this._showOrMove(subTooltipModel, function () {
  358. this._showTooltipContent(subTooltipModel, defaultHtml, subTooltipModel.get('formatterParams') || {}, asyncTicket, e.offsetX, e.offsetY, e.position, el);
  359. }); // If not dispatch showTip, tip may be hide triggered by axis.
  360. dispatchAction({
  361. type: 'showTip',
  362. from: this.uid
  363. });
  364. },
  365. _showTooltipContent: function (tooltipModel, defaultHtml, params, asyncTicket, x, y, positionExpr, el) {
  366. // Reset ticket
  367. this._ticket = '';
  368. if (!tooltipModel.get('showContent') || !tooltipModel.get('show')) {
  369. return;
  370. }
  371. var tooltipContent = this._tooltipContent;
  372. var formatter = tooltipModel.get('formatter');
  373. positionExpr = positionExpr || tooltipModel.get('position');
  374. var html = defaultHtml;
  375. if (formatter && typeof formatter === 'string') {
  376. html = formatUtil.formatTpl(formatter, params, true);
  377. } else if (typeof formatter === 'function') {
  378. var callback = bind(function (cbTicket, html) {
  379. if (cbTicket === this._ticket) {
  380. tooltipContent.setContent(html);
  381. this._updatePosition(tooltipModel, positionExpr, x, y, tooltipContent, params, el);
  382. }
  383. }, this);
  384. this._ticket = asyncTicket;
  385. html = formatter(params, asyncTicket, callback);
  386. }
  387. tooltipContent.setContent(html);
  388. tooltipContent.show(tooltipModel);
  389. this._updatePosition(tooltipModel, positionExpr, x, y, tooltipContent, params, el);
  390. },
  391. /**
  392. * @param {string|Function|Array.<number>|Object} positionExpr
  393. * @param {number} x Mouse x
  394. * @param {number} y Mouse y
  395. * @param {boolean} confine Whether confine tooltip content in view rect.
  396. * @param {Object|<Array.<Object>} params
  397. * @param {module:zrender/Element} el target element
  398. * @param {module:echarts/ExtensionAPI} api
  399. * @return {Array.<number>}
  400. */
  401. _updatePosition: function (tooltipModel, positionExpr, x, y, content, params, el) {
  402. var viewWidth = this._api.getWidth();
  403. var viewHeight = this._api.getHeight();
  404. positionExpr = positionExpr || tooltipModel.get('position');
  405. var contentSize = content.getSize();
  406. var align = tooltipModel.get('align');
  407. var vAlign = tooltipModel.get('verticalAlign');
  408. var rect = el && el.getBoundingRect().clone();
  409. el && rect.applyTransform(el.transform);
  410. if (typeof positionExpr === 'function') {
  411. // Callback of position can be an array or a string specify the position
  412. positionExpr = positionExpr([x, y], params, content.el, rect, {
  413. viewSize: [viewWidth, viewHeight],
  414. contentSize: contentSize.slice()
  415. });
  416. }
  417. if (zrUtil.isArray(positionExpr)) {
  418. x = parsePercent(positionExpr[0], viewWidth);
  419. y = parsePercent(positionExpr[1], viewHeight);
  420. } else if (zrUtil.isObject(positionExpr)) {
  421. positionExpr.width = contentSize[0];
  422. positionExpr.height = contentSize[1];
  423. var layoutRect = layoutUtil.getLayoutRect(positionExpr, {
  424. width: viewWidth,
  425. height: viewHeight
  426. });
  427. x = layoutRect.x;
  428. y = layoutRect.y;
  429. align = null; // When positionExpr is left/top/right/bottom,
  430. // align and verticalAlign will not work.
  431. vAlign = null;
  432. } // Specify tooltip position by string 'top' 'bottom' 'left' 'right' around graphic element
  433. else if (typeof positionExpr === 'string' && el) {
  434. var pos = calcTooltipPosition(positionExpr, rect, contentSize);
  435. x = pos[0];
  436. y = pos[1];
  437. } else {
  438. var pos = refixTooltipPosition(x, y, content.el, viewWidth, viewHeight, align ? null : 20, vAlign ? null : 20);
  439. x = pos[0];
  440. y = pos[1];
  441. }
  442. align && (x -= isCenterAlign(align) ? contentSize[0] / 2 : align === 'right' ? contentSize[0] : 0);
  443. vAlign && (y -= isCenterAlign(vAlign) ? contentSize[1] / 2 : vAlign === 'bottom' ? contentSize[1] : 0);
  444. if (tooltipModel.get('confine')) {
  445. var pos = confineTooltipPosition(x, y, content.el, viewWidth, viewHeight);
  446. x = pos[0];
  447. y = pos[1];
  448. }
  449. content.moveTo(x, y);
  450. },
  451. // FIXME
  452. // Should we remove this but leave this to user?
  453. _updateContentNotChangedOnAxis: function (dataByCoordSys) {
  454. var lastCoordSys = this._lastDataByCoordSys;
  455. var contentNotChanged = !!lastCoordSys && lastCoordSys.length === dataByCoordSys.length;
  456. contentNotChanged && each(lastCoordSys, function (lastItemCoordSys, indexCoordSys) {
  457. var lastDataByAxis = lastItemCoordSys.dataByAxis || {};
  458. var thisItemCoordSys = dataByCoordSys[indexCoordSys] || {};
  459. var thisDataByAxis = thisItemCoordSys.dataByAxis || [];
  460. contentNotChanged &= lastDataByAxis.length === thisDataByAxis.length;
  461. contentNotChanged && each(lastDataByAxis, function (lastItem, indexAxis) {
  462. var thisItem = thisDataByAxis[indexAxis] || {};
  463. var lastIndices = lastItem.seriesDataIndices || [];
  464. var newIndices = thisItem.seriesDataIndices || [];
  465. contentNotChanged &= lastItem.value === thisItem.value && lastItem.axisType === thisItem.axisType && lastItem.axisId === thisItem.axisId && lastIndices.length === newIndices.length;
  466. contentNotChanged && each(lastIndices, function (lastIdxItem, j) {
  467. var newIdxItem = newIndices[j];
  468. contentNotChanged &= lastIdxItem.seriesIndex === newIdxItem.seriesIndex && lastIdxItem.dataIndex === newIdxItem.dataIndex;
  469. });
  470. });
  471. });
  472. this._lastDataByCoordSys = dataByCoordSys;
  473. return !!contentNotChanged;
  474. },
  475. _hide: function (dispatchAction) {
  476. // Do not directly hideLater here, because this behavior may be prevented
  477. // in dispatchAction when showTip is dispatched.
  478. // FIXME
  479. // duplicated hideTip if manuallyHideTip is called from dispatchAction.
  480. this._lastDataByCoordSys = null;
  481. dispatchAction({
  482. type: 'hideTip',
  483. from: this.uid
  484. });
  485. },
  486. dispose: function (ecModel, api) {
  487. if (env.node) {
  488. return;
  489. }
  490. this._tooltipContent.hide();
  491. globalListener.unregister('itemTooltip', api);
  492. }
  493. });
  494. /**
  495. * @param {Array.<Object|module:echarts/model/Model>} modelCascade
  496. * From top to bottom. (the last one should be globalTooltipModel);
  497. */
  498. function buildTooltipModel(modelCascade) {
  499. var resultModel = modelCascade.pop();
  500. while (modelCascade.length) {
  501. var tooltipOpt = modelCascade.pop();
  502. if (tooltipOpt) {
  503. if (tooltipOpt instanceof Model) {
  504. tooltipOpt = tooltipOpt.get('tooltip', true);
  505. } // In each data item tooltip can be simply write:
  506. // {
  507. // value: 10,
  508. // tooltip: 'Something you need to know'
  509. // }
  510. if (typeof tooltipOpt === 'string') {
  511. tooltipOpt = {
  512. formatter: tooltipOpt
  513. };
  514. }
  515. resultModel = new Model(tooltipOpt, resultModel, resultModel.ecModel);
  516. }
  517. }
  518. return resultModel;
  519. }
  520. function makeDispatchAction(payload, api) {
  521. return payload.dispatchAction || zrUtil.bind(api.dispatchAction, api);
  522. }
  523. function refixTooltipPosition(x, y, el, viewWidth, viewHeight, gapH, gapV) {
  524. var size = getOuterSize(el);
  525. var width = size.width;
  526. var height = size.height;
  527. if (gapH != null) {
  528. if (x + width + gapH > viewWidth) {
  529. x -= width + gapH;
  530. } else {
  531. x += gapH;
  532. }
  533. }
  534. if (gapV != null) {
  535. if (y + height + gapV > viewHeight) {
  536. y -= height + gapV;
  537. } else {
  538. y += gapV;
  539. }
  540. }
  541. return [x, y];
  542. }
  543. function confineTooltipPosition(x, y, el, viewWidth, viewHeight) {
  544. var size = getOuterSize(el);
  545. var width = size.width;
  546. var height = size.height;
  547. x = Math.min(x + width, viewWidth) - width;
  548. y = Math.min(y + height, viewHeight) - height;
  549. x = Math.max(x, 0);
  550. y = Math.max(y, 0);
  551. return [x, y];
  552. }
  553. function getOuterSize(el) {
  554. var width = el.clientWidth;
  555. var height = el.clientHeight; // Consider browser compatibility.
  556. // IE8 does not support getComputedStyle.
  557. if (document.defaultView && document.defaultView.getComputedStyle) {
  558. var stl = document.defaultView.getComputedStyle(el);
  559. if (stl) {
  560. width += parseInt(stl.paddingLeft, 10) + parseInt(stl.paddingRight, 10) + parseInt(stl.borderLeftWidth, 10) + parseInt(stl.borderRightWidth, 10);
  561. height += parseInt(stl.paddingTop, 10) + parseInt(stl.paddingBottom, 10) + parseInt(stl.borderTopWidth, 10) + parseInt(stl.borderBottomWidth, 10);
  562. }
  563. }
  564. return {
  565. width: width,
  566. height: height
  567. };
  568. }
  569. function calcTooltipPosition(position, rect, contentSize) {
  570. var domWidth = contentSize[0];
  571. var domHeight = contentSize[1];
  572. var gap = 5;
  573. var x = 0;
  574. var y = 0;
  575. var rectWidth = rect.width;
  576. var rectHeight = rect.height;
  577. switch (position) {
  578. case 'inside':
  579. x = rect.x + rectWidth / 2 - domWidth / 2;
  580. y = rect.y + rectHeight / 2 - domHeight / 2;
  581. break;
  582. case 'top':
  583. x = rect.x + rectWidth / 2 - domWidth / 2;
  584. y = rect.y - domHeight - gap;
  585. break;
  586. case 'bottom':
  587. x = rect.x + rectWidth / 2 - domWidth / 2;
  588. y = rect.y + rectHeight + gap;
  589. break;
  590. case 'left':
  591. x = rect.x - domWidth - gap;
  592. y = rect.y + rectHeight / 2 - domHeight / 2;
  593. break;
  594. case 'right':
  595. x = rect.x + rectWidth + gap;
  596. y = rect.y + rectHeight / 2 - domHeight / 2;
  597. }
  598. return [x, y];
  599. }
  600. function isCenterAlign(align) {
  601. return align === 'center' || align === 'middle';
  602. }
  603. module.exports = _default;