@rfprodz/client-d3-charts 1.3.1 → 1.3.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -208,7 +208,10 @@ const onMouseOver = (self, d, g, config) => {
|
|
|
208
208
|
.style('font-size', '11px')
|
|
209
209
|
.attr('dx', () => (config.width - config.margin.left - config.margin.right) / tooltipShift)
|
|
210
210
|
.attr('dy', () => tooltipDy)
|
|
211
|
-
.text(() => `${d.value} (${new Date(d.timestamp).toUTCString()})`)
|
|
211
|
+
.text(() => `${d.value} (${new Date(d.timestamp).toUTCString()})`)
|
|
212
|
+
.transition()
|
|
213
|
+
.duration(config.transitionDuration)
|
|
214
|
+
.style('opacity', 1);
|
|
212
215
|
};
|
|
213
216
|
/**
|
|
214
217
|
* The mouse out event handler.
|
|
@@ -219,7 +222,11 @@ const onMouseOut = (self, config) => {
|
|
|
219
222
|
const duration = 400;
|
|
220
223
|
d3.select(self).attr('class', 'dot');
|
|
221
224
|
d3.select(self).transition().duration(duration).attr('r', config.dotRadius);
|
|
222
|
-
d3.selectAll('.chart-tooltip')
|
|
225
|
+
d3.selectAll('.chart-tooltip')
|
|
226
|
+
.transition()
|
|
227
|
+
.duration(config.transitionDuration / 2)
|
|
228
|
+
.style('opacity', 0)
|
|
229
|
+
.remove();
|
|
223
230
|
};
|
|
224
231
|
/**
|
|
225
232
|
* Draws the chart lines, dots, and sets the mouse pointer events.
|
|
@@ -303,4 +310,4 @@ export const drawLineChart = (container, data, options) => {
|
|
|
303
310
|
drawLinesDotsAndSetPointerEvents(g, x, y, config, data);
|
|
304
311
|
return config;
|
|
305
312
|
};
|
|
306
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"line-chart.util.js","sourceRoot":"","sources":["../../../../../../libs/client-d3-charts/src/lib/util/line-chart.util.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef } from '@angular/core';\nimport * as d3 from 'd3';\n\nimport { ILineChartDataNode, ILineChartOptions, TLineChartData } from '../interfaces/line-chart.interface';\nimport { generateConfiguration } from './configuration.util';\n\n/**\n * The line chart default configuration.\n */\nexport const defaultLineChartConfig: ILineChartOptions = Object.freeze(<ILineChartOptions>{\n  chartTitle: '',\n  width: 350,\n  height: 350,\n  margin: {\n    top: 70,\n    right: 50,\n    bottom: 50,\n    left: 50,\n  },\n  transitionDuration: 400,\n  dotRadius: 3.5,\n  xAxisTitle: '',\n  yAxisTitle: '',\n  ticks: {\n    x: 5,\n    y: 10,\n  },\n  displayAxisLabels: true,\n  dateFormat: 'default',\n  labelTextWrapWidth: 20, // the number of pixels after which a label needs to be given a new line\n  color: d3.scaleOrdinal(d3.schemeCategory10),\n});\n\n/**\n * Creates a container for the line chart.\n * @param container the chart container\n * @param config the chart configuration\n * @returns the object with the svg element and the g element\n */\nconst createContainer = (container: ElementRef<HTMLDivElement>, config: ILineChartOptions) => {\n  const id = container.nativeElement.id ?? 'line-0';\n\n  d3.select(`#${id}`).select('svg').remove();\n  const svg = d3\n    .select(`#${id}`)\n    .append('svg')\n    .attr('width', config.width + config.margin.left + config.margin.right)\n    .attr('height', config.height + config.margin.top + config.margin.bottom)\n    .attr('class', id);\n  const g = svg.append('g').attr('transform', `translate(${config.margin.left},${config.margin.top / 2})`);\n\n  return { svg, g };\n};\n\n/**\n * Wraps the line chart axis labels text.\n * @param svgText the svg text elements\n * @param width the chart axis label width\n */\nconst wrapSvgText = (svgText: d3.Selection<d3.BaseType, unknown, SVGGElement, unknown>, width: number) => {\n  svgText.each(function (this: d3.BaseType) {\n    const text = d3.select<d3.BaseType, string>(this);\n    const words = text.text().split(/\\s+/).reverse();\n    if (words.length > 1) {\n      let line: string[] = [];\n      let lineNumber = 0;\n      const lineHeight = 1.4;\n      const y = text.attr('y');\n      const x = text.attr('x');\n      const dy = parseFloat(text.attr('dy') ?? 0);\n      let tspan = text.text(null).append('tspan').attr('x', x).attr('y', y).attr('dy', `${dy}em`); // axis label\n\n      let word = words.pop();\n\n      while (typeof word !== 'undefined') {\n        line.push(word ?? '');\n        tspan.text(line.join(' '));\n        if ((tspan.node()?.getComputedTextLength() ?? 0) > width) {\n          line.pop();\n          tspan.text(line.join(' '));\n          line = [word ?? ''];\n          lineNumber += 1;\n          tspan = text\n            .append('tspan')\n            .attr('x', 0)\n            .attr('y', y)\n            .attr('dy', `${lineNumber * lineHeight + dy}em`)\n            .text(word ?? '');\n        }\n        word = words.pop();\n      }\n    }\n  });\n};\n\n/**\n * Creates the legend.\n * @param g the svg g element\n * @param config the chart configuration\n */\nconst createLegend = (g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>, config: ILineChartOptions) => {\n  if (config.displayAxisLabels && config.xAxisTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(0, ${config.height + config.margin.bottom})`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .attr('dx', '1.5em')\n      .attr('dy', '1em')\n      .text(`x - ${config.xAxisTitle}`);\n  }\n\n  if (config.displayAxisLabels && config.yAxisTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(0, ${config.height + config.margin.bottom})`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .attr('dx', '1.5em')\n      .attr('dy', '2.5em')\n      .text(`y - ${config.yAxisTitle}`);\n  }\n\n  if (config.chartTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(0, 0)`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .attr('dx', '1.5em')\n      .attr('dy', '-2em')\n      .text(config.chartTitle);\n  }\n};\n\n/**\n * Creates the x axis.\n * @param g the svg g element\n * @param x the x axis scale\n * @param config the chart configuration\n */\nconst createAxisX = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  x: d3.ScaleTime<number, number>,\n  config: ILineChartOptions,\n) => {\n  const xLabels = g\n    .append('g')\n    .attr('transform', `translate(0, ${config.height})`)\n    .call(\n      d3\n        .axisBottom(x)\n        .ticks(config.ticks.x)\n        .tickFormat(d => {\n          const date = new Date(d.valueOf());\n          const formattingOffset = 10;\n          const day = date.getDate();\n          const dd = day < formattingOffset ? `0${day}` : day;\n          const month = date.getMonth() + 1;\n          const mm = month < formattingOffset ? `0${month}` : month;\n          const year = date.getFullYear().toString();\n          const yy = year.slice(2);\n          const hours = date.getHours();\n          const hour = hours < formattingOffset ? `0${hours}` : hours;\n          const minutes = date.getMinutes();\n          const minute = minutes < formattingOffset ? `0${minutes}` : minutes;\n          let formattedDate = `${dd}/${mm}/${yy} ${hour}:${minute}`;\n          switch (config.dateFormat) {\n            case 'dd/mm/yyyy':\n              formattedDate = `${dd}/${mm}/${year}`;\n              break;\n            case 'dd/mm/yy':\n              formattedDate = `${dd}/${mm}/${yy}`;\n              break;\n            case 'mm/yyyy':\n              formattedDate = `${mm}/${year}`;\n              break;\n            case 'yyyy':\n              formattedDate = `${year}`;\n              break;\n            default:\n              break;\n          }\n          return formattedDate;\n        }),\n    )\n    .append('text');\n\n  g.selectAll('text').call(wrapSvgText, config.labelTextWrapWidth);\n\n  if (config.displayAxisLabels) {\n    xLabels.attr('transform', `translate(${config.width}, 0)`).attr('class', 'legend').attr('dx', '1.5em').attr('dy', '0.7em').text('x');\n  }\n};\n\n/**\n * Creates the y axis.\n * @param g the svg g element\n * @param y the y axis scale\n * @param config the chart configuration\n */\nconst createAxisY = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  y: d3.ScaleLinear<number, number>,\n  config: ILineChartOptions,\n) => {\n  const yLabels = g\n    .append('g')\n    .call(\n      d3\n        .axisLeft(y)\n        .ticks(config.ticks.y)\n        .tickFormat(d => `${d}`),\n    )\n    .append('text');\n\n  if (config.displayAxisLabels) {\n    yLabels.attr('class', 'legend').attr('dy', '-1.5em').attr('class', 'legend').text('y');\n  }\n};\n\n/**\n * The mouse over event handler.\n * @param self an svg circle element\n * @param d the chart data node\n * @param g the svg g element\n * @param config the chart configuration\n */\nconst onMouseOver = (\n  self: SVGCircleElement,\n  d: ILineChartDataNode,\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  config: ILineChartOptions,\n) => {\n  const duration = 400;\n  d3.select(self)\n    .transition()\n    .duration(duration)\n    .attr('r', config.dotRadius * 2);\n\n  const tooltipShift = 4;\n  const tooltipDy = -10;\n  g.append('text')\n    .attr('class', 'chart-tooltip')\n    .style('font-size', '11px')\n    .attr('dx', () => (config.width - config.margin.left - config.margin.right) / tooltipShift)\n    .attr('dy', () => tooltipDy)\n    .text(() => `${d.value} (${new Date(d.timestamp).toUTCString()})`);\n};\n\n/**\n * The mouse out event handler.\n * @param self an svg circle element\n * @param config the chart configuration\n */\nconst onMouseOut = (self: SVGCircleElement, config: ILineChartOptions) => {\n  const duration = 400;\n  d3.select(self).attr('class', 'dot');\n  d3.select(self).transition().duration(duration).attr('r', config.dotRadius);\n\n  d3.selectAll('.chart-tooltip').remove();\n};\n\n/**\n * Draws the chart lines, dots, and sets the mouse pointer events.\n * @param g the svg g element\n * @param x the x axis scale\n * @param y the y axis scale\n * @param config the chart configuration\n * @param data the chart data\n */\nconst drawLinesDotsAndSetPointerEvents = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  x: d3.ScaleTime<number, number>,\n  y: d3.ScaleLinear<number, number>,\n  config: ILineChartOptions,\n  data: TLineChartData[],\n) => {\n  const line = d3\n    .line<ILineChartDataNode>()\n    .x(d => x(d.timestamp))\n    .y(d => y(d.value))\n    .curve(d3.curveMonotoneX);\n\n  const flatData = data.flat();\n\n  for (let c = 0, maxC = data.length; c < maxC; c += 1) {\n    const chunk = data[c];\n\n    g.append('path')\n      .attr('id', `line-${c}`)\n      .style('fill', 'none')\n      .style('stroke', config.color(c.toString()))\n      .style('stroke-width', '2px')\n      .attr('d', line(chunk));\n  }\n\n  g.selectAll('.dot')\n    .data(flatData)\n    .enter()\n    .append('circle')\n    .attr('class', 'dot')\n    .style('pointer-events', 'all')\n    .style('fill', (d, i) => config.color(i.toString()))\n    .on('mouseover', function (this, event, d) {\n      return onMouseOver(this, d, g, config);\n    })\n    .on('mouseout', function (this) {\n      return onMouseOut(this, config);\n    })\n    .attr('cx', function (this, d) {\n      return x(d.timestamp);\n    })\n    .attr('cy', function (this, d) {\n      return y(d.value);\n    })\n    .attr('r', 0)\n    .transition()\n    .ease(d3.easeLinear)\n    .duration(config.transitionDuration)\n    .delay((d, i) => {\n      const multiplier = 50;\n      return i * multiplier;\n    })\n    .attr('r', config.dotRadius);\n};\n\n/**\n * Draws the line chart.\n * @param container the chart container\n * @param data the chart data\n * @param options the chart options\n * @returns the chart configuration\n */\nexport const drawLineChart = (container: ElementRef<HTMLDivElement>, data: TLineChartData[], options?: Partial<ILineChartOptions>) => {\n  const config: ILineChartOptions = generateConfiguration<ILineChartOptions>(defaultLineChartConfig, options, {});\n\n  const { g } = createContainer(container, config);\n\n  const range = data.reduce(\n    (accumulator: { minTime: number; maxTime: number; maxValue: number }, arr) => {\n      const timestamps = arr.map(item => item.timestamp);\n      const minItem = Math.min(...timestamps);\n      const maxItem = Math.max(...timestamps);\n      const minTime = Math.min(minItem, accumulator.minTime);\n      const maxTime = Math.max(maxItem, accumulator.maxTime);\n\n      const values = arr.map(item => item.value);\n      const maxItemValue = Math.max(...values);\n      const maxValue = Math.max(maxItemValue, accumulator.maxValue);\n      return { minTime, maxTime, maxValue };\n    },\n    { minTime: Number(Infinity), maxTime: -Infinity, maxValue: -Infinity },\n  );\n\n  const x = d3.scaleTime([0, config.width]).domain([range.minTime, range.maxTime]);\n  const y = d3.scaleLinear([config.height, 0]).domain([0, range.maxValue ?? 1]);\n\n  createAxisX(g, x, config);\n\n  createAxisY(g, y, config);\n\n  createLegend(g, config);\n\n  drawLinesDotsAndSetPointerEvents(g, x, y, config, data);\n\n  return config;\n};\n"]}
|
|
313
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"line-chart.util.js","sourceRoot":"","sources":["../../../../../../libs/client-d3-charts/src/lib/util/line-chart.util.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef } from '@angular/core';\nimport * as d3 from 'd3';\n\nimport { ILineChartDataNode, ILineChartOptions, TLineChartData } from '../interfaces/line-chart.interface';\nimport { generateConfiguration } from './configuration.util';\n\n/**\n * The line chart default configuration.\n */\nexport const defaultLineChartConfig: ILineChartOptions = Object.freeze(<ILineChartOptions>{\n  chartTitle: '',\n  width: 350,\n  height: 350,\n  margin: {\n    top: 70,\n    right: 50,\n    bottom: 50,\n    left: 50,\n  },\n  transitionDuration: 400,\n  dotRadius: 3.5,\n  xAxisTitle: '',\n  yAxisTitle: '',\n  ticks: {\n    x: 5,\n    y: 10,\n  },\n  displayAxisLabels: true,\n  dateFormat: 'default',\n  labelTextWrapWidth: 20, // the number of pixels after which a label needs to be given a new line\n  color: d3.scaleOrdinal(d3.schemeCategory10),\n});\n\n/**\n * Creates a container for the line chart.\n * @param container the chart container\n * @param config the chart configuration\n * @returns the object with the svg element and the g element\n */\nconst createContainer = (container: ElementRef<HTMLDivElement>, config: ILineChartOptions) => {\n  const id = container.nativeElement.id ?? 'line-0';\n\n  d3.select(`#${id}`).select('svg').remove();\n  const svg = d3\n    .select(`#${id}`)\n    .append('svg')\n    .attr('width', config.width + config.margin.left + config.margin.right)\n    .attr('height', config.height + config.margin.top + config.margin.bottom)\n    .attr('class', id);\n  const g = svg.append('g').attr('transform', `translate(${config.margin.left},${config.margin.top / 2})`);\n\n  return { svg, g };\n};\n\n/**\n * Wraps the line chart axis labels text.\n * @param svgText the svg text elements\n * @param width the chart axis label width\n */\nconst wrapSvgText = (svgText: d3.Selection<d3.BaseType, unknown, SVGGElement, unknown>, width: number) => {\n  svgText.each(function (this: d3.BaseType) {\n    const text = d3.select<d3.BaseType, string>(this);\n    const words = text.text().split(/\\s+/).reverse();\n    if (words.length > 1) {\n      let line: string[] = [];\n      let lineNumber = 0;\n      const lineHeight = 1.4;\n      const y = text.attr('y');\n      const x = text.attr('x');\n      const dy = parseFloat(text.attr('dy') ?? 0);\n      let tspan = text.text(null).append('tspan').attr('x', x).attr('y', y).attr('dy', `${dy}em`); // axis label\n\n      let word = words.pop();\n\n      while (typeof word !== 'undefined') {\n        line.push(word ?? '');\n        tspan.text(line.join(' '));\n        if ((tspan.node()?.getComputedTextLength() ?? 0) > width) {\n          line.pop();\n          tspan.text(line.join(' '));\n          line = [word ?? ''];\n          lineNumber += 1;\n          tspan = text\n            .append('tspan')\n            .attr('x', 0)\n            .attr('y', y)\n            .attr('dy', `${lineNumber * lineHeight + dy}em`)\n            .text(word ?? '');\n        }\n        word = words.pop();\n      }\n    }\n  });\n};\n\n/**\n * Creates the legend.\n * @param g the svg g element\n * @param config the chart configuration\n */\nconst createLegend = (g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>, config: ILineChartOptions) => {\n  if (config.displayAxisLabels && config.xAxisTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(0, ${config.height + config.margin.bottom})`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .attr('dx', '1.5em')\n      .attr('dy', '1em')\n      .text(`x - ${config.xAxisTitle}`);\n  }\n\n  if (config.displayAxisLabels && config.yAxisTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(0, ${config.height + config.margin.bottom})`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .attr('dx', '1.5em')\n      .attr('dy', '2.5em')\n      .text(`y - ${config.yAxisTitle}`);\n  }\n\n  if (config.chartTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(0, 0)`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .attr('dx', '1.5em')\n      .attr('dy', '-2em')\n      .text(config.chartTitle);\n  }\n};\n\n/**\n * Creates the x axis.\n * @param g the svg g element\n * @param x the x axis scale\n * @param config the chart configuration\n */\nconst createAxisX = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  x: d3.ScaleTime<number, number>,\n  config: ILineChartOptions,\n) => {\n  const xLabels = g\n    .append('g')\n    .attr('transform', `translate(0, ${config.height})`)\n    .call(\n      d3\n        .axisBottom(x)\n        .ticks(config.ticks.x)\n        .tickFormat(d => {\n          const date = new Date(d.valueOf());\n          const formattingOffset = 10;\n          const day = date.getDate();\n          const dd = day < formattingOffset ? `0${day}` : day;\n          const month = date.getMonth() + 1;\n          const mm = month < formattingOffset ? `0${month}` : month;\n          const year = date.getFullYear().toString();\n          const yy = year.slice(2);\n          const hours = date.getHours();\n          const hour = hours < formattingOffset ? `0${hours}` : hours;\n          const minutes = date.getMinutes();\n          const minute = minutes < formattingOffset ? `0${minutes}` : minutes;\n          let formattedDate = `${dd}/${mm}/${yy} ${hour}:${minute}`;\n          switch (config.dateFormat) {\n            case 'dd/mm/yyyy':\n              formattedDate = `${dd}/${mm}/${year}`;\n              break;\n            case 'dd/mm/yy':\n              formattedDate = `${dd}/${mm}/${yy}`;\n              break;\n            case 'mm/yyyy':\n              formattedDate = `${mm}/${year}`;\n              break;\n            case 'yyyy':\n              formattedDate = `${year}`;\n              break;\n            default:\n              break;\n          }\n          return formattedDate;\n        }),\n    )\n    .append('text');\n\n  g.selectAll('text').call(wrapSvgText, config.labelTextWrapWidth);\n\n  if (config.displayAxisLabels) {\n    xLabels.attr('transform', `translate(${config.width}, 0)`).attr('class', 'legend').attr('dx', '1.5em').attr('dy', '0.7em').text('x');\n  }\n};\n\n/**\n * Creates the y axis.\n * @param g the svg g element\n * @param y the y axis scale\n * @param config the chart configuration\n */\nconst createAxisY = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  y: d3.ScaleLinear<number, number>,\n  config: ILineChartOptions,\n) => {\n  const yLabels = g\n    .append('g')\n    .call(\n      d3\n        .axisLeft(y)\n        .ticks(config.ticks.y)\n        .tickFormat(d => `${d}`),\n    )\n    .append('text');\n\n  if (config.displayAxisLabels) {\n    yLabels.attr('class', 'legend').attr('dy', '-1.5em').attr('class', 'legend').text('y');\n  }\n};\n\n/**\n * The mouse over event handler.\n * @param self an svg circle element\n * @param d the chart data node\n * @param g the svg g element\n * @param config the chart configuration\n */\nconst onMouseOver = (\n  self: SVGCircleElement,\n  d: ILineChartDataNode,\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  config: ILineChartOptions,\n) => {\n  const duration = 400;\n  d3.select(self)\n    .transition()\n    .duration(duration)\n    .attr('r', config.dotRadius * 2);\n\n  const tooltipShift = 4;\n  const tooltipDy = -10;\n  g.append('text')\n    .attr('class', 'chart-tooltip')\n    .style('font-size', '11px')\n    .attr('dx', () => (config.width - config.margin.left - config.margin.right) / tooltipShift)\n    .attr('dy', () => tooltipDy)\n    .text(() => `${d.value} (${new Date(d.timestamp).toUTCString()})`)\n    .transition()\n    .duration(config.transitionDuration)\n    .style('opacity', 1);\n};\n\n/**\n * The mouse out event handler.\n * @param self an svg circle element\n * @param config the chart configuration\n */\nconst onMouseOut = (self: SVGCircleElement, config: ILineChartOptions) => {\n  const duration = 400;\n  d3.select(self).attr('class', 'dot');\n  d3.select(self).transition().duration(duration).attr('r', config.dotRadius);\n  d3.selectAll('.chart-tooltip')\n    .transition()\n    .duration(config.transitionDuration / 2)\n    .style('opacity', 0)\n    .remove();\n};\n\n/**\n * Draws the chart lines, dots, and sets the mouse pointer events.\n * @param g the svg g element\n * @param x the x axis scale\n * @param y the y axis scale\n * @param config the chart configuration\n * @param data the chart data\n */\nconst drawLinesDotsAndSetPointerEvents = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  x: d3.ScaleTime<number, number>,\n  y: d3.ScaleLinear<number, number>,\n  config: ILineChartOptions,\n  data: TLineChartData[],\n) => {\n  const line = d3\n    .line<ILineChartDataNode>()\n    .x(d => x(d.timestamp))\n    .y(d => y(d.value))\n    .curve(d3.curveMonotoneX);\n\n  const flatData = data.flat();\n\n  for (let c = 0, maxC = data.length; c < maxC; c += 1) {\n    const chunk = data[c];\n\n    g.append('path')\n      .attr('id', `line-${c}`)\n      .style('fill', 'none')\n      .style('stroke', config.color(c.toString()))\n      .style('stroke-width', '2px')\n      .attr('d', line(chunk));\n  }\n\n  g.selectAll('.dot')\n    .data(flatData)\n    .enter()\n    .append('circle')\n    .attr('class', 'dot')\n    .style('pointer-events', 'all')\n    .style('fill', (d, i) => config.color(i.toString()))\n    .on('mouseover', function (this, event, d) {\n      return onMouseOver(this, d, g, config);\n    })\n    .on('mouseout', function (this) {\n      return onMouseOut(this, config);\n    })\n    .attr('cx', function (this, d) {\n      return x(d.timestamp);\n    })\n    .attr('cy', function (this, d) {\n      return y(d.value);\n    })\n    .attr('r', 0)\n    .transition()\n    .ease(d3.easeLinear)\n    .duration(config.transitionDuration)\n    .delay((d, i) => {\n      const multiplier = 50;\n      return i * multiplier;\n    })\n    .attr('r', config.dotRadius);\n};\n\n/**\n * Draws the line chart.\n * @param container the chart container\n * @param data the chart data\n * @param options the chart options\n * @returns the chart configuration\n */\nexport const drawLineChart = (container: ElementRef<HTMLDivElement>, data: TLineChartData[], options?: Partial<ILineChartOptions>) => {\n  const config: ILineChartOptions = generateConfiguration<ILineChartOptions>(defaultLineChartConfig, options, {});\n\n  const { g } = createContainer(container, config);\n\n  const range = data.reduce(\n    (accumulator: { minTime: number; maxTime: number; maxValue: number }, arr) => {\n      const timestamps = arr.map(item => item.timestamp);\n      const minItem = Math.min(...timestamps);\n      const maxItem = Math.max(...timestamps);\n      const minTime = Math.min(minItem, accumulator.minTime);\n      const maxTime = Math.max(maxItem, accumulator.maxTime);\n\n      const values = arr.map(item => item.value);\n      const maxItemValue = Math.max(...values);\n      const maxValue = Math.max(maxItemValue, accumulator.maxValue);\n      return { minTime, maxTime, maxValue };\n    },\n    { minTime: Number(Infinity), maxTime: -Infinity, maxValue: -Infinity },\n  );\n\n  const x = d3.scaleTime([0, config.width]).domain([range.minTime, range.maxTime]);\n  const y = d3.scaleLinear([config.height, 0]).domain([0, range.maxValue ?? 1]);\n\n  createAxisX(g, x, config);\n\n  createAxisY(g, y, config);\n\n  createLegend(g, config);\n\n  drawLinesDotsAndSetPointerEvents(g, x, y, config, data);\n\n  return config;\n};\n"]}
|
|
@@ -53,7 +53,6 @@ export const drawPieChart = (container, data, options) => {
|
|
|
53
53
|
const pie = d3.pie().value(datum => datum.y);
|
|
54
54
|
const radius = Math.min(config.width, config.height) / 2;
|
|
55
55
|
const arc = d3.arc().innerRadius(config.innerRadius).outerRadius(radius);
|
|
56
|
-
const tooltip = g.append('text').attr('class', 'chart-tooltip').style('opacity', 0);
|
|
57
56
|
const arcs = g
|
|
58
57
|
.selectAll('arc')
|
|
59
58
|
.data(pie(data))
|
|
@@ -62,13 +61,10 @@ export const drawPieChart = (container, data, options) => {
|
|
|
62
61
|
.attr('class', 'arc')
|
|
63
62
|
.on('mouseover', function (event, d) {
|
|
64
63
|
this.style.opacity = '0.8';
|
|
65
|
-
const modifier = 10;
|
|
66
|
-
const x = parseFloat(d3.select(this).attr('cx')) - modifier;
|
|
67
|
-
const y = parseFloat(d3.select(this).attr('cy')) - modifier;
|
|
68
64
|
const tooltipText = `${d.data.key}: ${d.data.y}`;
|
|
69
|
-
|
|
70
|
-
.attr('
|
|
71
|
-
.
|
|
65
|
+
g.append('text')
|
|
66
|
+
.attr('class', 'chart-tooltip')
|
|
67
|
+
.style('opacity', 0)
|
|
72
68
|
.attr('dx', -config.width / (2 * 2 * 2))
|
|
73
69
|
.attr('dy', config.height / 2 + config.margin.top)
|
|
74
70
|
.text(tooltipText)
|
|
@@ -78,7 +74,11 @@ export const drawPieChart = (container, data, options) => {
|
|
|
78
74
|
})
|
|
79
75
|
.on('mouseout', function (event, d) {
|
|
80
76
|
this.style.opacity = 'unset';
|
|
81
|
-
|
|
77
|
+
d3.selectAll('.chart-tooltip')
|
|
78
|
+
.transition()
|
|
79
|
+
.duration(config.transitionDuration / 2)
|
|
80
|
+
.style('opacity', 0)
|
|
81
|
+
.remove();
|
|
82
82
|
});
|
|
83
83
|
arcs
|
|
84
84
|
.append('path')
|
|
@@ -101,4 +101,4 @@ export const drawPieChart = (container, data, options) => {
|
|
|
101
101
|
}
|
|
102
102
|
return config;
|
|
103
103
|
};
|
|
104
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pie-chart.util.js","sourceRoot":"","sources":["../../../../../../libs/client-d3-charts/src/lib/util/pie-chart.util.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,MAAM,IAAI,CAAC;AAGzB,OAAO,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAE7D;;GAEG;AACH,MAAM,CAAC,MAAM,qBAAqB,GAAqB,MAAM,CAAC,MAAM,CAAmB;IACrF,UAAU,EAAE,EAAE;IACd,KAAK,EAAE,GAAG;IACV,MAAM,EAAE,GAAG;IACX,MAAM,EAAE;QACN,GAAG,EAAE,EAAE;QACP,KAAK,EAAE,EAAE;QACT,MAAM,EAAE,EAAE;QACV,IAAI,EAAE,EAAE;KACT;IACD,WAAW,EAAE,CAAC;IACd,UAAU,EAAE,IAAI;IAChB,mBAAmB,EAAE,EAAE;IACvB,kBAAkB,EAAE,EAAE;IACtB,kBAAkB,EAAE,IAAI;IACxB,KAAK,EAAE,EAAE,CAAC,YAAY,CAAC,EAAE,CAAC,gBAAgB,CAAC;CAC5C,CAAC,CAAC;AAEH;;;;;GAKG;AACH,MAAM,eAAe,GAAG,CAAC,SAAqC,EAAE,MAAwB,EAAE,EAAE;IAC1F,MAAM,EAAE,GAAG,SAAS,CAAC,aAAa,CAAC,EAAE,IAAI,OAAO,CAAC;IAEjD,EAAE,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC;IAC3C,MAAM,GAAG,GAAG,EAAE;SACX,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC;SAChB,MAAM,CAAC,KAAK,CAAC;SACb,IAAI,CAAC,OAAO,EAAE,MAAM,CAAC,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC;SACtE,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,GAAG,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC;SACxE,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC;IACrB,MAAM,CAAC,GAAG,GAAG;SACV,MAAM,CAAC,GAAG,CAAC;SACX,IAAI,CAAC,WAAW,EAAE,aAAa,MAAM,CAAC,KAAK,GAAG,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,GAAG,GAAG,CAAC,CAAC;IAErH,OAAO,EAAE,GAAG,EAAE,CAAC,EAAE,CAAC;AACpB,CAAC,CAAC;AAEF;;;;;;GAMG;AACH,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,SAAqC,EAAE,IAAyB,EAAE,OAAmC,EAAE,EAAE;IACpI,MAAM,MAAM,GAAqB,qBAAqB,CAAmB,qBAAqB,EAAE,OAAO,EAAE,EAAE,CAAC,CAAC;IAE7G,MAAM,EAAE,CAAC,EAAE,GAAG,eAAe,CAAC,SAAS,EAAE,MAAM,CAAC,CAAC;IAEjD,MAAM,GAAG,GAAG,EAAE,CAAC,GAAG,EAAqB,CAAC,KAAK,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAEhE,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAEzD,MAAM,GAAG,GAAG,EAAE,CAAC,GAAG,EAAqC,CAAC,WAAW,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;IAE5G,MAAM,OAAO,GAAG,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,eAAe,CAAC,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;IAEpF,MAAM,IAAI,GAAG,CAAC;SACX,SAAS,CAAC,KAAK,CAAC;SAChB,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;SACf,KAAK,EAAE;SACP,MAAM,CAAC,GAAG,CAAC;SACX,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC;SACpB,EAAE,CAAC,WAAW,EAAE,UAAgB,KAAiB,EAAE,CAAC;QACnD,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,KAAK,CAAC;QAE3B,MAAM,QAAQ,GAAG,EAAE,CAAC;QACpB,MAAM,CAAC,GAAG,UAAU,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,QAAQ,CAAC;QAC5D,MAAM,CAAC,GAAG,UAAU,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,QAAQ,CAAC;QAE5D,MAAM,WAAW,GAAG,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC;QAEjD,OAAO;aACJ,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;aACZ,IAAI,CAAC,GAAG,EAAE,CAAC,CAAC;aACZ,IAAI,CAAC,IAAI,EAAE,CAAC,MAAM,CAAC,KAAK,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC;aACvC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,MAAM,GAAG,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC;aACjD,IAAI,CAAC,WAAW,CAAC;aACjB,UAAU,EAAE;aACZ,QAAQ,CAAC,MAAM,CAAC,kBAAkB,CAAC;aACnC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;IACzB,CAAC,CAAC;SACD,EAAE,CAAC,UAAU,EAAE,UAAgB,KAAK,EAAE,CAAC;QACtC,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,OAAO,CAAC;QAC7B,OAAO,CAAC,UAAU,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,KAAK,CAAC,SAAS,EAAE,CAAC,CAAC,CAAC;IAC/E,CAAC,CAAC,CAAC;IAEL,IAAI;SACD,MAAM,CAAC,MAAM,CAAC;SACd,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC;SAClD,IAAI,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;IAElB,IAAI,MAAM,CAAC,UAAU,EAAE;QACrB,MAAM,KAAK,GAAG,EAAE;aACb,GAAG,EAAqC;aACxC,WAAW,CAAC,MAAM,CAAC;aACnB,WAAW,CAAC,MAAM,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QAEpD,MAAM,MAAM,GAAG,CAAC,CAAC;QACjB,IAAI;aACD,MAAM,CAAC,MAAM,CAAC;aACd,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC;aACvB,IAAI,CAAC,aAAa,EAAE,QAAQ,CAAC;aAC7B,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC;aAClB,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,EAAE,CAAC,aAAa,KAAK,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC;aACzD,KAAK,CAAC,WAAW,EAAE,MAAM,CAAC;aAC1B,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;KACxB;IAED,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC","sourcesContent":["import { ElementRef } from '@angular/core';\nimport * as d3 from 'd3';\n\nimport { IPieChartDataNode, IPieChartOptions } from '../interfaces/pie-chart.interface';\nimport { generateConfiguration } from './configuration.util';\n\n/**\n * The pie chart default configuration.\n */\nexport const defaultPieChartConfig: IPieChartOptions = Object.freeze(<IPieChartOptions>{\n  chartTitle: '',\n  width: 600,\n  height: 600,\n  margin: {\n    top: 20,\n    right: 20,\n    bottom: 20,\n    left: 20,\n  },\n  innerRadius: 0, // increase inner radius to render a donut chart\n  showLabels: true,\n  labelRadiusModifier: 50,\n  labelTextWrapWidth: 60,\n  transitionDuration: 1000,\n  color: d3.scaleOrdinal(d3.schemeCategory10),\n});\n\n/**\n * Creates a container for the pie chart.\n * @param container the chart container\n * @param config the chart configuration\n * @returns the object with the svg element and the g element\n */\nconst createContainer = (container: ElementRef<HTMLDivElement>, config: IPieChartOptions) => {\n  const id = container.nativeElement.id ?? 'pie-0';\n\n  d3.select(`#${id}`).select('svg').remove();\n  const svg = d3\n    .select(`#${id}`)\n    .append('svg')\n    .attr('width', config.width + config.margin.left + config.margin.right)\n    .attr('height', config.height + config.margin.top + config.margin.bottom)\n    .attr('class', id);\n  const g = svg\n    .append('g')\n    .attr('transform', `translate(${config.width / 2 + config.margin.left},${config.height / 2 + config.margin.top})`);\n\n  return { svg, g };\n};\n\n/**\n * Draws the pie chart.\n * @param container the chart container\n * @param data the chart data\n * @param options the chart options\n * @returns the chart configuration\n */\nexport const drawPieChart = (container: ElementRef<HTMLDivElement>, data: IPieChartDataNode[], options?: Partial<IPieChartOptions>) => {\n  const config: IPieChartOptions = generateConfiguration<IPieChartOptions>(defaultPieChartConfig, options, {});\n\n  const { g } = createContainer(container, config);\n\n  const pie = d3.pie<IPieChartDataNode>().value(datum => datum.y);\n\n  const radius = Math.min(config.width, config.height) / 2;\n\n  const arc = d3.arc<d3.PieArcDatum<IPieChartDataNode>>().innerRadius(config.innerRadius).outerRadius(radius);\n\n  const tooltip = g.append('text').attr('class', 'chart-tooltip').style('opacity', 0);\n\n  const arcs = g\n    .selectAll('arc')\n    .data(pie(data))\n    .enter()\n    .append('g')\n    .attr('class', 'arc')\n    .on('mouseover', function (this, event: MouseEvent, d) {\n      this.style.opacity = '0.8';\n\n      const modifier = 10;\n      const x = parseFloat(d3.select(this).attr('cx')) - modifier;\n      const y = parseFloat(d3.select(this).attr('cy')) - modifier;\n\n      const tooltipText = `${d.data.key}: ${d.data.y}`;\n\n      tooltip\n        .attr('x', x)\n        .attr('y', y)\n        .attr('dx', -config.width / (2 * 2 * 2))\n        .attr('dy', config.height / 2 + config.margin.top)\n        .text(tooltipText)\n        .transition()\n        .duration(config.transitionDuration)\n        .style('opacity', 1);\n    })\n    .on('mouseout', function (this, event, d) {\n      this.style.opacity = 'unset';\n      tooltip.transition().duration(config.transitionDuration).style('opacity', 0);\n    });\n\n  arcs\n    .append('path')\n    .attr('fill', (d, i) => config.color(i.toString()))\n    .attr('d', arc);\n\n  if (config.showLabels) {\n    const label = d3\n      .arc<d3.PieArcDatum<IPieChartDataNode>>()\n      .innerRadius(radius)\n      .outerRadius(radius + config.labelRadiusModifier);\n\n    const textDy = 5;\n    arcs\n      .append('text')\n      .attr('class', 'legend')\n      .attr('text-anchor', 'middle')\n      .attr('dy', textDy)\n      .attr('transform', d => `translate(${label.centroid(d)})`)\n      .style('font-size', '12px')\n      .text(d => d.data.y);\n  }\n\n  return config;\n};\n"]}
|
|
104
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pie-chart.util.js","sourceRoot":"","sources":["../../../../../../libs/client-d3-charts/src/lib/util/pie-chart.util.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef } from '@angular/core';\nimport * as d3 from 'd3';\n\nimport { IPieChartDataNode, IPieChartOptions } from '../interfaces/pie-chart.interface';\nimport { generateConfiguration } from './configuration.util';\n\n/**\n * The pie chart default configuration.\n */\nexport const defaultPieChartConfig: IPieChartOptions = Object.freeze(<IPieChartOptions>{\n  chartTitle: '',\n  width: 600,\n  height: 600,\n  margin: {\n    top: 20,\n    right: 20,\n    bottom: 20,\n    left: 20,\n  },\n  innerRadius: 0, // increase inner radius to render a donut chart\n  showLabels: true,\n  labelRadiusModifier: 50,\n  labelTextWrapWidth: 60,\n  transitionDuration: 1000,\n  color: d3.scaleOrdinal(d3.schemeCategory10),\n});\n\n/**\n * Creates a container for the pie chart.\n * @param container the chart container\n * @param config the chart configuration\n * @returns the object with the svg element and the g element\n */\nconst createContainer = (container: ElementRef<HTMLDivElement>, config: IPieChartOptions) => {\n  const id = container.nativeElement.id ?? 'pie-0';\n\n  d3.select(`#${id}`).select('svg').remove();\n  const svg = d3\n    .select(`#${id}`)\n    .append('svg')\n    .attr('width', config.width + config.margin.left + config.margin.right)\n    .attr('height', config.height + config.margin.top + config.margin.bottom)\n    .attr('class', id);\n  const g = svg\n    .append('g')\n    .attr('transform', `translate(${config.width / 2 + config.margin.left},${config.height / 2 + config.margin.top})`);\n\n  return { svg, g };\n};\n\n/**\n * Draws the pie chart.\n * @param container the chart container\n * @param data the chart data\n * @param options the chart options\n * @returns the chart configuration\n */\nexport const drawPieChart = (container: ElementRef<HTMLDivElement>, data: IPieChartDataNode[], options?: Partial<IPieChartOptions>) => {\n  const config: IPieChartOptions = generateConfiguration<IPieChartOptions>(defaultPieChartConfig, options, {});\n\n  const { g } = createContainer(container, config);\n\n  const pie = d3.pie<IPieChartDataNode>().value(datum => datum.y);\n\n  const radius = Math.min(config.width, config.height) / 2;\n\n  const arc = d3.arc<d3.PieArcDatum<IPieChartDataNode>>().innerRadius(config.innerRadius).outerRadius(radius);\n\n  const arcs = g\n    .selectAll('arc')\n    .data(pie(data))\n    .enter()\n    .append('g')\n    .attr('class', 'arc')\n    .on('mouseover', function (this, event: MouseEvent, d) {\n      this.style.opacity = '0.8';\n\n      const tooltipText = `${d.data.key}: ${d.data.y}`;\n\n      g.append('text')\n        .attr('class', 'chart-tooltip')\n        .style('opacity', 0)\n        .attr('dx', -config.width / (2 * 2 * 2))\n        .attr('dy', config.height / 2 + config.margin.top)\n        .text(tooltipText)\n        .transition()\n        .duration(config.transitionDuration)\n        .style('opacity', 1);\n    })\n    .on('mouseout', function (this, event, d) {\n      this.style.opacity = 'unset';\n      d3.selectAll('.chart-tooltip')\n        .transition()\n        .duration(config.transitionDuration / 2)\n        .style('opacity', 0)\n        .remove();\n    });\n\n  arcs\n    .append('path')\n    .attr('fill', (d, i) => config.color(i.toString()))\n    .attr('d', arc);\n\n  if (config.showLabels) {\n    const label = d3\n      .arc<d3.PieArcDatum<IPieChartDataNode>>()\n      .innerRadius(radius)\n      .outerRadius(radius + config.labelRadiusModifier);\n\n    const textDy = 5;\n    arcs\n      .append('text')\n      .attr('class', 'legend')\n      .attr('text-anchor', 'middle')\n      .attr('dy', textDy)\n      .attr('transform', d => `translate(${label.centroid(d)})`)\n      .style('font-size', '12px')\n      .text(d => d.data.y);\n  }\n\n  return config;\n};\n"]}
|
|
@@ -237,8 +237,6 @@ const drawRadarChartBlobs = (radiusScale, angleSlice, g, data, config) => {
|
|
|
237
237
|
const appendInvisibleTooltipCircles = (g, data, radiusScale, angleSlice, config) => {
|
|
238
238
|
// wrapper for the invisible circles on top
|
|
239
239
|
const blobCircleWrapper = g.selectAll('.radar-circle-wrapper').data(data).enter().append('g').attr('class', 'radar-circle-wrapper');
|
|
240
|
-
// set up the small tooltip for when you hover over a circle
|
|
241
|
-
const tooltip = g.append('text').attr('class', 'chart-tooltip').style('opacity', 0);
|
|
242
240
|
// append a set of invisible circles on top for the mouseover pop-up
|
|
243
241
|
const blobCircleWrapperRadiusMultiplier = 1.5;
|
|
244
242
|
blobCircleWrapper
|
|
@@ -258,10 +256,22 @@ const appendInvisibleTooltipCircles = (g, data, radiusScale, angleSlice, config)
|
|
|
258
256
|
const newY = parseFloat(d3.select(this).attr('cy')) - modifier;
|
|
259
257
|
const nodeData = event.target['__data__'];
|
|
260
258
|
const tooltipText = `${nodeData.value} ${nodeData.unit}`;
|
|
261
|
-
|
|
259
|
+
g.append('text')
|
|
260
|
+
.attr('class', 'chart-tooltip')
|
|
261
|
+
.style('opacity', 0)
|
|
262
|
+
.attr('x', newX)
|
|
263
|
+
.attr('y', newY)
|
|
264
|
+
.text(tooltipText)
|
|
265
|
+
.transition()
|
|
266
|
+
.duration(config.transitionDuration)
|
|
267
|
+
.style('opacity', 1);
|
|
262
268
|
})
|
|
263
269
|
.on('mouseout', () => {
|
|
264
|
-
|
|
270
|
+
d3.selectAll('.chart-tooltip')
|
|
271
|
+
.transition()
|
|
272
|
+
.duration(config.transitionDuration / 2)
|
|
273
|
+
.style('opacity', 0)
|
|
274
|
+
.remove();
|
|
265
275
|
});
|
|
266
276
|
};
|
|
267
277
|
/**
|
|
@@ -294,4 +304,4 @@ export const drawRadarChart = (container, data, options) => {
|
|
|
294
304
|
appendInvisibleTooltipCircles(g, data, radiusScale, angleSlice, config);
|
|
295
305
|
return config;
|
|
296
306
|
};
|
|
297
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radar-chart.util.js","sourceRoot":"","sources":["../../../../../../libs/client-d3-charts/src/lib/util/radar-chart.util.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef } from '@angular/core';\nimport * as d3 from 'd3';\n\nimport { IRadarChartDataNode, IRadarChartOptions, TRadarChartData } from '../interfaces/radar-chart.interface';\nimport { generateConfiguration } from './configuration.util';\n\n/**\n * The radar chart default configuration.\n */\nexport const defaultRadarChartConfig: IRadarChartOptions = Object.freeze(<IRadarChartOptions>{\n  chartTitle: '',\n  width: 350,\n  height: 350,\n  margin: {\n    top: 50,\n    right: 50,\n    bottom: 50,\n    left: 50,\n  },\n  levels: 3, // how many levels or inner circles should there be drawn\n  maxValue: 0, // what is the value that the biggest circle will represent\n  lineFactor: 1.1, // how much farther than the radius of the outer circle should the lines be stretched\n  labelFactor: 1.15, // how much farther than the radius of the outer circle should the labels be placed\n  labelTextWrapWidth: 60, // the number of pixels after which a label needs to be given a new line\n  opacityArea: 0.35, // the opacity of the area of the blob\n  dotRadius: 4, // the size of the colored circles of each blog\n  opacityCircles: 0.1, // the opacity of the circles of each blob\n  strokeWidth: 2, // the width of the stroke around each blob\n  roundStrokes: false, // if true the area and stroke will follow a round path (cardinal-closed)\n  transitionDuration: 200,\n  color: d3.scaleOrdinal(d3.schemeCategory10),\n});\n\n/**\n * Creates a container for the radar chart.\n * @param container the chart container\n * @param config the chart configuration\n * @returns the object with the svg element and the g element\n */\nconst createContainer = (container: ElementRef<HTMLDivElement>, config: IRadarChartOptions) => {\n  const id = container.nativeElement.id ?? 'radar-0';\n\n  d3.select(`#${id}`).select('svg').remove();\n  const svg = d3\n    .select(`#${id}`)\n    .append('svg')\n    .attr('width', config.width + config.margin.left + config.margin.right)\n    .attr('height', config.height + config.margin.top + config.margin.bottom)\n    .attr('class', id);\n  const g = svg\n    .append('g')\n    .attr('transform', `translate(${config.width / 2 + config.margin.left},${config.height / 2 + config.margin.top})`);\n\n  return { svg, g };\n};\n\n/**\n * Draws the radar chart circular grid.\n * @param axisGrid the chart axis grid\n * @param radius the chart radius value\n * @param maxValue the maximum value of the chart axis\n * @param config the chart configuration\n */\nconst drawCircularGrid = (\n  axisGrid: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  radius: number,\n  maxValue: number,\n  config: IRadarChartOptions,\n) => {\n  // background circles\n  axisGrid\n    .selectAll('.levels')\n    .data(d3.range(1, config.levels + 1).reverse())\n    .enter()\n    .append('circle')\n    .attr('class', 'grid-circle')\n    .attr('r', (d, i) => (radius / config.levels) * d)\n    .style('fill', '#CDCDCD')\n    .style('stroke', '#CDCDCD')\n    .style('fill-opacity', config.opacityCircles)\n    .style('filter', 'url(#glow)');\n  // text indicating at what % each level is\n  const axisGridX = 4;\n  axisGrid\n    .selectAll('.axis-label')\n    .data(d3.range(1, config.levels + 1).reverse())\n    .enter()\n    .append('text')\n    .attr('class', 'axis-label')\n    .attr('x', axisGridX)\n    .attr('y', d => (-d * radius) / config.levels)\n    .attr('dy', '0.4em')\n    .style('font-size', '10px')\n    .attr('fill', '#737373')\n    .text((d, i) => (maxValue * d) / config.levels);\n};\n\n/**\n * Wraps the chart axis labels text.\n * @param svgText the svg text elements\n * @param width the chart axis label width\n */\nconst wrapSvgText = (svgText: d3.Selection<SVGTextElement, string, SVGGElement, unknown>, width: number) => {\n  svgText.each(function (this: SVGTextElement) {\n    const text = d3.select<SVGElement, string>(this);\n    const words = text.text().split(/\\s+/).reverse();\n    if (words.length > 1) {\n      let line: string[] = [];\n      let lineNumber = 0;\n      const lineHeight = 1.4;\n      const y = text.attr('y');\n      const x = text.attr('x');\n      const dy = parseFloat(text.attr('dy') ?? 0);\n      let tspan = text.text(null).append('tspan').attr('x', x).attr('y', y).attr('dy', `${dy}em`);\n\n      let word = words.pop();\n\n      while (typeof word !== 'undefined') {\n        line.push(word ?? '');\n        tspan.text(line.join(' '));\n        if ((tspan.node()?.getComputedTextLength() ?? 0) > width) {\n          line.pop();\n          tspan.text(line.join(' '));\n          line = [word ?? ''];\n          lineNumber += 1;\n          tspan = text\n            .append('tspan')\n            .attr('x', x)\n            .attr('y', y)\n            .attr('dy', `${lineNumber * lineHeight + dy}em`)\n            .text(word ?? '');\n        }\n        word = words.pop();\n      }\n    }\n  });\n};\n\n/**\n * Creates the legend.\n * @param g the svg g element\n * @param config the chart configuration\n */\nconst createLegend = (g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>, config: IRadarChartOptions) => {\n  if (config.chartTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(-${config.width / 2 + config.margin.left / 2}, -${config.height / 2 + config.margin.top / 2})`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .text(config.chartTitle);\n  }\n};\n\n/**\n * Draws the radar chart axis.\n * @param axisGrid the chart axis grid\n * @param axisNames the chart axis names\n * @param radiusScale the chart radius scale\n * @param maxValue the maximum value of the chart axis\n * @param angleSlice the chart angle slice value\n * @param config the chart configuration\n */\nconst drawAxis = (\n  axisGrid: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  axisNames: string[],\n  radiusScale: d3.ScaleLinear<number, number>,\n  maxValue: number,\n  angleSlice: number,\n  config: IRadarChartOptions,\n) => {\n  // create the straight lines radiating outward from the center\n  const axis = axisGrid.selectAll('.axis').data(axisNames).enter().append('g').attr('class', 'axis');\n  // append the lines\n  axis\n    .append('line')\n    .attr('x1', 0)\n    .attr('y1', 0)\n    .attr('x2', (d, i) => radiusScale(maxValue * config.lineFactor) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('y2', (d, i) => radiusScale(maxValue * config.lineFactor) * Math.sin(angleSlice * i - Math.PI / 2))\n    .attr('class', 'line')\n    .style('stroke', 'white')\n    .style('stroke-width', '2px');\n  // append the labels at each axis\n  axis\n    .append('text')\n    .attr('class', 'legend')\n    .style('font-size', '11px')\n    .attr('text-anchor', 'middle')\n    .attr('dy', '0.35em')\n    .attr('x', (d, i) => radiusScale(maxValue * config.labelFactor) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('y', (d, i) => radiusScale(maxValue * config.labelFactor) * Math.sin(angleSlice * i - Math.PI / 2))\n    .text(d => d)\n    .call(wrapSvgText, config.labelTextWrapWidth);\n};\n\n/**\n * Draws the radar chart blobs.\n * @param radiusScale the chart radius scale\n * @param angleSlice the chart angle slice value\n * @param g the svg g element\n * @param data the chart data\n * @param config the chart configuration\n */\nconst drawRadarChartBlobs = (\n  radiusScale: d3.ScaleLinear<number, number>,\n  angleSlice: number,\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  data: TRadarChartData,\n  config: IRadarChartOptions,\n) => {\n  // the radial line function\n  const radarLine = d3\n    .lineRadial<IRadarChartDataNode>()\n    .radius(d => radiusScale(d.value))\n    .angle((d, i) => i * angleSlice);\n  // create a wrapper for the blobs\n  const blobWrapper = g.selectAll('.radar-wrapper').data(data).enter().append('g').attr('class', 'radar-wrapper');\n  // append the backgrounds\n  blobWrapper\n    .append('path')\n    .attr('class', 'radar-area')\n    .attr('d', (d, i) => radarLine(d))\n    .style('fill', (d, i) => config.color(i.toString()))\n    .style('fill-opacity', config.opacityArea)\n    .on('mouseover', function (d, i) {\n      // dim all blobs\n      const radarAreaFillOpacity = 0.1;\n      d3.selectAll('.radar-area').transition().duration(config.transitionDuration).style('fill-opacity', radarAreaFillOpacity);\n      // bring back the hovered over blob\n      const fillOpacity = 0.7;\n      d3.select(this).transition().duration(config.transitionDuration).style('fill-opacity', fillOpacity);\n    })\n    .on('mouseout', () => {\n      // bring back all blobs\n      d3.selectAll('.radar-area').transition().duration(config.transitionDuration).style('fill-opacity', config.opacityArea);\n    });\n  // create the outlines\n  blobWrapper\n    .append('path')\n    .attr('class', 'radar-stroke')\n    .attr('d', (d, i) => radarLine(d))\n    .style('stroke-width', `${config.strokeWidth}px`)\n    .style('stroke', (d, i) => config.color(i.toString()))\n    .style('fill', 'none')\n    .style('filter', 'url(#glow)');\n  // append the circles\n  const blobWrapperFillOpacity = 0.8;\n  blobWrapper\n    .selectAll('.radar-circle')\n    .data((d, i) => d)\n    .enter()\n    .append('circle')\n    .attr('class', 'radar-circle')\n    .attr('r', config.dotRadius)\n    .attr('cx', (d, i) => radiusScale(d.value) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('cy', (d, i) => radiusScale(d.value) * Math.sin(angleSlice * i - Math.PI / 2))\n    .style('fill', (d, i, j) => config.color(j.toString()))\n    .style('fill-opacity', blobWrapperFillOpacity);\n};\n\n/**\n * Appends the invisible tooltip circles.\n * @param g the svg g element\n * @param data the chart data\n * @param radiusScale the chart radius scale\n * @param angleSlice the chart angle slice value\n * @param config the chart configuration\n */\nconst appendInvisibleTooltipCircles = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  data: TRadarChartData,\n  radiusScale: d3.ScaleLinear<number, number>,\n  angleSlice: number,\n  config: IRadarChartOptions,\n) => {\n  // wrapper for the invisible circles on top\n  const blobCircleWrapper = g.selectAll('.radar-circle-wrapper').data(data).enter().append('g').attr('class', 'radar-circle-wrapper');\n  // set up the small tooltip for when you hover over a circle\n  const tooltip = g.append('text').attr('class', 'chart-tooltip').style('opacity', 0);\n  // append a set of invisible circles on top for the mouseover pop-up\n  const blobCircleWrapperRadiusMultiplier = 1.5;\n  blobCircleWrapper\n    .selectAll<SVGElement, IRadarChartDataNode>('.radar-invisible-circle')\n    .data((d, i) => d)\n    .enter()\n    .append('circle')\n    .attr('class', 'radar-invisible-circle')\n    .attr('r', config.dotRadius * blobCircleWrapperRadiusMultiplier)\n    .attr('cx', (d, i) => radiusScale(d.value) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('cy', (d, i) => radiusScale(d.value) * Math.sin(angleSlice * i - Math.PI / 2))\n    .style('fill', 'none')\n    .style('pointer-events', 'all')\n    .on('mouseover', function (event: MouseEvent, i) {\n      const modifier = 10;\n      const newX = parseFloat(d3.select(this).attr('cx')) - modifier;\n      const newY = parseFloat(d3.select(this).attr('cy')) - modifier;\n\n      const nodeData = (event.target as unknown as Record<string, IRadarChartDataNode>)['__data__'];\n      const tooltipText = `${nodeData.value} ${nodeData.unit}`;\n      tooltip.attr('x', newX).attr('y', newY).text(tooltipText).transition().duration(config.transitionDuration).style('opacity', 1);\n    })\n    .on('mouseout', () => {\n      tooltip.transition().duration(config.transitionDuration).style('opacity', 0);\n    });\n};\n\n/**\n * Draws the radar chart.\n * @param container the chart container\n * @param data the chart data\n * @param options the chart options\n * @returns the hart configuration\n */\nexport const drawRadarChart = (container: ElementRef<HTMLDivElement>, data: TRadarChartData, options?: Partial<IRadarChartOptions>) => {\n  const config: IRadarChartOptions = generateConfiguration<IRadarChartOptions>(defaultRadarChartConfig, options, {});\n\n  const maxValue = Math.max(config.maxValue, d3.max(data, i => d3.max(i.map(o => o.value))) ?? 0);\n  const axisNames = data[0].map((i, j) => i.axis);\n  const totalAxis = axisNames.length;\n  const radius = Math.min(config.width / 2 - config.margin.left / 2, config.height / 2 - config.margin.top / 2);\n  const angleSlice = (Math.PI * 2) / totalAxis;\n  const radiusScale = d3.scaleLinear([0, radius]).domain([0, maxValue]);\n\n  const { g } = createContainer(container, config);\n\n  // filter for the outside glow\n  const filter = g.append('defs').append('filter').attr('id', 'glow');\n  filter.append('feGaussianBlur').attr('stdDeviation', '2.5').attr('result', 'coloredBlur');\n  const feMerge = filter.append('feMerge');\n  feMerge.append('feMergeNode').attr('in', 'coloredBlur');\n  feMerge.append('feMergeNode').attr('in', 'SourceGraphic');\n\n  const axisGrid = g.append('g').attr('class', 'axis-wrapper');\n\n  drawCircularGrid(axisGrid, radius, maxValue, config);\n\n  drawAxis(axisGrid, axisNames, radiusScale, maxValue, angleSlice, config);\n\n  createLegend(g, config);\n\n  drawRadarChartBlobs(radiusScale, angleSlice, g, data, config);\n\n  appendInvisibleTooltipCircles(g, data, radiusScale, angleSlice, config);\n\n  return config;\n};\n"]}
|
|
307
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radar-chart.util.js","sourceRoot":"","sources":["../../../../../../libs/client-d3-charts/src/lib/util/radar-chart.util.ts"],"names":[],"mappings":"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","sourcesContent":["import { ElementRef } from '@angular/core';\nimport * as d3 from 'd3';\n\nimport { IRadarChartDataNode, IRadarChartOptions, TRadarChartData } from '../interfaces/radar-chart.interface';\nimport { generateConfiguration } from './configuration.util';\n\n/**\n * The radar chart default configuration.\n */\nexport const defaultRadarChartConfig: IRadarChartOptions = Object.freeze(<IRadarChartOptions>{\n  chartTitle: '',\n  width: 350,\n  height: 350,\n  margin: {\n    top: 50,\n    right: 50,\n    bottom: 50,\n    left: 50,\n  },\n  levels: 3, // how many levels or inner circles should there be drawn\n  maxValue: 0, // what is the value that the biggest circle will represent\n  lineFactor: 1.1, // how much farther than the radius of the outer circle should the lines be stretched\n  labelFactor: 1.15, // how much farther than the radius of the outer circle should the labels be placed\n  labelTextWrapWidth: 60, // the number of pixels after which a label needs to be given a new line\n  opacityArea: 0.35, // the opacity of the area of the blob\n  dotRadius: 4, // the size of the colored circles of each blog\n  opacityCircles: 0.1, // the opacity of the circles of each blob\n  strokeWidth: 2, // the width of the stroke around each blob\n  roundStrokes: false, // if true the area and stroke will follow a round path (cardinal-closed)\n  transitionDuration: 200,\n  color: d3.scaleOrdinal(d3.schemeCategory10),\n});\n\n/**\n * Creates a container for the radar chart.\n * @param container the chart container\n * @param config the chart configuration\n * @returns the object with the svg element and the g element\n */\nconst createContainer = (container: ElementRef<HTMLDivElement>, config: IRadarChartOptions) => {\n  const id = container.nativeElement.id ?? 'radar-0';\n\n  d3.select(`#${id}`).select('svg').remove();\n  const svg = d3\n    .select(`#${id}`)\n    .append('svg')\n    .attr('width', config.width + config.margin.left + config.margin.right)\n    .attr('height', config.height + config.margin.top + config.margin.bottom)\n    .attr('class', id);\n  const g = svg\n    .append('g')\n    .attr('transform', `translate(${config.width / 2 + config.margin.left},${config.height / 2 + config.margin.top})`);\n\n  return { svg, g };\n};\n\n/**\n * Draws the radar chart circular grid.\n * @param axisGrid the chart axis grid\n * @param radius the chart radius value\n * @param maxValue the maximum value of the chart axis\n * @param config the chart configuration\n */\nconst drawCircularGrid = (\n  axisGrid: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  radius: number,\n  maxValue: number,\n  config: IRadarChartOptions,\n) => {\n  // background circles\n  axisGrid\n    .selectAll('.levels')\n    .data(d3.range(1, config.levels + 1).reverse())\n    .enter()\n    .append('circle')\n    .attr('class', 'grid-circle')\n    .attr('r', (d, i) => (radius / config.levels) * d)\n    .style('fill', '#CDCDCD')\n    .style('stroke', '#CDCDCD')\n    .style('fill-opacity', config.opacityCircles)\n    .style('filter', 'url(#glow)');\n  // text indicating at what % each level is\n  const axisGridX = 4;\n  axisGrid\n    .selectAll('.axis-label')\n    .data(d3.range(1, config.levels + 1).reverse())\n    .enter()\n    .append('text')\n    .attr('class', 'axis-label')\n    .attr('x', axisGridX)\n    .attr('y', d => (-d * radius) / config.levels)\n    .attr('dy', '0.4em')\n    .style('font-size', '10px')\n    .attr('fill', '#737373')\n    .text((d, i) => (maxValue * d) / config.levels);\n};\n\n/**\n * Wraps the chart axis labels text.\n * @param svgText the svg text elements\n * @param width the chart axis label width\n */\nconst wrapSvgText = (svgText: d3.Selection<SVGTextElement, string, SVGGElement, unknown>, width: number) => {\n  svgText.each(function (this: SVGTextElement) {\n    const text = d3.select<SVGElement, string>(this);\n    const words = text.text().split(/\\s+/).reverse();\n    if (words.length > 1) {\n      let line: string[] = [];\n      let lineNumber = 0;\n      const lineHeight = 1.4;\n      const y = text.attr('y');\n      const x = text.attr('x');\n      const dy = parseFloat(text.attr('dy') ?? 0);\n      let tspan = text.text(null).append('tspan').attr('x', x).attr('y', y).attr('dy', `${dy}em`);\n\n      let word = words.pop();\n\n      while (typeof word !== 'undefined') {\n        line.push(word ?? '');\n        tspan.text(line.join(' '));\n        if ((tspan.node()?.getComputedTextLength() ?? 0) > width) {\n          line.pop();\n          tspan.text(line.join(' '));\n          line = [word ?? ''];\n          lineNumber += 1;\n          tspan = text\n            .append('tspan')\n            .attr('x', x)\n            .attr('y', y)\n            .attr('dy', `${lineNumber * lineHeight + dy}em`)\n            .text(word ?? '');\n        }\n        word = words.pop();\n      }\n    }\n  });\n};\n\n/**\n * Creates the legend.\n * @param g the svg g element\n * @param config the chart configuration\n */\nconst createLegend = (g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>, config: IRadarChartOptions) => {\n  if (config.chartTitle !== '') {\n    g.append('g')\n      .attr('transform', `translate(-${config.width / 2 + config.margin.left / 2}, -${config.height / 2 + config.margin.top / 2})`)\n      .append('text')\n      .style('font-size', '12px')\n      .attr('class', 'legend')\n      .text(config.chartTitle);\n  }\n};\n\n/**\n * Draws the radar chart axis.\n * @param axisGrid the chart axis grid\n * @param axisNames the chart axis names\n * @param radiusScale the chart radius scale\n * @param maxValue the maximum value of the chart axis\n * @param angleSlice the chart angle slice value\n * @param config the chart configuration\n */\nconst drawAxis = (\n  axisGrid: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  axisNames: string[],\n  radiusScale: d3.ScaleLinear<number, number>,\n  maxValue: number,\n  angleSlice: number,\n  config: IRadarChartOptions,\n) => {\n  // create the straight lines radiating outward from the center\n  const axis = axisGrid.selectAll('.axis').data(axisNames).enter().append('g').attr('class', 'axis');\n  // append the lines\n  axis\n    .append('line')\n    .attr('x1', 0)\n    .attr('y1', 0)\n    .attr('x2', (d, i) => radiusScale(maxValue * config.lineFactor) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('y2', (d, i) => radiusScale(maxValue * config.lineFactor) * Math.sin(angleSlice * i - Math.PI / 2))\n    .attr('class', 'line')\n    .style('stroke', 'white')\n    .style('stroke-width', '2px');\n  // append the labels at each axis\n  axis\n    .append('text')\n    .attr('class', 'legend')\n    .style('font-size', '11px')\n    .attr('text-anchor', 'middle')\n    .attr('dy', '0.35em')\n    .attr('x', (d, i) => radiusScale(maxValue * config.labelFactor) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('y', (d, i) => radiusScale(maxValue * config.labelFactor) * Math.sin(angleSlice * i - Math.PI / 2))\n    .text(d => d)\n    .call(wrapSvgText, config.labelTextWrapWidth);\n};\n\n/**\n * Draws the radar chart blobs.\n * @param radiusScale the chart radius scale\n * @param angleSlice the chart angle slice value\n * @param g the svg g element\n * @param data the chart data\n * @param config the chart configuration\n */\nconst drawRadarChartBlobs = (\n  radiusScale: d3.ScaleLinear<number, number>,\n  angleSlice: number,\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  data: TRadarChartData,\n  config: IRadarChartOptions,\n) => {\n  // the radial line function\n  const radarLine = d3\n    .lineRadial<IRadarChartDataNode>()\n    .radius(d => radiusScale(d.value))\n    .angle((d, i) => i * angleSlice);\n  // create a wrapper for the blobs\n  const blobWrapper = g.selectAll('.radar-wrapper').data(data).enter().append('g').attr('class', 'radar-wrapper');\n  // append the backgrounds\n  blobWrapper\n    .append('path')\n    .attr('class', 'radar-area')\n    .attr('d', (d, i) => radarLine(d))\n    .style('fill', (d, i) => config.color(i.toString()))\n    .style('fill-opacity', config.opacityArea)\n    .on('mouseover', function (d, i) {\n      // dim all blobs\n      const radarAreaFillOpacity = 0.1;\n      d3.selectAll('.radar-area').transition().duration(config.transitionDuration).style('fill-opacity', radarAreaFillOpacity);\n      // bring back the hovered over blob\n      const fillOpacity = 0.7;\n      d3.select(this).transition().duration(config.transitionDuration).style('fill-opacity', fillOpacity);\n    })\n    .on('mouseout', () => {\n      // bring back all blobs\n      d3.selectAll('.radar-area').transition().duration(config.transitionDuration).style('fill-opacity', config.opacityArea);\n    });\n  // create the outlines\n  blobWrapper\n    .append('path')\n    .attr('class', 'radar-stroke')\n    .attr('d', (d, i) => radarLine(d))\n    .style('stroke-width', `${config.strokeWidth}px`)\n    .style('stroke', (d, i) => config.color(i.toString()))\n    .style('fill', 'none')\n    .style('filter', 'url(#glow)');\n  // append the circles\n  const blobWrapperFillOpacity = 0.8;\n  blobWrapper\n    .selectAll('.radar-circle')\n    .data((d, i) => d)\n    .enter()\n    .append('circle')\n    .attr('class', 'radar-circle')\n    .attr('r', config.dotRadius)\n    .attr('cx', (d, i) => radiusScale(d.value) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('cy', (d, i) => radiusScale(d.value) * Math.sin(angleSlice * i - Math.PI / 2))\n    .style('fill', (d, i, j) => config.color(j.toString()))\n    .style('fill-opacity', blobWrapperFillOpacity);\n};\n\n/**\n * Appends the invisible tooltip circles.\n * @param g the svg g element\n * @param data the chart data\n * @param radiusScale the chart radius scale\n * @param angleSlice the chart angle slice value\n * @param config the chart configuration\n */\nconst appendInvisibleTooltipCircles = (\n  g: d3.Selection<SVGGElement, unknown, HTMLElement, unknown>,\n  data: TRadarChartData,\n  radiusScale: d3.ScaleLinear<number, number>,\n  angleSlice: number,\n  config: IRadarChartOptions,\n) => {\n  // wrapper for the invisible circles on top\n  const blobCircleWrapper = g.selectAll('.radar-circle-wrapper').data(data).enter().append('g').attr('class', 'radar-circle-wrapper');\n  // append a set of invisible circles on top for the mouseover pop-up\n  const blobCircleWrapperRadiusMultiplier = 1.5;\n  blobCircleWrapper\n    .selectAll<SVGElement, IRadarChartDataNode>('.radar-invisible-circle')\n    .data((d, i) => d)\n    .enter()\n    .append('circle')\n    .attr('class', 'radar-invisible-circle')\n    .attr('r', config.dotRadius * blobCircleWrapperRadiusMultiplier)\n    .attr('cx', (d, i) => radiusScale(d.value) * Math.cos(angleSlice * i - Math.PI / 2))\n    .attr('cy', (d, i) => radiusScale(d.value) * Math.sin(angleSlice * i - Math.PI / 2))\n    .style('fill', 'none')\n    .style('pointer-events', 'all')\n    .on('mouseover', function (event: MouseEvent, i) {\n      const modifier = 10;\n      const newX = parseFloat(d3.select(this).attr('cx')) - modifier;\n      const newY = parseFloat(d3.select(this).attr('cy')) - modifier;\n\n      const nodeData = (event.target as unknown as Record<string, IRadarChartDataNode>)['__data__'];\n      const tooltipText = `${nodeData.value} ${nodeData.unit}`;\n      g.append('text')\n        .attr('class', 'chart-tooltip')\n        .style('opacity', 0)\n        .attr('x', newX)\n        .attr('y', newY)\n        .text(tooltipText)\n        .transition()\n        .duration(config.transitionDuration)\n        .style('opacity', 1);\n    })\n    .on('mouseout', () => {\n      d3.selectAll('.chart-tooltip')\n        .transition()\n        .duration(config.transitionDuration / 2)\n        .style('opacity', 0)\n        .remove();\n    });\n};\n\n/**\n * Draws the radar chart.\n * @param container the chart container\n * @param data the chart data\n * @param options the chart options\n * @returns the hart configuration\n */\nexport const drawRadarChart = (container: ElementRef<HTMLDivElement>, data: TRadarChartData, options?: Partial<IRadarChartOptions>) => {\n  const config: IRadarChartOptions = generateConfiguration<IRadarChartOptions>(defaultRadarChartConfig, options, {});\n\n  const maxValue = Math.max(config.maxValue, d3.max(data, i => d3.max(i.map(o => o.value))) ?? 0);\n  const axisNames = data[0].map((i, j) => i.axis);\n  const totalAxis = axisNames.length;\n  const radius = Math.min(config.width / 2 - config.margin.left / 2, config.height / 2 - config.margin.top / 2);\n  const angleSlice = (Math.PI * 2) / totalAxis;\n  const radiusScale = d3.scaleLinear([0, radius]).domain([0, maxValue]);\n\n  const { g } = createContainer(container, config);\n\n  // filter for the outside glow\n  const filter = g.append('defs').append('filter').attr('id', 'glow');\n  filter.append('feGaussianBlur').attr('stdDeviation', '2.5').attr('result', 'coloredBlur');\n  const feMerge = filter.append('feMerge');\n  feMerge.append('feMergeNode').attr('in', 'coloredBlur');\n  feMerge.append('feMergeNode').attr('in', 'SourceGraphic');\n\n  const axisGrid = g.append('g').attr('class', 'axis-wrapper');\n\n  drawCircularGrid(axisGrid, radius, maxValue, config);\n\n  drawAxis(axisGrid, axisNames, radiusScale, maxValue, angleSlice, config);\n\n  createLegend(g, config);\n\n  drawRadarChartBlobs(radiusScale, angleSlice, g, data, config);\n\n  appendInvisibleTooltipCircles(g, data, radiusScale, angleSlice, config);\n\n  return config;\n};\n"]}
|
|
@@ -754,7 +754,10 @@ const onMouseOver = (self, d, g, config) => {
|
|
|
754
754
|
.style('font-size', '11px')
|
|
755
755
|
.attr('dx', () => (config.width - config.margin.left - config.margin.right) / tooltipShift)
|
|
756
756
|
.attr('dy', () => tooltipDy)
|
|
757
|
-
.text(() => `${d.value} (${new Date(d.timestamp).toUTCString()})`)
|
|
757
|
+
.text(() => `${d.value} (${new Date(d.timestamp).toUTCString()})`)
|
|
758
|
+
.transition()
|
|
759
|
+
.duration(config.transitionDuration)
|
|
760
|
+
.style('opacity', 1);
|
|
758
761
|
};
|
|
759
762
|
/**
|
|
760
763
|
* The mouse out event handler.
|
|
@@ -765,7 +768,11 @@ const onMouseOut = (self, config) => {
|
|
|
765
768
|
const duration = 400;
|
|
766
769
|
d3.select(self).attr('class', 'dot');
|
|
767
770
|
d3.select(self).transition().duration(duration).attr('r', config.dotRadius);
|
|
768
|
-
d3.selectAll('.chart-tooltip')
|
|
771
|
+
d3.selectAll('.chart-tooltip')
|
|
772
|
+
.transition()
|
|
773
|
+
.duration(config.transitionDuration / 2)
|
|
774
|
+
.style('opacity', 0)
|
|
775
|
+
.remove();
|
|
769
776
|
};
|
|
770
777
|
/**
|
|
771
778
|
* Draws the chart lines, dots, and sets the mouse pointer events.
|
|
@@ -903,7 +910,6 @@ const drawPieChart = (container, data, options) => {
|
|
|
903
910
|
const pie = d3.pie().value(datum => datum.y);
|
|
904
911
|
const radius = Math.min(config.width, config.height) / 2;
|
|
905
912
|
const arc = d3.arc().innerRadius(config.innerRadius).outerRadius(radius);
|
|
906
|
-
const tooltip = g.append('text').attr('class', 'chart-tooltip').style('opacity', 0);
|
|
907
913
|
const arcs = g
|
|
908
914
|
.selectAll('arc')
|
|
909
915
|
.data(pie(data))
|
|
@@ -912,13 +918,10 @@ const drawPieChart = (container, data, options) => {
|
|
|
912
918
|
.attr('class', 'arc')
|
|
913
919
|
.on('mouseover', function (event, d) {
|
|
914
920
|
this.style.opacity = '0.8';
|
|
915
|
-
const modifier = 10;
|
|
916
|
-
const x = parseFloat(d3.select(this).attr('cx')) - modifier;
|
|
917
|
-
const y = parseFloat(d3.select(this).attr('cy')) - modifier;
|
|
918
921
|
const tooltipText = `${d.data.key}: ${d.data.y}`;
|
|
919
|
-
|
|
920
|
-
.attr('
|
|
921
|
-
.
|
|
922
|
+
g.append('text')
|
|
923
|
+
.attr('class', 'chart-tooltip')
|
|
924
|
+
.style('opacity', 0)
|
|
922
925
|
.attr('dx', -config.width / (2 * 2 * 2))
|
|
923
926
|
.attr('dy', config.height / 2 + config.margin.top)
|
|
924
927
|
.text(tooltipText)
|
|
@@ -928,7 +931,11 @@ const drawPieChart = (container, data, options) => {
|
|
|
928
931
|
})
|
|
929
932
|
.on('mouseout', function (event, d) {
|
|
930
933
|
this.style.opacity = 'unset';
|
|
931
|
-
|
|
934
|
+
d3.selectAll('.chart-tooltip')
|
|
935
|
+
.transition()
|
|
936
|
+
.duration(config.transitionDuration / 2)
|
|
937
|
+
.style('opacity', 0)
|
|
938
|
+
.remove();
|
|
932
939
|
});
|
|
933
940
|
arcs
|
|
934
941
|
.append('path')
|
|
@@ -1189,8 +1196,6 @@ const drawRadarChartBlobs = (radiusScale, angleSlice, g, data, config) => {
|
|
|
1189
1196
|
const appendInvisibleTooltipCircles = (g, data, radiusScale, angleSlice, config) => {
|
|
1190
1197
|
// wrapper for the invisible circles on top
|
|
1191
1198
|
const blobCircleWrapper = g.selectAll('.radar-circle-wrapper').data(data).enter().append('g').attr('class', 'radar-circle-wrapper');
|
|
1192
|
-
// set up the small tooltip for when you hover over a circle
|
|
1193
|
-
const tooltip = g.append('text').attr('class', 'chart-tooltip').style('opacity', 0);
|
|
1194
1199
|
// append a set of invisible circles on top for the mouseover pop-up
|
|
1195
1200
|
const blobCircleWrapperRadiusMultiplier = 1.5;
|
|
1196
1201
|
blobCircleWrapper
|
|
@@ -1210,10 +1215,22 @@ const appendInvisibleTooltipCircles = (g, data, radiusScale, angleSlice, config)
|
|
|
1210
1215
|
const newY = parseFloat(d3.select(this).attr('cy')) - modifier;
|
|
1211
1216
|
const nodeData = event.target['__data__'];
|
|
1212
1217
|
const tooltipText = `${nodeData.value} ${nodeData.unit}`;
|
|
1213
|
-
|
|
1218
|
+
g.append('text')
|
|
1219
|
+
.attr('class', 'chart-tooltip')
|
|
1220
|
+
.style('opacity', 0)
|
|
1221
|
+
.attr('x', newX)
|
|
1222
|
+
.attr('y', newY)
|
|
1223
|
+
.text(tooltipText)
|
|
1224
|
+
.transition()
|
|
1225
|
+
.duration(config.transitionDuration)
|
|
1226
|
+
.style('opacity', 1);
|
|
1214
1227
|
})
|
|
1215
1228
|
.on('mouseout', () => {
|
|
1216
|
-
|
|
1229
|
+
d3.selectAll('.chart-tooltip')
|
|
1230
|
+
.transition()
|
|
1231
|
+
.duration(config.transitionDuration / 2)
|
|
1232
|
+
.style('opacity', 0)
|
|
1233
|
+
.remove();
|
|
1217
1234
|
});
|
|
1218
1235
|
};
|
|
1219
1236
|
/**
|