UNPKG

26.2 kBJavaScriptView Raw
1import { Component, Input, Output, EventEmitter, ViewChild, ChangeDetectionStrategy, PLATFORM_ID, Inject } from '@angular/core';
2import { trigger, style, animate, transition } from '@angular/animations';
3import { createMouseEvent } from '../events';
4import { isPlatformBrowser } from '@angular/common';
5import { PlacementTypes } from './tooltip/position';
6import { StyleTypes } from './tooltip/style.type';
7import { ScaleType } from './types/scale-type.enum';
8export 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}
151TooltipArea.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];
214TooltipArea.ctorParameters = () => [
215 { type: undefined, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
216];
217TooltipArea.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