Source: chart.js

  1. /*
  2. * JBoss, Home of Professional Open Source
  3. * Copyright ${year}, Red Hat, Inc. and individual contributors
  4. * by the @authors tag. See the copyright.txt in the distribution for a
  5. * full listing of individual contributors.
  6. *
  7. * This is free software; you can redistribute it and/or modify it
  8. * under the terms of the GNU Lesser General Public License as
  9. * published by the Free Software Foundation; either version 2.1 of
  10. * the License, or (at your option) any later version.
  11. *
  12. * This software is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public
  18. * License along with this software; if not, write to the Free
  19. * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
  20. * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
  21. */
  22. (function ($, rf) {
  23. var defaultOptions = {
  24. /**
  25. * Specifies the type of chart.
  26. * According to this property and xtypex ytype, proper default options corresponding to chart type
  27. * are applied and data are transformed to the format expected by flot.
  28. * Following values are supported: `line`,`bar`,`pie`
  29. * Options is not required. If it is not used the options and data format remains the same as flot uses by default.
  30. * @property charttype
  31. * @default null
  32. */
  33. charttype: '',
  34. /**
  35. * Specify the data type of values plotted on x-axis
  36. * Following options are supported: number,string,date
  37. * @property xtype
  38. * @default null
  39. */
  40. xtype: '',
  41. /**
  42. * Specify the data type of values plotted on x-axis
  43. * Following options are supported: number,string,date
  44. * @property xtype
  45. * @default null
  46. */
  47. ytype: '',
  48. /**
  49. * Allows to zoom the chart. Supported only when line chart is used. Requires charttype to be set.
  50. * @property zoom
  51. * @default false
  52. */
  53. zoom: false,
  54. /**
  55. * Options customizing the chart grid.
  56. * @type Object
  57. * @default {clickable:true, hoverable:true}
  58. * @property grid
  59. * @property clickable {boolean}
  60. * Grid accepts click events.
  61. * @property hoverable {boolean}
  62. * Grid fires mouseover event. Necessary for tooltip to work.
  63. *
  64. */
  65. grid: {
  66. clickable: true,
  67. hoverable: true
  68. },
  69. /**
  70. * Turns on tooltip (text shown when mouse points to a part of a chart)
  71. * @property tooltip
  72. * @type boolean
  73. * @default true
  74. */
  75. tooltip: true,
  76. /**
  77. * Customizes the tooltip.
  78. * @type Object
  79. * @default { content: '%s [%x,%y]'}
  80. * @property tooltipOpts
  81. * @property content {string}
  82. * Specify the tooltip format. Use %s for series label, %x for X values, %y for Y value
  83. * @property defaultTheme
  84. */
  85. tooltipOpts: {
  86. content: '%s [%x,%y]',
  87. shifts: {
  88. x: 20,
  89. y: 0
  90. },
  91. defaultTheme: false
  92. },
  93. /**
  94. * Legend properties
  95. * @type Object
  96. * @default {postion:'ne', sorted: 'ascending'}
  97. * @property legend
  98. * @property position {string}
  99. * Defines the placement of the legend in the grid. One of ne,nw,se,sw
  100. * @property sorted {string}
  101. * Defines the order of labels in the legend. One of ascending,descending,false.
  102. */
  103. legend: {
  104. postion:'ne',
  105. sorted: 'ascending'
  106. },
  107. /**
  108. * Customizes the horizontal axis
  109. * @type Object
  110. * @default {min: null, max: null,autoscaleMargin: null, axisLabel: ''}
  111. * @property xaxis
  112. * @property min {number}
  113. * Minimal value shown on axis
  114. * @property max {number}
  115. * Maximal values show on axis
  116. * @property autoscaleMargin {number}
  117. * It's the fraction of margin that the scaling algorithm will add
  118. * to avoid that the outermost points ends up on the grid border
  119. * @property axisLabel {string}
  120. * Axis description
  121. */
  122. xaxis:{
  123. min: null,
  124. max: null,
  125. autoscaleMargin: null,
  126. axisLabel: ''
  127. },
  128. /**
  129. * Customizes the vertical axis
  130. * @type Object
  131. * @default {min: null, max: null,autoscaleMargin: 0.2,axisLabel: ''}
  132. * @property yaxis
  133. * @property min {number}
  134. * Minimal value shown on axis
  135. * @property max {number}
  136. * Maximal values show on axis
  137. * @property autoscaleMargin {number}
  138. * It's the fraction of margin that the scaling algorithm will add to
  139. * avoid that the outermost points ends up on the grid border.
  140. * @property axisLabel {string}
  141. * Axis description
  142. */
  143. yaxis:{
  144. min: null,
  145. max: null,
  146. autoscaleMargin: 0.2,
  147. axisLabel: ''
  148. },
  149. /**
  150. * Data to be plotted. The format is the same used by flot. The format may differ if the charttype
  151. * option is set to pie or bar.
  152. * @property data
  153. * @default []
  154. */
  155. data:[]
  156. };
  157. var pieDefaults = {
  158. series: {
  159. pie: {
  160. show: true
  161. }
  162. },
  163. tooltipOpts: {
  164. content: ' %p.0%, %s'
  165. }
  166. };
  167. var _plotClickServerSide = function (event, clientId) {
  168. var params = {};
  169. params[clientId + 'name'] = "plotclick";
  170. params[clientId + 'seriesIndex'] = event.data.seriesIndex;
  171. params[clientId + 'dataIndex'] = event.data.dataIndex;
  172. params[clientId + 'x'] = event.data.x;
  173. params[clientId + 'y'] = event.data.y;
  174. rf.ajax(clientId, event, {
  175. parameters: params,
  176. incId: 1
  177. });
  178. };
  179. var handleStringTicks = function(options) {
  180. /*
  181. * data transformation:
  182. * data: [["label1", value1], ["label2", value2"], …]
  183. *
  184. * =>
  185. *
  186. * data: [[0, value1], [1, value2], …]
  187. * ticks: [[0, "label1"], [1, "label2"], …]
  188. *
  189. */
  190. options.xaxis.tickLength = 0;
  191. var seriesLength,
  192. seriesTotal = options.data.length,
  193. _tickNames = {},
  194. _tickTotal = 0,
  195. _currTickNumber,
  196. _currItem;
  197. if (options.charttype === 'bar') {
  198. options.bars = options.bars || {};
  199. options.bars.barWidth = 1 / (seriesTotal + 1);
  200. }
  201. for (var i = 0; i < seriesTotal; i++) {
  202. seriesLength = options.data[i].data.length;
  203. if (options.charttype === 'bar') {
  204. options.data[i].bars.order = i;
  205. }
  206. for (var j = 0; j < seriesLength; j++) {
  207. _currItem = options.data[i].data[j];
  208. _currTickNumber = _tickNames[_currItem[0]];
  209. if (_currTickNumber == undefined) {
  210. _currTickNumber = _tickTotal++;
  211. _tickNames[_currItem[0]] = _currTickNumber;
  212. }
  213. _currItem[0] = _currTickNumber;
  214. }
  215. }
  216. options.xaxis.ticks = [];
  217. for (tick in _tickNames) {
  218. options.xaxis.ticks.push([_tickNames[tick],tick]);
  219. }
  220. };
  221. rf.ui = rf.ui || {};
  222. rf.ui.Chart = rf.BaseComponent.extendClass({
  223. // class name
  224. name:"Chart",
  225. /**
  226. * Backing object for rich:chart
  227. *
  228. * @extends RichFaces.BaseComponent
  229. * @memberOf! RichFaces.ui
  230. * @constructs RichFaces.ui.Chart
  231. *
  232. * @param componentId
  233. * @param options
  234. */
  235. init : function (componentId, options) {
  236. $super.constructor.call(this, componentId, options);
  237. this.namespace = this.namespace || "." + RichFaces.Event.createNamespace(this.name, this.id);
  238. this.attachToDom();
  239. this.options = $.extend(true,{},defaultOptions,options);
  240. this.element = $(document.getElementById(componentId));
  241. this.chartElement = this.element.find(".chart");
  242. if (this.options.charttype === 'pie') {
  243. this.options = $.extend(true,{},this.options,pieDefaults);
  244. //transform data from
  245. // [[{data:1, label:"label1"},{data:2,label:"label2"},...]]
  246. //to
  247. // [{data:1, label:"label1"},{data:2,label:"label2"},...]
  248. this.options.data = this.options.data[0]; //pie chart data should not be in a collection
  249. }
  250. else if (this.options.charttype === 'bar') {
  251. if (this.options.xtype === 'string') {
  252. handleStringTicks(this.options);
  253. }
  254. }
  255. else if (options.charttype === 'line') {
  256. if (this.options.xtype === 'string') {
  257. handleStringTicks(this.options);
  258. }
  259. if (options.zoom) {
  260. this.options.selection = {mode: 'xy'};
  261. }
  262. if (this.options.xtype === 'date') {
  263. this.options = $.extend({},this.options, dateDefaults);
  264. if (this.options.xaxis.format) {
  265. this.options.xaxis.timeformat = this.options.xaxis.format;
  266. }
  267. }
  268. }
  269. this.plot = $.plot(this.chartElement,this.options.data,this.options);
  270. this.__bindEventHandlers(this.chartElement,this.options);
  271. },
  272. /***************************** Public Methods ****************************************************************/
  273. /**
  274. * Returns chart object
  275. *
  276. * @method
  277. * @name RichFaces.ui.Chart#getPlotObject
  278. * @return {Plot} chart object
  279. */
  280. getPlotObject: function () {
  281. return this.plot;
  282. },
  283. /**
  284. * Highlights the point in chart selected by seriesIndex or point index. Does not work for pie charts.
  285. *
  286. * @method
  287. * @name RichFaces.ui.Chart#highlight
  288. * @param seriesIndex {int} index of the series
  289. * @param pointIndex {int} index of the point
  290. */
  291. highlight: function(seriesIndex,pointIndex){
  292. this.plot.highlight(seriesIndex,pointIndex);
  293. },
  294. /**
  295. * Removes highlighting of point. If method is called without parameters it unhighlights all points.
  296. *
  297. * @method
  298. * @name RichFaces.ui.Chart#unhighlight
  299. * @param seriesIndex {int} index of the series
  300. * @param pointIndex {int} index of the point
  301. */
  302. unhighlight: function(seriesIndex,pointIndex){
  303. this.plot.unhighlight(seriesIndex,pointIndex);
  304. },
  305. /***************************** Private Methods ********************************************************/
  306. __bindEventHandlers:function(element,options){
  307. element.on('plotclick', this._getPlotClickHandler(options, element.get(0), _plotClickServerSide));
  308. element.on('plothover', this._getPlotHoverHandler(options, element.get(0)));
  309. if (options.handlers && options.handlers.onmouseout) {
  310. element.on('mouseout', options.handlers.onmouseout);
  311. }
  312. if (options.zoom) {
  313. element.on('plotselected', $.proxy(this._zoomFunction, this));
  314. }
  315. },
  316. //function handles plotclick event. it calls server-side, client-side and particular series handlers if set.
  317. _getPlotClickHandler: function (options, element, serverSideHandler) {
  318. var clickHandler = options.handlers['onplotclick'];
  319. var particularClickHandlers= options.particularSeriesHandlers['onplotclick'];
  320. var clientId = this.element.attr('id');
  321. return function (event, pos, item) {
  322. if (item !== null) {
  323. //point in a chart clicked
  324. event.data = {
  325. seriesIndex: item.seriesIndex,
  326. dataIndex: item.dataIndex,
  327. x: item.datapoint[0],
  328. y: item.datapoint[1],
  329. item: item
  330. };
  331. if(options.charttype=="pie"){
  332. event.data.x = options.data[item.seriesIndex].label;
  333. event.data.y = item.datapoint[1][0][1];
  334. }
  335. else if(options.charttype=="bar" && options.xtype=="string"){
  336. event.data.x = options.xaxis.ticks[item.dataIndex][1];
  337. }
  338. } else {
  339. event.data = {
  340. x: pos.x,
  341. y: pos.y
  342. };
  343. }
  344. //sent request only if a server-side listener attached
  345. if (options.serverSideListener) {
  346. //server-side
  347. if (serverSideHandler) {
  348. serverSideHandler(event, clientId);
  349. }
  350. }
  351. //client-side
  352. if (clickHandler) {
  353. clickHandler.call(element, event);
  354. }
  355. //client-side particular series handler
  356. if (particularClickHandlers[event.data.seriesIndex]) {
  357. particularClickHandlers[event.data.seriesIndex].call(element, event);
  358. }
  359. };
  360. },
  361. //function handles plothover event. it calls client-side and particular series handlers if set.
  362. _getPlotHoverHandler: function (options, element) {
  363. var hoverHandler = options.handlers['onplothover'];
  364. var particularHoverHandlers = options.particularSeriesHandlers['onplothover'];
  365. return function (event, pos, item) {
  366. if (item !== null) {
  367. //point in a chart clicked
  368. event.data = {
  369. seriesIndex: item.seriesIndex,
  370. dataIndex: item.dataIndex,
  371. x: item.datapoint[0],
  372. y: item.datapoint[1],
  373. item: item
  374. };
  375. } else {
  376. event.data = {
  377. x: pos.x,
  378. y: pos.y
  379. };
  380. }
  381. //client-side
  382. if (hoverHandler) {
  383. hoverHandler.call(element, event);
  384. }
  385. //client-side particular series handler
  386. if (particularHoverHandlers[event.data.seriesIndex]) {
  387. particularHoverHandlers[event.data.seriesIndex].call(element, event);
  388. }
  389. };
  390. },
  391. _zoomFunction: function (event, ranges) {
  392. var plot = this.getPlotObject();
  393. $.each(plot.getXAxes(), function(_, axis) {
  394. var opts = axis.options;
  395. opts.min = ranges.xaxis.from;
  396. opts.max = ranges.xaxis.to;
  397. });
  398. plot.setupGrid();
  399. plot.draw();
  400. plot.clearSelection();
  401. },
  402. /**
  403. * Reset the zoom level
  404. *
  405. * @method
  406. * @name RichFaces.ui.Chart#resetZoom
  407. */
  408. resetZoom: function () {
  409. this.plot = $.plot(this.chartElement,this.options.data,this.options);
  410. },
  411. destroy: function () {
  412. rf.Event.unbindById(this.id, "." + this.namespace);
  413. $super.destroy.call(this);
  414. }
  415. });
  416. // define super class link
  417. var $super = rf.ui.Chart.$super;
  418. })(RichFaces.jQuery, RichFaces);