mirror of https://github.com/ghostfolio/ghostfolio
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
148 lines
4.0 KiB
148 lines
4.0 KiB
import { getLocale } from '@ghostfolio/common/helper';
|
|
|
|
import { CommonModule } from '@angular/common';
|
|
import {
|
|
CUSTOM_ELEMENTS_SCHEMA,
|
|
ChangeDetectionStrategy,
|
|
Component,
|
|
Input,
|
|
OnChanges,
|
|
computed,
|
|
input
|
|
} from '@angular/core';
|
|
import { IonIcon } from '@ionic/angular/standalone';
|
|
import { isNumber } from 'lodash';
|
|
import { NgxSkeletonLoaderModule } from 'ngx-skeleton-loader';
|
|
|
|
@Component({
|
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
|
imports: [CommonModule, IonIcon, NgxSkeletonLoaderModule],
|
|
schemas: [CUSTOM_ELEMENTS_SCHEMA],
|
|
selector: 'gf-value',
|
|
styleUrls: ['./value.component.scss'],
|
|
templateUrl: './value.component.html'
|
|
})
|
|
export class GfValueComponent implements OnChanges {
|
|
@Input() colorizeSign = false;
|
|
@Input() deviceType: string;
|
|
@Input() icon = '';
|
|
@Input() isAbsolute = false;
|
|
@Input() isCurrency = false;
|
|
@Input() isDate = false;
|
|
@Input() isPercent = false;
|
|
@Input() locale: string;
|
|
@Input() position = '';
|
|
@Input() size: 'large' | 'medium' | 'small' = 'small';
|
|
@Input() subLabel = '';
|
|
@Input() unit = '';
|
|
@Input() value: number | string = '';
|
|
|
|
public absoluteValue = 0;
|
|
public formattedValue = '';
|
|
public isNumber = false;
|
|
public isString = false;
|
|
public useAbsoluteValue = false;
|
|
|
|
public readonly precision = input<number>();
|
|
|
|
private readonly formatOptions = computed<Intl.NumberFormatOptions>(() => {
|
|
const digits = this.hasPrecision ? this.precision() : 2;
|
|
|
|
return {
|
|
maximumFractionDigits: digits,
|
|
minimumFractionDigits: digits
|
|
};
|
|
});
|
|
|
|
private get hasPrecision(): boolean {
|
|
const precision = this.precision();
|
|
return precision !== undefined && precision >= 0;
|
|
}
|
|
|
|
public ngOnChanges() {
|
|
this.initializeVariables();
|
|
|
|
if (this.value || this.value === 0) {
|
|
if (isNumber(this.value)) {
|
|
this.isNumber = true;
|
|
this.isString = false;
|
|
this.absoluteValue = Math.abs(this.value);
|
|
|
|
if (this.colorizeSign) {
|
|
if (this.isCurrency) {
|
|
try {
|
|
this.formattedValue = this.absoluteValue.toLocaleString(
|
|
this.locale,
|
|
this.formatOptions()
|
|
);
|
|
} catch {}
|
|
} else if (this.isPercent) {
|
|
try {
|
|
this.formattedValue = (this.absoluteValue * 100).toLocaleString(
|
|
this.locale,
|
|
this.formatOptions()
|
|
);
|
|
} catch {}
|
|
}
|
|
} else if (this.isCurrency) {
|
|
try {
|
|
this.formattedValue = this.value?.toLocaleString(
|
|
this.locale,
|
|
this.formatOptions()
|
|
);
|
|
} catch {}
|
|
} else if (this.isPercent) {
|
|
try {
|
|
this.formattedValue = (this.value * 100).toLocaleString(
|
|
this.locale,
|
|
this.formatOptions()
|
|
);
|
|
} catch {}
|
|
} else if (this.hasPrecision) {
|
|
try {
|
|
this.formattedValue = this.value?.toLocaleString(
|
|
this.locale,
|
|
this.formatOptions()
|
|
);
|
|
} catch {}
|
|
} else {
|
|
this.formattedValue = this.value?.toLocaleString(this.locale);
|
|
}
|
|
|
|
if (this.isAbsolute) {
|
|
// Remove algebraic sign
|
|
this.formattedValue = this.formattedValue.replace(/^-/, '');
|
|
}
|
|
} else {
|
|
this.isNumber = false;
|
|
this.isString = true;
|
|
|
|
if (this.isDate) {
|
|
this.formattedValue = new Date(this.value).toLocaleDateString(
|
|
this.locale,
|
|
{
|
|
day: '2-digit',
|
|
month: '2-digit',
|
|
year: this.deviceType === 'mobile' ? '2-digit' : 'numeric'
|
|
}
|
|
);
|
|
} else {
|
|
this.formattedValue = this.value;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.formattedValue === '0.00') {
|
|
this.useAbsoluteValue = true;
|
|
}
|
|
}
|
|
|
|
private initializeVariables() {
|
|
this.absoluteValue = 0;
|
|
this.formattedValue = '';
|
|
this.isNumber = false;
|
|
this.isString = false;
|
|
this.locale = this.locale || getLocale();
|
|
this.useAbsoluteValue = false;
|
|
}
|
|
}
|
|
|