Plugin/tableview/index.ts

  1. /**
  2. * Copyright (c) 2021 ~ present NAVER Corp.
  3. * billboard.js project is licensed under the MIT license
  4. */
  5. import {loadConfig} from "../../config/config";
  6. import {isNumber, tplProcess} from "../../module/util";
  7. import Plugin from "../Plugin";
  8. import {defaultStyle, tpl} from "./const";
  9. import Options from "./Options";
  10. /**
  11. * Table view plugin.<br>
  12. * Generates table view for bound dataset.
  13. * - **NOTE:**
  14. * - Plugins aren't built-in. Need to be loaded or imported to be used.
  15. * - Non required modules from billboard.js core, need to be installed separately.
  16. * @class plugin-tableview
  17. * @param {object} options table view plugin options
  18. * @augments Plugin
  19. * @returns {TableView}
  20. * @example
  21. * // Plugin must be loaded before the use.
  22. * <script src="$YOUR_PATH/plugin/billboardjs-plugin-tableview.js"></script>
  23. *
  24. * var chart = bb.generate({
  25. * ...
  26. * plugins: [
  27. * new bb.plugin.tableview({
  28. * selector: "#my-table-view",
  29. * categoryTitle: "Category",
  30. * categoryFormat: function(v) {
  31. * // do some transformation
  32. * ...
  33. * return v;
  34. * },
  35. * class: "my-class-name",
  36. * style: true,
  37. * title: "My Data List",
  38. * updateOnToggle: false,
  39. * nullString: "N/A"
  40. * }),
  41. * ]
  42. * });
  43. * @example
  44. * import {bb} from "billboard.js";
  45. * import TableView from "billboard.js/dist/billboardjs-plugin-tableview";
  46. *
  47. * bb.generate({
  48. * ...
  49. * plugins: [
  50. * new TableView({ ... })
  51. * ]
  52. * })
  53. */
  54. export default class TableView extends Plugin {
  55. private config;
  56. private element;
  57. constructor(options) {
  58. super(options);
  59. this.config = new Options();
  60. return this;
  61. }
  62. $beforeInit(): void {
  63. loadConfig.call(this, this.options);
  64. }
  65. $init(): void {
  66. const {class: className, selector, style} = this.config;
  67. let element = document.querySelector(
  68. selector || `.${className || defaultStyle.class}`
  69. );
  70. if (!element) {
  71. const chart = this.$$.$el.chart.node();
  72. element = document.createElement("table");
  73. chart.parentNode.insertBefore(element, chart.nextSibling);
  74. }
  75. if (element.tagName !== "TABLE") {
  76. const table = document.createElement("table");
  77. element.appendChild(table);
  78. element = table;
  79. }
  80. // append default css style
  81. if (style && !document.getElementById(defaultStyle.id)) {
  82. const s = document.createElement("style");
  83. s.id = defaultStyle.id;
  84. s.innerHTML = defaultStyle.rule;
  85. (document.head || document.getElementsByTagName("head")[0])
  86. .appendChild(s);
  87. }
  88. element.classList.add(...[style && defaultStyle.class, className].filter(Boolean));
  89. this.element = element;
  90. }
  91. /**
  92. * Generate table
  93. * @private
  94. */
  95. generateTable(): void {
  96. const {$$, config, element} = this;
  97. const dataToShow = $$.filterTargetsToShow($$.data.targets);
  98. let thead = tplProcess(tpl.thead, {
  99. title: dataToShow.length ? this.config.categoryTitle : ""
  100. });
  101. let tbody = "";
  102. const rows: (number | string)[][] = [];
  103. dataToShow.forEach(v => {
  104. thead += tplProcess(tpl.thead, {title: v.id});
  105. // make up value rows
  106. v.values.forEach((d, i: number) => {
  107. if (!rows[i]) {
  108. rows[i] = [d.x];
  109. }
  110. rows[i].push(d.value);
  111. });
  112. });
  113. rows.forEach(v => {
  114. tbody += `<tr>${
  115. v.map((d, i) =>
  116. tplProcess(i ? tpl.tbody : tpl.tbodyHeader, {
  117. value: i === 0 ?
  118. config.categoryFormat.bind(this)(d) :
  119. (isNumber(d) ? d.toLocaleString() : config.nullString)
  120. })
  121. ).join("")
  122. }</tr>`;
  123. });
  124. const rx = /(<\/?(script|img)[^>]*>|<[^>]+><\/[^>]+>)/ig;
  125. const r = tplProcess(tpl.body, {
  126. ...config,
  127. title: config.title || $$.config.title_text || "",
  128. thead,
  129. tbody
  130. }).replace(rx, "");
  131. element.innerHTML = r;
  132. }
  133. $redraw(): void {
  134. const {state} = this.$$;
  135. const doNotUpdate = state.resizing || (!this.config.updateOnToggle && state.toggling);
  136. !doNotUpdate && this.generateTable();
  137. }
  138. $willDestroy(): void {
  139. this.element.parentNode?.removeChild(this.element);
  140. // remove default css style when left one chart instance
  141. if (this.$$.charts.length === 1) {
  142. const s = document.getElementById(defaultStyle.id);
  143. s?.parentNode?.removeChild(s);
  144. }
  145. }
  146. }