1 | import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, PLATFORM_ID, Inject } from '@angular/core';
|
2 | import { trigger, style, animate, transition } from '@angular/animations';
|
3 | import { createMouseEvent } from '../events';
|
4 | import { isPlatformBrowser } from '@angular/common';
|
5 | import { PlacementTypes } from './tooltip/position';
|
6 | import { StyleTypes } from './tooltip/style.type';
|
7 | import { ScaleType } from './types/scale-type.enum';
|
8 | export class TooltipArea {
|
9 | constructor(platformId) {
|
10 | this.platformId = platformId;
|
11 | this.anchorOpacity = 0;
|
12 | this.anchorPos = -1;
|
13 | this.anchorValues = [];
|
14 | this.placementTypes = PlacementTypes;
|
15 | this.styleTypes = StyleTypes;
|
16 | this.showPercentage = false;
|
17 | this.tooltipDisabled = false;
|
18 | this.hover = new EventEmitter();
|
19 | }
|
20 | getValues(xVal) {
|
21 | const results = [];
|
22 | for (const group of this.results) {
|
23 | const item = group.series.find(d => d.name.toString() === xVal.toString());
|
24 | let groupName = group.name;
|
25 | if (groupName instanceof Date) {
|
26 | groupName = groupName.toLocaleDateString();
|
27 | }
|
28 | if (item) {
|
29 | const label = item.name;
|
30 | let val = item.value;
|
31 | if (this.showPercentage) {
|
32 | val = (item.d1 - item.d0).toFixed(2) + '%';
|
33 | }
|
34 | let color;
|
35 | if (this.colors.scaleType === ScaleType.Linear) {
|
36 | let v = val;
|
37 | if (item.d1) {
|
38 | v = item.d1;
|
39 | }
|
40 | color = this.colors.getColor(v);
|
41 | }
|
42 | else {
|
43 | color = this.colors.getColor(group.name);
|
44 | }
|
45 | const data = Object.assign({}, item, {
|
46 | value: val,
|
47 | name: label,
|
48 | series: groupName,
|
49 | min: item.min,
|
50 | max: item.max,
|
51 | color
|
52 | });
|
53 | results.push(data);
|
54 | }
|
55 | }
|
56 | return results;
|
57 | }
|
58 | mouseMove(event) {
|
59 | if (!isPlatformBrowser(this.platformId)) {
|
60 | return;
|
61 | }
|
62 | const xPos = event.pageX - event.target.getBoundingClientRect().left;
|
63 | const closestIndex = this.findClosestPointIndex(xPos);
|
64 | const closestPoint = this.xSet[closestIndex];
|
65 | this.anchorPos = this.xScale(closestPoint);
|
66 | this.anchorPos = Math.max(0, this.anchorPos);
|
67 | this.anchorPos = Math.min(this.dims.width, this.anchorPos);
|
68 | this.anchorValues = this.getValues(closestPoint);
|
69 | if (this.anchorPos !== this.lastAnchorPos) {
|
70 | const ev = createMouseEvent('mouseleave');
|
71 | this.tooltipAnchor.nativeElement.dispatchEvent(ev);
|
72 | this.anchorOpacity = 0.7;
|
73 | this.hover.emit({
|
74 | value: closestPoint
|
75 | });
|
76 | this.showTooltip();
|
77 | this.lastAnchorPos = this.anchorPos;
|
78 | }
|
79 | }
|
80 | findClosestPointIndex(xPos) {
|
81 | let minIndex = 0;
|
82 | let maxIndex = this.xSet.length - 1;
|
83 | let minDiff = Number.MAX_VALUE;
|
84 | let closestIndex = 0;
|
85 | while (minIndex <= maxIndex) {
|
86 | const currentIndex = ((minIndex + maxIndex) / 2) | 0;
|
87 | const currentElement = this.xScale(this.xSet[currentIndex]);
|
88 | const curDiff = Math.abs(currentElement - xPos);
|
89 | if (curDiff < minDiff) {
|
90 | minDiff = curDiff;
|
91 | closestIndex = currentIndex;
|
92 | }
|
93 | if (currentElement < xPos) {
|
94 | minIndex = currentIndex + 1;
|
95 | }
|
96 | else if (currentElement > xPos) {
|
97 | maxIndex = currentIndex - 1;
|
98 | }
|
99 | else {
|
100 | minDiff = 0;
|
101 | closestIndex = currentIndex;
|
102 | break;
|
103 | }
|
104 | }
|
105 | return closestIndex;
|
106 | }
|
107 | showTooltip() {
|
108 | const event = createMouseEvent('mouseenter');
|
109 | this.tooltipAnchor.nativeElement.dispatchEvent(event);
|
110 | }
|
111 | hideTooltip() {
|
112 | const event = createMouseEvent('mouseleave');
|
113 | this.tooltipAnchor.nativeElement.dispatchEvent(event);
|
114 | this.anchorOpacity = 0;
|
115 | this.lastAnchorPos = -1;
|
116 | }
|
117 | getToolTipText(tooltipItem) {
|
118 | let result = '';
|
119 | if (tooltipItem.series !== undefined) {
|
120 | result += tooltipItem.series;
|
121 | }
|
122 | else {
|
123 | result += '???';
|
124 | }
|
125 | result += ': ';
|
126 | if (tooltipItem.value !== undefined) {
|
127 | result += tooltipItem.value.toLocaleString();
|
128 | }
|
129 | if (tooltipItem.min !== undefined || tooltipItem.max !== undefined) {
|
130 | result += ' (';
|
131 | if (tooltipItem.min !== undefined) {
|
132 | if (tooltipItem.max === undefined) {
|
133 | result += '≥';
|
134 | }
|
135 | result += tooltipItem.min.toLocaleString();
|
136 | if (tooltipItem.max !== undefined) {
|
137 | result += ' - ';
|
138 | }
|
139 | }
|
140 | else if (tooltipItem.max !== undefined) {
|
141 | result += '≤';
|
142 | }
|
143 | if (tooltipItem.max !== undefined) {
|
144 | result += tooltipItem.max.toLocaleString();
|
145 | }
|
146 | result += ')';
|
147 | }
|
148 | return result;
|
149 | }
|
150 | }
|
151 | TooltipArea.decorators = [
|
152 | { type: Component, args: [{
|
153 | selector: 'g[ngx-charts-tooltip-area]',
|
154 | template: `
|
155 | <svg:g>
|
156 | <svg:rect
|
157 | class="tooltip-area"
|
158 | [attr.x]="0"
|
159 | y="0"
|
160 | [attr.width]="dims.width"
|
161 | [attr.height]="dims.height"
|
162 | style="opacity: 0; cursor: 'auto';"
|
163 | (mousemove)="mouseMove($event)"
|
164 | (mouseleave)="hideTooltip()"
|
165 | />
|
166 | <ng-template #defaultTooltipTemplate let-model="model">
|
167 | <xhtml:div class="area-tooltip-container">
|
168 | <xhtml:div *ngFor="let tooltipItem of model" class="tooltip-item">
|
169 | <xhtml:span class="tooltip-item-color" [style.background-color]="tooltipItem.color"></xhtml:span>
|
170 | {{ getToolTipText(tooltipItem) }}
|
171 | </xhtml:div>
|
172 | </xhtml:div>
|
173 | </ng-template>
|
174 | <svg:rect
|
175 | #tooltipAnchor
|
176 | [@animationState]="anchorOpacity !== 0 ? 'active' : 'inactive'"
|
177 | class="tooltip-anchor"
|
178 | [attr.x]="anchorPos"
|
179 | y="0"
|
180 | [attr.width]="1"
|
181 | [attr.height]="dims.height"
|
182 | [style.opacity]="anchorOpacity"
|
183 | [style.pointer-events]="'none'"
|
184 | ngx-tooltip
|
185 | [tooltipDisabled]="tooltipDisabled"
|
186 | [tooltipPlacement]="placementTypes.Right"
|
187 | [tooltipType]="styleTypes.tooltip"
|
188 | [tooltipSpacing]="15"
|
189 | [tooltipTemplate]="tooltipTemplate ? tooltipTemplate : defaultTooltipTemplate"
|
190 | [tooltipContext]="anchorValues"
|
191 | [tooltipImmediateExit]="true"
|
192 | />
|
193 | </svg:g>
|
194 | `,
|
195 | changeDetection: ChangeDetectionStrategy.OnPush,
|
196 | animations: [
|
197 | trigger('animationState', [
|
198 | transition('inactive => active', [
|
199 | style({
|
200 | opacity: 0
|
201 | }),
|
202 | animate(250, style({ opacity: 0.7 }))
|
203 | ]),
|
204 | transition('active => inactive', [
|
205 | style({
|
206 | opacity: 0.7
|
207 | }),
|
208 | animate(250, style({ opacity: 0 }))
|
209 | ])
|
210 | ])
|
211 | ]
|
212 | },] }
|
213 | ];
|
214 | TooltipArea.ctorParameters = () => [
|
215 | { type: undefined, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
216 | ];
|
217 | TooltipArea.propDecorators = {
|
218 | dims: [{ type: Input }],
|
219 | xSet: [{ type: Input }],
|
220 | xScale: [{ type: Input }],
|
221 | yScale: [{ type: Input }],
|
222 | results: [{ type: Input }],
|
223 | colors: [{ type: Input }],
|
224 | showPercentage: [{ type: Input }],
|
225 | tooltipDisabled: [{ type: Input }],
|
226 | tooltipTemplate: [{ type: Input }],
|
227 | hover: [{ type: Output }],
|
228 | tooltipAnchor: [{ type: ViewChild, args: ['tooltipAnchor', { static: false },] }]
|
229 | };
|
230 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tooltip-area.component.js","sourceRoot":"","sources":["../../../../../../projects/swimlane/ngx-charts/src/lib/common/tooltip-area.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  ViewChild,\n  ChangeDetectionStrategy,\n  TemplateRef,\n  PLATFORM_ID,\n  Inject\n} from '@angular/core';\nimport { trigger, style, animate, transition } from '@angular/animations';\nimport { createMouseEvent } from '../events';\nimport { isPlatformBrowser } from '@angular/common';\nimport { ColorHelper } from '../common/color.helper';\nimport { PlacementTypes } from './tooltip/position';\nimport { StyleTypes } from './tooltip/style.type';\nimport { ViewDimensions } from './types/view-dimension.interface';\nimport { ScaleType } from './types/scale-type.enum';\n\nexport interface Tooltip {\n  color: string;\n  d0: number;\n  d1: number;\n  max: number;\n  min: number;\n  name: any;\n  series: any;\n  value: any;\n}\n\n@Component({\n  selector: 'g[ngx-charts-tooltip-area]',\n  template: `\n    <svg:g>\n      <svg:rect\n        class=\"tooltip-area\"\n        [attr.x]=\"0\"\n        y=\"0\"\n        [attr.width]=\"dims.width\"\n        [attr.height]=\"dims.height\"\n        style=\"opacity: 0; cursor: 'auto';\"\n        (mousemove)=\"mouseMove($event)\"\n        (mouseleave)=\"hideTooltip()\"\n      />\n      <ng-template #defaultTooltipTemplate let-model=\"model\">\n        <xhtml:div class=\"area-tooltip-container\">\n          <xhtml:div *ngFor=\"let tooltipItem of model\" class=\"tooltip-item\">\n            <xhtml:span class=\"tooltip-item-color\" [style.background-color]=\"tooltipItem.color\"></xhtml:span>\n            {{ getToolTipText(tooltipItem) }}\n          </xhtml:div>\n        </xhtml:div>\n      </ng-template>\n      <svg:rect\n        #tooltipAnchor\n        [@animationState]=\"anchorOpacity !== 0 ? 'active' : 'inactive'\"\n        class=\"tooltip-anchor\"\n        [attr.x]=\"anchorPos\"\n        y=\"0\"\n        [attr.width]=\"1\"\n        [attr.height]=\"dims.height\"\n        [style.opacity]=\"anchorOpacity\"\n        [style.pointer-events]=\"'none'\"\n        ngx-tooltip\n        [tooltipDisabled]=\"tooltipDisabled\"\n        [tooltipPlacement]=\"placementTypes.Right\"\n        [tooltipType]=\"styleTypes.tooltip\"\n        [tooltipSpacing]=\"15\"\n        [tooltipTemplate]=\"tooltipTemplate ? tooltipTemplate : defaultTooltipTemplate\"\n        [tooltipContext]=\"anchorValues\"\n        [tooltipImmediateExit]=\"true\"\n      />\n    </svg:g>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  animations: [\n    trigger('animationState', [\n      transition('inactive => active', [\n        style({\n          opacity: 0\n        }),\n        animate(250, style({ opacity: 0.7 }))\n      ]),\n      transition('active => inactive', [\n        style({\n          opacity: 0.7\n        }),\n        animate(250, style({ opacity: 0 }))\n      ])\n    ])\n  ]\n})\nexport class TooltipArea {\n  anchorOpacity: number = 0;\n  anchorPos: number = -1;\n  anchorValues: Tooltip[] = [];\n  lastAnchorPos: number;\n\n  placementTypes = PlacementTypes;\n  styleTypes = StyleTypes;\n\n  @Input() dims: ViewDimensions;\n  @Input() xSet: any[];\n  @Input() xScale;\n  @Input() yScale;\n  @Input() results: any[];\n  @Input() colors: ColorHelper;\n  @Input() showPercentage: boolean = false;\n  @Input() tooltipDisabled: boolean = false;\n  @Input() tooltipTemplate: TemplateRef<any>;\n\n  @Output() hover: EventEmitter<{ value: any }> = new EventEmitter();\n\n  @ViewChild('tooltipAnchor', { static: false }) tooltipAnchor;\n\n  constructor(@Inject(PLATFORM_ID) private platformId: any) {}\n\n  getValues(xVal): Tooltip[] {\n    const results = [];\n\n    for (const group of this.results) {\n      const item = group.series.find(d => d.name.toString() === xVal.toString());\n      let groupName = group.name;\n      if (groupName instanceof Date) {\n        groupName = groupName.toLocaleDateString();\n      }\n\n      if (item) {\n        const label = item.name;\n        let val = item.value;\n        if (this.showPercentage) {\n          val = (item.d1 - item.d0).toFixed(2) + '%';\n        }\n        let color;\n        if (this.colors.scaleType === ScaleType.Linear) {\n          let v = val;\n          if (item.d1) {\n            v = item.d1;\n          }\n          color = this.colors.getColor(v);\n        } else {\n          color = this.colors.getColor(group.name);\n        }\n\n        const data = Object.assign({}, item, {\n          value: val,\n          name: label,\n          series: groupName,\n          min: item.min,\n          max: item.max,\n          color\n        });\n\n        results.push(data);\n      }\n    }\n\n    return results;\n  }\n\n  mouseMove(event) {\n    if (!isPlatformBrowser(this.platformId)) {\n      return;\n    }\n\n    const xPos = event.pageX - event.target.getBoundingClientRect().left;\n\n    const closestIndex = this.findClosestPointIndex(xPos);\n    const closestPoint = this.xSet[closestIndex];\n    this.anchorPos = this.xScale(closestPoint);\n    this.anchorPos = Math.max(0, this.anchorPos);\n    this.anchorPos = Math.min(this.dims.width, this.anchorPos);\n\n    this.anchorValues = this.getValues(closestPoint);\n    if (this.anchorPos !== this.lastAnchorPos) {\n      const ev = createMouseEvent('mouseleave');\n      this.tooltipAnchor.nativeElement.dispatchEvent(ev);\n      this.anchorOpacity = 0.7;\n      this.hover.emit({\n        value: closestPoint\n      });\n      this.showTooltip();\n\n      this.lastAnchorPos = this.anchorPos;\n    }\n  }\n\n  findClosestPointIndex(xPos: number): number {\n    let minIndex = 0;\n    let maxIndex = this.xSet.length - 1;\n    let minDiff = Number.MAX_VALUE;\n    let closestIndex = 0;\n\n    while (minIndex <= maxIndex) {\n      const currentIndex = ((minIndex + maxIndex) / 2) | 0;\n      const currentElement = this.xScale(this.xSet[currentIndex]);\n\n      const curDiff = Math.abs(currentElement - xPos);\n\n      if (curDiff < minDiff) {\n        minDiff = curDiff;\n        closestIndex = currentIndex;\n      }\n\n      if (currentElement < xPos) {\n        minIndex = currentIndex + 1;\n      } else if (currentElement > xPos) {\n        maxIndex = currentIndex - 1;\n      } else {\n        minDiff = 0;\n        closestIndex = currentIndex;\n        break;\n      }\n    }\n\n    return closestIndex;\n  }\n\n  showTooltip(): void {\n    const event = createMouseEvent('mouseenter');\n    this.tooltipAnchor.nativeElement.dispatchEvent(event);\n  }\n\n  hideTooltip(): void {\n    const event = createMouseEvent('mouseleave');\n    this.tooltipAnchor.nativeElement.dispatchEvent(event);\n    this.anchorOpacity = 0;\n    this.lastAnchorPos = -1;\n  }\n\n  getToolTipText(tooltipItem: Tooltip): string {\n    let result: string = '';\n    if (tooltipItem.series !== undefined) {\n      result += tooltipItem.series;\n    } else {\n      result += '???';\n    }\n    result += ': ';\n    if (tooltipItem.value !== undefined) {\n      result += tooltipItem.value.toLocaleString();\n    }\n    if (tooltipItem.min !== undefined || tooltipItem.max !== undefined) {\n      result += ' (';\n      if (tooltipItem.min !== undefined) {\n        if (tooltipItem.max === undefined) {\n          result += '≥';\n        }\n        result += tooltipItem.min.toLocaleString();\n        if (tooltipItem.max !== undefined) {\n          result += ' - ';\n        }\n      } else if (tooltipItem.max !== undefined) {\n        result += '≤';\n      }\n      if (tooltipItem.max !== undefined) {\n        result += tooltipItem.max.toLocaleString();\n      }\n      result += ')';\n    }\n    return result;\n  }\n}\n"]} |
\ | No newline at end of file |