zoom_plot_jqplot.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587
  1. // TODO: change the axis
  2. /**
  3. * @fileoverview JavaScript functions used on /table/search
  4. *
  5. * @requires jQuery
  6. * @requires js/functions.js
  7. **/
  8. /* global changeValueFieldType, verifyAfterSearchFieldChange */ // js/table/change.js
  9. /**
  10. * Display Help/Info
  11. * @return {false}
  12. **/
  13. function displayHelp() {
  14. var modal = $('#helpModal');
  15. modal.modal('show');
  16. modal.find('.modal-body').first().html(Messages.strDisplayHelp);
  17. $('#helpModalLabel').first().html(Messages.strHelpTitle);
  18. return false;
  19. }
  20. /**
  21. * Extend the array object for max function
  22. * @param {number[]} array
  23. * @return {int}
  24. **/
  25. Array.max = function (array) {
  26. return Math.max.apply(Math, array);
  27. };
  28. /**
  29. * Extend the array object for min function
  30. * @param {number[]} array
  31. * @return {int}
  32. **/
  33. Array.min = function (array) {
  34. return Math.min.apply(Math, array);
  35. };
  36. /**
  37. * Checks if a string contains only numeric value
  38. * @param {string} n (to be checked)
  39. * @return {bool}
  40. **/
  41. function isNumeric(n) {
  42. return !isNaN(parseFloat(n)) && isFinite(n);
  43. }
  44. /**
  45. ** Checks if an object is empty
  46. * @param {object} obj (to be checked)
  47. * @return {bool}
  48. **/
  49. function isEmpty(obj) {
  50. var name;
  51. for (name in obj) {
  52. return false;
  53. }
  54. return true;
  55. }
  56. /**
  57. * Converts a date/time into timestamp
  58. * @param {string} val Date
  59. * @param {string} type Field type(datetime/timestamp/time/date)
  60. * @return {any} A value
  61. **/
  62. function getTimeStamp(val, type) {
  63. if (type.toString().search(/datetime/i) !== -1 || type.toString().search(/timestamp/i) !== -1) {
  64. return $.datepicker.parseDateTime('yy-mm-dd', 'HH:mm:ss', val);
  65. } else if (type.toString().search(/time/i) !== -1) {
  66. return $.datepicker.parseDateTime('yy-mm-dd', 'HH:mm:ss', '1970-01-01 ' + val);
  67. } else if (type.toString().search(/date/i) !== -1) {
  68. return $.datepicker.parseDate('yy-mm-dd', val);
  69. }
  70. }
  71. /**
  72. * Classifies the field type into numeric,timeseries or text
  73. * @param {object} field field type (as in database structure)
  74. * @return {'text'|'numeric'|'time'}
  75. **/
  76. function getType(field) {
  77. if (field.toString().search(/int/i) !== -1 || field.toString().search(/decimal/i) !== -1 || field.toString().search(/year/i) !== -1) {
  78. return 'numeric';
  79. } else if (field.toString().search(/time/i) !== -1 || field.toString().search(/date/i) !== -1) {
  80. return 'time';
  81. } else {
  82. return 'text';
  83. }
  84. }
  85. /**
  86. * Unbind all event handlers before tearing down a page
  87. */
  88. AJAX.registerTeardown('table/zoom_plot_jqplot.js', function () {
  89. $('#tableid_0').off('change');
  90. $('#tableid_1').off('change');
  91. $('#tableid_2').off('change');
  92. $('#tableid_3').off('change');
  93. $('#inputFormSubmitId').off('click');
  94. $('#togglesearchformlink').off('click');
  95. $(document).off('keydown', '#dataDisplay :input');
  96. $('button.button-reset').off('click');
  97. $('div#resizer').off('resizestop');
  98. $('div#querychart').off('jqplotDataClick');
  99. });
  100. AJAX.registerOnload('table/zoom_plot_jqplot.js', function () {
  101. var currentChart = null;
  102. var searchedDataKey = null;
  103. var xLabel = $('#tableid_0').val();
  104. var yLabel = $('#tableid_1').val();
  105. // will be updated via Ajax
  106. var xType = $('#types_0').val();
  107. var yType = $('#types_1').val();
  108. var dataLabel = $('#dataLabel').val();
  109. // Get query result
  110. var searchedData;
  111. try {
  112. searchedData = JSON.parse($('#querydata').html());
  113. } catch (err) {
  114. searchedData = null;
  115. }
  116. // adding event listener on select after AJAX request
  117. var comparisonOperatorOnChange = function () {
  118. var tableRows = $('#inputSection select.column-operator');
  119. $.each(tableRows, function (index, item) {
  120. $(item).on('change', function () {
  121. changeValueFieldType(this, index);
  122. verifyAfterSearchFieldChange(index, '#zoom_search_form');
  123. });
  124. });
  125. };
  126. /**
  127. ** Input form submit on field change
  128. **/
  129. // first column choice corresponds to the X axis
  130. $('#tableid_0').on('change', function () {
  131. // AJAX request for field type, collation, operators, and value field
  132. $.post('index.php?route=/table/zoom-search', {
  133. 'ajax_request': true,
  134. 'change_tbl_info': true,
  135. 'server': CommonParams.get('server'),
  136. 'db': CommonParams.get('db'),
  137. 'table': CommonParams.get('table'),
  138. 'field': $('#tableid_0').val(),
  139. 'it': 0
  140. }, function (data) {
  141. $('#tableFieldsId').find('tr').eq(1).find('td').eq(0).html(data.field_type);
  142. $('#tableFieldsId').find('tr').eq(1).find('td').eq(1).html(data.field_collation);
  143. $('#tableFieldsId').find('tr').eq(1).find('td').eq(2).html(data.field_operators);
  144. $('#tableFieldsId').find('tr').eq(1).find('td').eq(3).html(data.field_value);
  145. xLabel = $('#tableid_0').val();
  146. $('#types_0').val(data.field_type);
  147. xType = data.field_type;
  148. $('#collations_0').val(data.field_collations);
  149. comparisonOperatorOnChange();
  150. Functions.addDateTimePicker();
  151. });
  152. });
  153. // second column choice corresponds to the Y axis
  154. $('#tableid_1').on('change', function () {
  155. // AJAX request for field type, collation, operators, and value field
  156. $.post('index.php?route=/table/zoom-search', {
  157. 'ajax_request': true,
  158. 'change_tbl_info': true,
  159. 'server': CommonParams.get('server'),
  160. 'db': CommonParams.get('db'),
  161. 'table': CommonParams.get('table'),
  162. 'field': $('#tableid_1').val(),
  163. 'it': 1
  164. }, function (data) {
  165. $('#tableFieldsId').find('tr').eq(2).find('td').eq(0).html(data.field_type);
  166. $('#tableFieldsId').find('tr').eq(2).find('td').eq(1).html(data.field_collation);
  167. $('#tableFieldsId').find('tr').eq(2).find('td').eq(2).html(data.field_operators);
  168. $('#tableFieldsId').find('tr').eq(2).find('td').eq(3).html(data.field_value);
  169. yLabel = $('#tableid_1').val();
  170. $('#types_1').val(data.field_type);
  171. yType = data.field_type;
  172. $('#collations_1').val(data.field_collations);
  173. comparisonOperatorOnChange();
  174. Functions.addDateTimePicker();
  175. });
  176. });
  177. $('#tableid_2').on('change', function () {
  178. // AJAX request for field type, collation, operators, and value field
  179. $.post('index.php?route=/table/zoom-search', {
  180. 'ajax_request': true,
  181. 'change_tbl_info': true,
  182. 'server': CommonParams.get('server'),
  183. 'db': CommonParams.get('db'),
  184. 'table': CommonParams.get('table'),
  185. 'field': $('#tableid_2').val(),
  186. 'it': 2
  187. }, function (data) {
  188. $('#tableFieldsId').find('tr').eq(4).find('td').eq(0).html(data.field_type);
  189. $('#tableFieldsId').find('tr').eq(4).find('td').eq(1).html(data.field_collation);
  190. $('#tableFieldsId').find('tr').eq(4).find('td').eq(2).html(data.field_operators);
  191. $('#tableFieldsId').find('tr').eq(4).find('td').eq(3).html(data.field_value);
  192. $('#types_2').val(data.field_type);
  193. $('#collations_2').val(data.field_collations);
  194. comparisonOperatorOnChange();
  195. Functions.addDateTimePicker();
  196. });
  197. });
  198. $('#tableid_3').on('change', function () {
  199. // AJAX request for field type, collation, operators, and value field
  200. $.post('index.php?route=/table/zoom-search', {
  201. 'ajax_request': true,
  202. 'change_tbl_info': true,
  203. 'server': CommonParams.get('server'),
  204. 'db': CommonParams.get('db'),
  205. 'table': CommonParams.get('table'),
  206. 'field': $('#tableid_3').val(),
  207. 'it': 3
  208. }, function (data) {
  209. $('#tableFieldsId').find('tr').eq(5).find('td').eq(0).html(data.field_type);
  210. $('#tableFieldsId').find('tr').eq(5).find('td').eq(1).html(data.field_collation);
  211. $('#tableFieldsId').find('tr').eq(5).find('td').eq(2).html(data.field_operators);
  212. $('#tableFieldsId').find('tr').eq(5).find('td').eq(3).html(data.field_value);
  213. $('#types_3').val(data.field_type);
  214. $('#collations_3').val(data.field_collations);
  215. comparisonOperatorOnChange();
  216. Functions.addDateTimePicker();
  217. });
  218. });
  219. /**
  220. * Input form validation
  221. **/
  222. $('#inputFormSubmitId').on('click', function () {
  223. if ($('#tableid_0').get(0).selectedIndex === 0 || $('#tableid_1').get(0).selectedIndex === 0) {
  224. Functions.ajaxShowMessage(Messages.strInputNull);
  225. } else if (xLabel === yLabel) {
  226. Functions.ajaxShowMessage(Messages.strSameInputs);
  227. }
  228. });
  229. /**
  230. ** Prepare a div containing a link, otherwise it's incorrectly displayed
  231. ** after a couple of clicks
  232. **/
  233. $('<div id="togglesearchformdiv"><a id="togglesearchformlink"></a></div>').insertAfter('#zoom_search_form')
  234. // don't show it until we have results on-screen
  235. .hide();
  236. $('#togglesearchformlink').html(Messages.strShowSearchCriteria).on('click', function () {
  237. var $link = $(this);
  238. $('#zoom_search_form').slideToggle();
  239. if ($link.text() === Messages.strHideSearchCriteria) {
  240. $link.text(Messages.strShowSearchCriteria);
  241. } else {
  242. $link.text(Messages.strHideSearchCriteria);
  243. }
  244. // avoid default click action
  245. return false;
  246. });
  247. /**
  248. * Handle saving of a row in the editor
  249. */
  250. var dataPointSave = function () {
  251. // Find changed values by comparing form values with selectedRow Object
  252. var newValues = {}; // Stores the values changed from original
  253. var sqlTypes = {};
  254. var it = 0;
  255. var xChange = false;
  256. var yChange = false;
  257. var key;
  258. var tempGetVal = function () {
  259. return $(this).val();
  260. };
  261. for (key in selectedRow) {
  262. var oldVal = selectedRow[key];
  263. var newVal = $('#edit_fields_null_id_' + it).prop('checked') ? null : $('#edit_fieldID_' + it).val();
  264. if (newVal instanceof Array) {
  265. // when the column is of type SET
  266. newVal = $('#edit_fieldID_' + it).map(tempGetVal).get().join(',');
  267. }
  268. if (oldVal !== newVal) {
  269. selectedRow[key] = newVal;
  270. newValues[key] = newVal;
  271. if (key === xLabel) {
  272. xChange = true;
  273. searchedData[searchedDataKey][xLabel] = newVal;
  274. } else if (key === yLabel) {
  275. yChange = true;
  276. searchedData[searchedDataKey][yLabel] = newVal;
  277. }
  278. }
  279. var $input = $('#edit_fieldID_' + it);
  280. if ($input.hasClass('bit')) {
  281. sqlTypes[key] = 'bit';
  282. } else {
  283. sqlTypes[key] = null;
  284. }
  285. it++;
  286. } // End data update
  287. // Update the chart series and replot
  288. if (xChange || yChange) {
  289. // Logic similar to plot generation, replot only if xAxis changes or yAxis changes.
  290. // Code includes a lot of checks so as to replot only when necessary
  291. if (xChange) {
  292. xCord[searchedDataKey] = selectedRow[xLabel];
  293. // [searchedDataKey][0] contains the x value
  294. if (xType === 'numeric') {
  295. series[0][searchedDataKey][0] = selectedRow[xLabel];
  296. } else if (xType === 'time') {
  297. series[0][searchedDataKey][0] = getTimeStamp(selectedRow[xLabel], $('#types_0').val());
  298. } else {
  299. series[0][searchedDataKey][0] = '';
  300. // TODO: text values
  301. }
  302. currentChart.series[0].data = series[0];
  303. // TODO: axis changing
  304. currentChart.replot();
  305. }
  306. if (yChange) {
  307. yCord[searchedDataKey] = selectedRow[yLabel];
  308. // [searchedDataKey][1] contains the y value
  309. if (yType === 'numeric') {
  310. series[0][searchedDataKey][1] = selectedRow[yLabel];
  311. } else if (yType === 'time') {
  312. series[0][searchedDataKey][1] = getTimeStamp(selectedRow[yLabel], $('#types_1').val());
  313. } else {
  314. series[0][searchedDataKey][1] = '';
  315. // TODO: text values
  316. }
  317. currentChart.series[0].data = series[0];
  318. // TODO: axis changing
  319. currentChart.replot();
  320. }
  321. } // End plot update
  322. // Generate SQL query for update
  323. if (!isEmpty(newValues)) {
  324. var sqlQuery = 'UPDATE `' + CommonParams.get('table') + '` SET ';
  325. for (key in newValues) {
  326. sqlQuery += '`' + key + '`=';
  327. var value = newValues[key];
  328. // null
  329. if (value === null) {
  330. sqlQuery += 'NULL, ';
  331. // empty
  332. } else if (value.trim() === '') {
  333. sqlQuery += '\'\', ';
  334. // other
  335. } else {
  336. // type explicitly identified
  337. if (sqlTypes[key] !== null) {
  338. if (sqlTypes[key] === 'bit') {
  339. sqlQuery += 'b\'' + value + '\', ';
  340. }
  341. // type not explicitly identified
  342. } else {
  343. if (!isNumeric(value)) {
  344. sqlQuery += '\'' + value + '\', ';
  345. } else {
  346. sqlQuery += value + ', ';
  347. }
  348. }
  349. }
  350. }
  351. // remove two extraneous characters ', '
  352. sqlQuery = sqlQuery.substring(0, sqlQuery.length - 2);
  353. sqlQuery += ' WHERE ' + Sql.urlDecode(searchedData[searchedDataKey].where_clause);
  354. $.post('index.php?route=/sql', {
  355. 'server': CommonParams.get('server'),
  356. 'db': CommonParams.get('db'),
  357. 'ajax_request': true,
  358. 'sql_query': sqlQuery,
  359. 'inline_edit': false
  360. }, function (data) {
  361. if (typeof data !== 'undefined' && data.success === true) {
  362. $('#sqlqueryresultsouter').html(data.sql_query);
  363. Functions.highlightSql($('#sqlqueryresultsouter'));
  364. } else {
  365. Functions.ajaxShowMessage(data.error, false);
  366. }
  367. }); // End $.post
  368. } // End database update
  369. };
  370. $('#dataPointSaveButton').on('click', function () {
  371. dataPointSave();
  372. });
  373. $('#dataPointModalLabel').first().html(Messages.strDataPointContent);
  374. /**
  375. * Attach Ajax event handlers for input fields
  376. * in the dialog. Used to submit the Ajax
  377. * request when the ENTER key is pressed.
  378. */
  379. $(document).on('keydown', '#dataDisplay :input', function (e) {
  380. if (e.which === 13) {
  381. // 13 is the ENTER key
  382. e.preventDefault();
  383. if (typeof dataPointSave === 'function') {
  384. dataPointSave();
  385. }
  386. }
  387. });
  388. /*
  389. * Generate plot using jqplot
  390. */
  391. if (searchedData !== null) {
  392. $('#zoom_search_form').slideToggle().hide();
  393. $('#togglesearchformlink').text(Messages.strShowSearchCriteria);
  394. $('#togglesearchformdiv').show();
  395. var selectedRow;
  396. var series = [];
  397. var xCord = [];
  398. var yCord = [];
  399. var xVal;
  400. var yVal;
  401. var format;
  402. var options = {
  403. series: [
  404. // for a scatter plot
  405. {
  406. showLine: false
  407. }],
  408. grid: {
  409. drawBorder: false,
  410. shadow: false,
  411. background: 'rgba(0,0,0,0)'
  412. },
  413. axes: {
  414. xaxis: {
  415. label: $('#tableid_0').val(),
  416. labelRenderer: $.jqplot.CanvasAxisLabelRenderer
  417. },
  418. yaxis: {
  419. label: $('#tableid_1').val(),
  420. labelRenderer: $.jqplot.CanvasAxisLabelRenderer
  421. }
  422. },
  423. highlighter: {
  424. show: true,
  425. tooltipAxes: 'y',
  426. yvalues: 2,
  427. // hide the first y value
  428. formatString: '<span class="hide">%s</span>%s'
  429. },
  430. cursor: {
  431. show: true,
  432. zoom: true,
  433. showTooltip: false
  434. }
  435. };
  436. // If data label is not set, do not show tooltips
  437. if (dataLabel === '') {
  438. options.highlighter.show = false;
  439. }
  440. // Classify types as either numeric,time,text
  441. xType = getType(xType);
  442. yType = getType(yType);
  443. // could have multiple series but we'll have just one
  444. series[0] = [];
  445. if (xType === 'time') {
  446. var originalXType = $('#types_0').val();
  447. if (originalXType === 'date') {
  448. format = '%Y-%m-%d';
  449. }
  450. // TODO: does not seem to work
  451. // else if (originalXType === 'time') {
  452. // format = '%H:%M';
  453. // } else {
  454. // format = '%Y-%m-%d %H:%M';
  455. // }
  456. $.extend(options.axes.xaxis, {
  457. renderer: $.jqplot.DateAxisRenderer,
  458. tickOptions: {
  459. formatString: format
  460. }
  461. });
  462. }
  463. if (yType === 'time') {
  464. var originalYType = $('#types_1').val();
  465. if (originalYType === 'date') {
  466. format = '%Y-%m-%d';
  467. }
  468. $.extend(options.axes.yaxis, {
  469. renderer: $.jqplot.DateAxisRenderer,
  470. tickOptions: {
  471. formatString: format
  472. }
  473. });
  474. }
  475. $.each(searchedData, function (key, value) {
  476. if (xType === 'numeric') {
  477. xVal = parseFloat(value[xLabel]);
  478. }
  479. if (xType === 'time') {
  480. xVal = getTimeStamp(value[xLabel], originalXType);
  481. }
  482. if (yType === 'numeric') {
  483. yVal = parseFloat(value[yLabel]);
  484. }
  485. if (yType === 'time') {
  486. yVal = getTimeStamp(value[yLabel], originalYType);
  487. }
  488. series[0].push([xVal, yVal,
  489. // extra Y values
  490. value[dataLabel],
  491. // for highlighter
  492. // (may set an undefined value)
  493. value.where_clause,
  494. // for click on point
  495. key,
  496. // key from searchedData
  497. value.where_clause_sign]);
  498. });
  499. // under IE 8, the initial display is mangled; after a manual
  500. // resizing, it's ok
  501. // under IE 9, everything is fine
  502. currentChart = $.jqplot('querychart', series, options);
  503. currentChart.resetZoom();
  504. $('button.button-reset').on('click', function (event) {
  505. event.preventDefault();
  506. currentChart.resetZoom();
  507. });
  508. $('div#resizer').resizable();
  509. $('div#resizer').on('resizestop', function () {
  510. // make room so that the handle will still appear
  511. $('div#querychart').height($('div#resizer').height() * 0.96);
  512. $('div#querychart').width($('div#resizer').width() * 0.96);
  513. currentChart.replot({
  514. resetAxes: true
  515. });
  516. });
  517. $('div#querychart').on('jqplotDataClick', function (event, seriesIndex, pointIndex, data) {
  518. searchedDataKey = data[4]; // key from searchedData (global)
  519. var fieldId = 0;
  520. var postParams = {
  521. 'ajax_request': true,
  522. 'get_data_row': true,
  523. 'server': CommonParams.get('server'),
  524. 'db': CommonParams.get('db'),
  525. 'table': CommonParams.get('table'),
  526. 'where_clause': data[3],
  527. 'where_clause_sign': data[5]
  528. };
  529. $.post('index.php?route=/table/zoom-search', postParams, function (data) {
  530. // Row is contained in data.row_info,
  531. // now fill the displayResultForm with row values
  532. var key;
  533. for (key in data.row_info) {
  534. var $field = $('#edit_fieldID_' + fieldId);
  535. var $fieldNull = $('#edit_fields_null_id_' + fieldId);
  536. if (data.row_info[key] === null) {
  537. $fieldNull.prop('checked', true);
  538. $field.val('');
  539. } else {
  540. $fieldNull.prop('checked', false);
  541. if ($field.attr('multiple')) {
  542. // when the column is of type SET
  543. $field.val(data.row_info[key].split(','));
  544. } else {
  545. $field.val(data.row_info[key]);
  546. }
  547. }
  548. fieldId++;
  549. }
  550. selectedRow = data.row_info;
  551. });
  552. $('#dataPointModal').modal('show');
  553. });
  554. }
  555. $('#help_dialog').on('click', function () {
  556. displayHelp();
  557. });
  558. });