我目前在Angular2中使用ngx-charts,我想自定义一些图例,例如将图例放在图表下方,或重命名图例字段,或扩大图例的范围(部分文本会被切除。)等等。
有什么好的选项可以自定义图例?还是一般不可能,或者最好的方法是什么? 目前,我的图表图例如下:
有些图例太宽,有些图例被切除,我想将图例放置在图表下方,例如...
答案 0 :(得分:4)
要将图例放置在图表下方,可以使用legendPosition
输入:
<ngx-charts-chart [legendPosition]="'below'" [results]="chartData" [legend]="true"></ngx-charts-chart>
legendPosition
的唯一选项是'right'(默认)和'below'。
似乎没有自定义图例字段的选项(高级饼图中有一个更可自定义的图例,但您的示例是其他类型的图。)我认为最好的解决方法是传递字段标题完全按照您希望它们出现在图例中的方式输入到chartData的chartData[i].name
值中,然后自定义工具提示以在其中显示其他字段名称。
This answer gives an overview of how to customize tooltips.从那里,我整理了一个带有折线图的示例,该折线图看起来像默认的工具提示,但字段名称不同:
<ngx-charts-line-chart [legendPosition]="'right'" [results]="chartData" [legend]="true">
<ng-template #tooltipTemplate let-model="model">
<xhtml:div class="area-tooltip-container">
<span class="tooltip-label">{{ formatFieldName(model.series) }} • {{ formatXvalue(model.name) }}</span>
<span class="tooltip-val">{{ formatYvalue(model.value) }}</span>
</xhtml:div>
</ng-template>
<ng-template #seriesTooltipTemplate let-model="model">
<xhtml:div class="area-tooltip-container">
<xhtml:div *ngFor="let tooltipItem of model" class="tooltip-item">
<span class="tooltip-item-color" [style.background-color]="tooltipItem.color">
</span>
{{ formatFieldName(tooltipItem.series) }}: {{ formatYvalue(tooltipItem.value) }}
</xhtml:div>
</xhtml:div>
</ng-template>
</ngx-charts-line-chart>
图例宽度是在其他图表扩展的ngx-charts-chart
组件中计算的。它将宽度设置为图表容器的大约1/6或1/12,具体取决于您的数据类型。无法为此图例输入不同的宽度,因此最简单的解决方案是在自动宽度不适合您时将legendPosition
设置为“ below”。
但是,您不需要使用图表中内置的图例!这是一个更复杂(但更易于调整)的替代方法:在图表中设置[legend]="false"
,然后在图表外部添加一个新的ngx-charts-legend
组件。
您可以为此外部图例输入宽度和高度,也可以将其包装在一个div中,以更灵敏地处理尺寸。使用后一种方法时,我必须将图例的宽度设置为其容器的100%。
要使此解决方案生效,您必须将外部图例激活事件绑定到图表,并在init上设置一些图例输入属性。在包含图表的组件中,您将需要以下内容:
import { ColorHelper } from '@swimlane/ngx-charts';
...
export class ChartContainerComponent implements OnInit {
public activeEntries: any[];
public chartData: { name: string, series: { name: string, value?: string | number }[] }[];
public chartNames: string[];
public colors: ColorHelper;
public colorScheme = { domain: ['#0000FF', '#008000'] }; // Custom color scheme in hex
public legendLabelActivate(item: any): void {
this.activeEntries = [item];
}
public legendLabelDeactivate(item: any): void {
this.activeEntries = [];
}
public ngOnInit(): void {
// Get chartNames
this.chartNames = this.chartData.map((d: any) => d.name);
// Convert hex colors to ColorHelper for consumption by legend
this.colors = new ColorHelper(this.colorScheme, 'ordinal', this.chartNames, this.colorScheme);
}
然后我的模板(图表和图例各占一半的宽度)如下:
<div fxLayout="row">
<div fxFlex="50%">
<ngx-charts-line-chart [legend]="false" [activeEntries]="activeEntries" [results]="chartData" [scheme]="colorScheme"></ngx-charts-line-chart>
</div>
<div fxFlex="50%">
<ngx-charts-legend fxFlex="100%" class="chart-legend" [data]="chartNames" [title]="'Legend Title'" [colors]="colors" (labelActivate)="legendLabelActivate($event)" (labelDeactivate)="legendLabelDeactivate($event)"></ngx-charts-legend>
</div>
</div>
答案 1 :(得分:1)
如果有人受到@Maya Sol的帮助,但想知道与@Ollie相同的事情,请按以下步骤进行。我要添加答案,因为我没有足够的声誉来添加评论。
修改legendLabelActivate函数以执行以下操作:
public legendLabelActivate(item: any): void {
// The if statement is here so both legend
// and chart element mouse over events will work.
if (item.value === undefined) {
this.activeEntries = [item];
} else {
this.activeEntries = [{ name: item.value.name }];
}
}
然后,使用ngx-charts-line-chart和ngx-charts-legend中@Maya Sol ...的示例html执行以下操作:
<div fxLayout="row">
<div fxFlex="50%">
<ngx-charts-line-chart [legend]="false"
[activeEntries]="activeEntries"
(activate)="legendLabelActivate($event)"
(deactivate)="legendLabelDeactivate($event)"
[results]="chartData"
[scheme]="colorScheme">
</ngx-charts-line-chart>
</div>
<div fxFlex="50%">
<ngx-charts-legend fxFlex="100%"
class="chart-legend"
[data]="chartNames"
[title]="'Legend Title'"
[colors]="colors"
[activeEntries]="activeEntries"
(labelActivate)="legendLabelActivate($event)"
(labelDeactivate)="legendLabelDeactivate($event)">
</ngx-charts-legend>
</div>
</div>
答案 2 :(得分:0)
如果您正在寻找纯CSS解决方案,那么这将是很好的起点。在我的应用程序中,我们有一个半屏大小的图表,其中图例切断了较长的标签。
使用12.0.1版。这在很大程度上取决于DOM结构,因此,如果将来的发行版中发生某些更改,则可能无法正常工作。
图表的父容器中必须有额外的空间,否则图例将包裹在图表下方。在此示例中,假定父容器的宽度> 800px,并且图表使用的宽度为620px的固定视图。您还必须关闭组件的视图封装。
在组件文件中:
@Component({
...
encapsulation: ViewEncapsulation.None
})
在CSS文件中:
/* Makes the chart container expand to the full width of the parent container*/
.ngx-charts-outer {
width: 100% !important;
}
/* Sets the legend width to auto expand to fit labels, but caps at 200px */
.chart-legend > div {
width: auto !important;
max-width: 200px;
}
/* Some extra space is needed to offset the way the labels display */
.legend-labels {
width: calc(100% + 10px) !important;
}
/* Wrap the label text if it ends up being longer than the max width */
.legend-label-text {
white-space: normal !important;
width: 100% !important;
}
答案 3 :(得分:0)
这是一个很好的解决方案,可以为图例添加更好的定位。
确保最初将 activeEntries 变量设置为空数组以避免错误。
public activeEntries = [];
答案 4 :(得分:0)
我设法通过在组件的 CSS 文件中手动覆盖其 CSS 类来自定义图例。
通过添加如下所示的 CSS 语法,我设法删除了添加到图例中的 alpha 值。
图例的格式基于一个名为 legend-labels
的类,为了修改它,我将这个类封装在一个 ng-deep
中。
::ng-deep{
.legend-labels{
background: none !important;
}
}