mirror of
https://github.com/Alfresco/alfresco-ng2-components.git
synced 2026-04-16 22:24:49 +00:00
AAE-39940 Localisation fixes (#11359)
This commit is contained in:
@@ -31,9 +31,23 @@ describe('DataTableCellComponent', () => {
|
||||
let testingUtils: UnitTestingUtils;
|
||||
|
||||
const renderTextCell = (value: string, tooltip: string) => {
|
||||
// Set up mock data if not already set
|
||||
if (!component.data) {
|
||||
component.data = {
|
||||
getValue: () => value
|
||||
} as any;
|
||||
}
|
||||
if (!component.row) {
|
||||
component.row = { id: '1', getValue: () => value } as any;
|
||||
}
|
||||
if (!component.column) {
|
||||
component.column = { key: 'text' } as any;
|
||||
}
|
||||
|
||||
component.value$ = new BehaviorSubject<string>(value);
|
||||
component.tooltip = tooltip;
|
||||
|
||||
component.ngOnInit();
|
||||
fixture.detectChanges();
|
||||
};
|
||||
|
||||
@@ -136,7 +150,7 @@ describe('DataTableCellComponent', () => {
|
||||
component.row = row;
|
||||
|
||||
expect(() => fixture.detectChanges()).not.toThrow();
|
||||
expect(component.computedTitle).toBe('');
|
||||
expect(component.title()).toBe('');
|
||||
expect(testingUtils.getByCSS('span').nativeElement.title).toBe('');
|
||||
});
|
||||
});
|
||||
|
||||
@@ -15,7 +15,7 @@
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { ChangeDetectionStrategy, Component, DestroyRef, inject, Input, OnInit, ViewEncapsulation } from '@angular/core';
|
||||
import { ChangeDetectionStrategy, Component, DestroyRef, inject, Input, OnInit, ViewEncapsulation, signal, computed } from '@angular/core';
|
||||
import { DataColumn } from '../../data/data-column.model';
|
||||
import { DataRow } from '../../data/data-row.model';
|
||||
import { DataTableAdapter } from '../../data/datatable-adapter';
|
||||
@@ -31,22 +31,21 @@ import { TruncatePipe } from '../../../pipes/truncate.pipe';
|
||||
imports: [CommonModule, ClipboardDirective, TruncatePipe],
|
||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||
template: `
|
||||
<ng-container>
|
||||
@let value = value$ | async;
|
||||
@let displayValue = column?.maxTextLength ? (value | truncate: column?.maxTextLength) : value;
|
||||
|
||||
@if (copyContent) {
|
||||
<span
|
||||
*ngIf="copyContent; else defaultCell"
|
||||
adf-clipboard="CLIPBOARD.CLICK_TO_COPY"
|
||||
[clipboard-notification]="'CLIPBOARD.SUCCESS_COPY'"
|
||||
[attr.aria-label]="value$ | async"
|
||||
[title]="tooltip ? tooltip : computedTitle"
|
||||
[attr.aria-label]="value"
|
||||
[title]="title()"
|
||||
class="adf-datatable-cell-value"
|
||||
>{{ column?.maxTextLength ? (value$ | async | truncate: column?.maxTextLength) : (value$ | async) }}</span
|
||||
>{{ displayValue }}</span
|
||||
>
|
||||
</ng-container>
|
||||
<ng-template #defaultCell>
|
||||
<span [title]="tooltip ? tooltip : computedTitle" class="adf-datatable-cell-value">{{
|
||||
column?.maxTextLength ? (value$ | async | truncate: column?.maxTextLength) : (value$ | async)
|
||||
}}</span>
|
||||
</ng-template>
|
||||
} @else {
|
||||
<span [title]="title()" class="adf-datatable-cell-value">{{ displayValue }}</span>
|
||||
}
|
||||
`,
|
||||
encapsulation: ViewEncapsulation.None,
|
||||
host: { class: 'adf-datatable-content-cell' }
|
||||
@@ -79,7 +78,12 @@ export class DataTableCellComponent implements OnInit {
|
||||
protected destroyRef = inject(DestroyRef);
|
||||
protected dataTableService = inject(DataTableService, { optional: true });
|
||||
value$ = new BehaviorSubject<any>('');
|
||||
computedTitle: string = '';
|
||||
|
||||
// Signal to track the raw computed title (without tooltip override)
|
||||
protected rawComputedTitle = signal<string>('');
|
||||
|
||||
// Computed signal that automatically combines tooltip input with computed title
|
||||
title = computed(() => this.tooltip || this.rawComputedTitle());
|
||||
|
||||
ngOnInit() {
|
||||
this.updateValue();
|
||||
@@ -90,7 +94,7 @@ export class DataTableCellComponent implements OnInit {
|
||||
if (this.column?.key && this.row && this.data) {
|
||||
const value = this.data.getValue(this.row, this.column, this.resolverFn);
|
||||
this.value$.next(value);
|
||||
this.computedTitle = this.computeTitle(value);
|
||||
this.rawComputedTitle.set(this.computeTitle(value));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -113,10 +117,15 @@ export class DataTableCellComponent implements OnInit {
|
||||
return path.split('.').reduce((source, key) => (source ? source[key] : ''), obj);
|
||||
}
|
||||
|
||||
private computeTitle(value: string): string {
|
||||
if (this.tooltip) {
|
||||
return this.tooltip;
|
||||
}
|
||||
/**
|
||||
* Computes the title/tooltip for the cell based on the value.
|
||||
* Override this in derived classes to provide custom tooltip logic.
|
||||
* Note: The tooltip input always takes precedence (handled by title signal).
|
||||
*
|
||||
* @param value - The cell value to compute the title for
|
||||
* @returns The computed title string, or empty string if no title should be shown
|
||||
*/
|
||||
protected computeTitle(value: string): string {
|
||||
const rawValue = value;
|
||||
const max = this.column?.maxTextLength;
|
||||
|
||||
|
||||
@@ -1,15 +1,19 @@
|
||||
<ng-container *ngIf="value$ | async as date">
|
||||
<span
|
||||
[title]="tooltip | adfLocalizedDate: config.tooltipFormat: config.locale"
|
||||
class="adf-datatable-cell-value"
|
||||
*ngIf="config.format === 'timeAgo'; else standard_date">
|
||||
{{ date | adfTimeAgo: config.locale }}
|
||||
</span>
|
||||
<ng-template #standard_date>
|
||||
<span
|
||||
class="adf-datatable-cell-value"
|
||||
[title]="tooltip | adfLocalizedDate: config.tooltipFormat: config.locale">
|
||||
@let date = value$ | async;
|
||||
|
||||
@if (date) {
|
||||
<span [title]="title()" class="adf-datatable-cell-value">
|
||||
@if (config.format === 'timeAgo') {
|
||||
@if (config.locale) {
|
||||
{{ date | adfTimeAgo: config.locale }}
|
||||
} @else {
|
||||
{{ date | adfTimeAgo }}
|
||||
}
|
||||
} @else {
|
||||
@if (config.locale) {
|
||||
{{ date | adfLocalizedDate: config.format: config.locale }}
|
||||
</span>
|
||||
</ng-template>
|
||||
</ng-container>
|
||||
} @else {
|
||||
{{ date | adfLocalizedDate: config.format }}
|
||||
}
|
||||
}
|
||||
</span>
|
||||
}
|
||||
|
||||
@@ -31,18 +31,33 @@ let fixture: ComponentFixture<DateCellComponent>;
|
||||
let testingUtils: UnitTestingUtils;
|
||||
|
||||
let mockDate;
|
||||
let mockTooltip = '';
|
||||
const mockColumn: DataColumn = {
|
||||
key: 'mock-date',
|
||||
type: 'date',
|
||||
format: 'full'
|
||||
};
|
||||
|
||||
const renderDateCell = (dateConfig: DateConfig, value: number | string | Date, tooltip: string) => {
|
||||
const renderDateCell = (dateConfig: DateConfig, value: number | string | Date, tooltip?: string) => {
|
||||
// Set up mock data if not already set
|
||||
if (!component.data) {
|
||||
component.data = {
|
||||
getValue: () => value
|
||||
} as any;
|
||||
}
|
||||
if (!component.row) {
|
||||
component.row = { id: '1', getValue: () => value } as any;
|
||||
}
|
||||
if (!component.column) {
|
||||
component.column = { key: 'date' } as any;
|
||||
}
|
||||
|
||||
component.value$ = new BehaviorSubject<number | string | Date>(value);
|
||||
component.dateConfig = dateConfig;
|
||||
component.tooltip = tooltip;
|
||||
if (tooltip) {
|
||||
component.tooltip = tooltip;
|
||||
}
|
||||
|
||||
component.ngOnInit();
|
||||
fixture.detectChanges();
|
||||
};
|
||||
|
||||
@@ -85,7 +100,6 @@ describe('DateCellComponent', () => {
|
||||
registerLocaleData(localePL);
|
||||
configureTestingModule([]);
|
||||
mockDate = new Date('2023-10-25T00:00:00');
|
||||
mockTooltip = mockDate.toISOString();
|
||||
});
|
||||
|
||||
it('should set default date config', () => {
|
||||
@@ -103,7 +117,7 @@ describe('DateCellComponent', () => {
|
||||
const expectedDate = '10/25/23, 12:00 AM';
|
||||
const expectedTooltip = '10/25/23';
|
||||
|
||||
renderDateCell(mockDateConfig, mockDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
checkDisplayedTooltip(expectedTooltip);
|
||||
});
|
||||
@@ -113,7 +127,7 @@ describe('DateCellComponent', () => {
|
||||
const expectedDate = 'Oct 25, 2023';
|
||||
const expectedTooltip = 'October 25, 2023 at 12:00:00 AM GMT+0';
|
||||
|
||||
renderDateCell(mockDateConfig, mockDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
checkDisplayedTooltip(expectedTooltip);
|
||||
|
||||
@@ -131,7 +145,7 @@ describe('DateCellComponent', () => {
|
||||
const expectedDate = 'Oct 25, 2023, 12:00:00 AM';
|
||||
const expectedTooltip = expectedDate;
|
||||
|
||||
renderDateCell(mockDateConfig, mockDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
checkDisplayedTooltip(expectedTooltip);
|
||||
});
|
||||
@@ -146,7 +160,7 @@ describe('DateCellComponent', () => {
|
||||
|
||||
const expectedDate = '1 day ago';
|
||||
|
||||
renderDateCell(mockDateConfig, yesterday, mockTooltip);
|
||||
renderDateCell(mockDateConfig, yesterday);
|
||||
checkDisplayedDate(expectedDate);
|
||||
});
|
||||
|
||||
@@ -158,7 +172,7 @@ describe('DateCellComponent', () => {
|
||||
yesterday.setDate(today.getDate() - 1);
|
||||
|
||||
const expectedDate = '1 day ago';
|
||||
renderDateCell(mockDateConfig, yesterday, mockTooltip);
|
||||
renderDateCell(mockDateConfig, yesterday);
|
||||
checkDisplayedDate(expectedDate);
|
||||
});
|
||||
//eslint-disable-next-line
|
||||
@@ -170,7 +184,7 @@ describe('DateCellComponent', () => {
|
||||
|
||||
const expectedDate = 'Wednesday, October 25, 2023 at 12:00:00 AM GMT+00:00';
|
||||
|
||||
renderDateCell(mockDateConfig, mockDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
});
|
||||
|
||||
@@ -182,7 +196,7 @@ describe('DateCellComponent', () => {
|
||||
|
||||
const expectedDate = '10/25/23, 12:00 AM';
|
||||
|
||||
renderDateCell(mockDateConfig, mockDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
});
|
||||
|
||||
@@ -195,7 +209,7 @@ describe('DateCellComponent', () => {
|
||||
|
||||
const expectedDate = '10/25/23, 12:00 AM';
|
||||
|
||||
renderDateCell(mockDateConfig, mockStringDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockStringDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
});
|
||||
|
||||
@@ -208,7 +222,7 @@ describe('DateCellComponent', () => {
|
||||
|
||||
const expectedDate = '10/25/23, 12:00 AM';
|
||||
|
||||
renderDateCell(mockDateConfig, mockTimestamp, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockTimestamp);
|
||||
checkDisplayedDate(expectedDate);
|
||||
});
|
||||
});
|
||||
@@ -227,7 +241,7 @@ describe('DateCellComponent locale', () => {
|
||||
const expectedDate = '25.10.2023, 00:00';
|
||||
const expectedTooltip = '25 paź 2023, 00:00:00';
|
||||
|
||||
renderDateCell(mockDateConfig, mockDate, mockTooltip);
|
||||
renderDateCell(mockDateConfig, mockDate);
|
||||
checkDisplayedDate(expectedDate);
|
||||
checkDisplayedTooltip(expectedTooltip);
|
||||
});
|
||||
|
||||
@@ -15,20 +15,23 @@
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { ChangeDetectionStrategy, Component, Input, OnInit, ViewEncapsulation, inject } from '@angular/core';
|
||||
import { ChangeDetectionStrategy, Component, Input, OnInit, ViewEncapsulation, inject, ChangeDetectorRef } from '@angular/core';
|
||||
import { DataTableCellComponent } from '../datatable-cell/datatable-cell.component';
|
||||
import { AppConfigService } from '../../../app-config/app-config.service';
|
||||
import { DateConfig } from '../../data/data-column.model';
|
||||
import { CommonModule } from '@angular/common';
|
||||
import { LocalizedDatePipe, TimeAgoPipe } from '../../../pipes';
|
||||
import { AsyncPipe } from '@angular/common';
|
||||
import { UserPreferencesService, UserPreferenceValues } from '../../../common/services/user-preferences.service';
|
||||
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
||||
|
||||
@Component({
|
||||
imports: [CommonModule, LocalizedDatePipe, TimeAgoPipe],
|
||||
imports: [LocalizedDatePipe, TimeAgoPipe, AsyncPipe],
|
||||
selector: 'adf-date-cell',
|
||||
templateUrl: './date-cell.component.html',
|
||||
encapsulation: ViewEncapsulation.None,
|
||||
host: { class: 'adf-datatable-content-cell' },
|
||||
changeDetection: ChangeDetectionStrategy.OnPush
|
||||
changeDetection: ChangeDetectionStrategy.OnPush,
|
||||
providers: [LocalizedDatePipe]
|
||||
})
|
||||
export class DateCellComponent extends DataTableCellComponent implements OnInit {
|
||||
@Input()
|
||||
@@ -37,6 +40,11 @@ export class DateCellComponent extends DataTableCellComponent implements OnInit
|
||||
config: DateConfig = {};
|
||||
|
||||
private readonly appConfig: AppConfigService = inject(AppConfigService);
|
||||
private readonly localizedDatePipe: LocalizedDatePipe = inject(LocalizedDatePipe);
|
||||
private readonly userPreferencesService: UserPreferencesService = inject(UserPreferencesService);
|
||||
private readonly cdr: ChangeDetectorRef = inject(ChangeDetectorRef);
|
||||
|
||||
private userLocale: string = 'en';
|
||||
|
||||
readonly defaultDateConfig: DateConfig = {
|
||||
format: 'medium',
|
||||
@@ -45,8 +53,26 @@ export class DateCellComponent extends DataTableCellComponent implements OnInit
|
||||
};
|
||||
|
||||
ngOnInit(): void {
|
||||
super.ngOnInit();
|
||||
// Subscribe to locale changes
|
||||
this.userPreferencesService
|
||||
.select(UserPreferenceValues.Locale)
|
||||
.pipe(takeUntilDestroyed(this.destroyRef))
|
||||
.subscribe((locale) => {
|
||||
this.userLocale = locale || 'en';
|
||||
this.setConfig();
|
||||
this.updateValue(); // Recalculate computedTitle with new locale
|
||||
this.cdr.markForCheck();
|
||||
});
|
||||
|
||||
this.setConfig();
|
||||
super.ngOnInit();
|
||||
}
|
||||
|
||||
protected override computeTitle(value: any): string {
|
||||
if (value) {
|
||||
return this.localizedDatePipe.transform(value, this.config.tooltipFormat, this.config.locale) || '';
|
||||
}
|
||||
return '';
|
||||
}
|
||||
|
||||
private setConfig(): void {
|
||||
@@ -60,13 +86,25 @@ export class DateCellComponent extends DataTableCellComponent implements OnInit
|
||||
private setCustomConfig(): void {
|
||||
this.config.format = this.dateConfig?.format || this.getDefaultFormat();
|
||||
this.config.tooltipFormat = this.dateConfig?.tooltipFormat || this.getDefaultTooltipFormat();
|
||||
this.config.locale = this.dateConfig?.locale || this.getDefaultLocale();
|
||||
this.config.locale = this.normalizeLocale(this.dateConfig?.locale || this.getDefaultLocale());
|
||||
}
|
||||
|
||||
private setDefaultConfig(): void {
|
||||
this.config.format = this.getDefaultFormat();
|
||||
this.config.tooltipFormat = this.getDefaultTooltipFormat();
|
||||
this.config.locale = this.getDefaultLocale();
|
||||
this.config.locale = this.normalizeLocale(this.getDefaultLocale());
|
||||
}
|
||||
|
||||
private normalizeLocale(locale: string): string {
|
||||
if (!locale) {
|
||||
return locale;
|
||||
}
|
||||
// Extract language code from locale like 'fr-FR' -> 'fr', 'en-US' -> 'en'
|
||||
// but keep special cases like 'pt-BR' and 'zh-CN' intact
|
||||
if (locale === 'pt-BR' || locale === 'zh-CN') {
|
||||
return locale;
|
||||
}
|
||||
return locale.split('-')[0];
|
||||
}
|
||||
|
||||
private getDefaultFormat(): string {
|
||||
@@ -74,7 +112,9 @@ export class DateCellComponent extends DataTableCellComponent implements OnInit
|
||||
}
|
||||
|
||||
private getDefaultLocale(): string {
|
||||
return this.getAppConfigPropertyValue('dateValues.defaultLocale', this.defaultDateConfig.locale);
|
||||
// Always use the user locale from UserPreferencesService
|
||||
// This is kept in sync via subscription and reflects the user's current locale choice
|
||||
return this.userLocale;
|
||||
}
|
||||
|
||||
private getDefaultTooltipFormat(): string {
|
||||
|
||||
@@ -15,24 +15,33 @@
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
import { Component, OnInit, ViewEncapsulation } from '@angular/core';
|
||||
import { Component, OnInit, ViewEncapsulation, inject } from '@angular/core';
|
||||
import { DataTableCellComponent } from '../datatable-cell/datatable-cell.component';
|
||||
import { CommonModule } from '@angular/common';
|
||||
import { FileSizePipe } from '../../../pipes';
|
||||
import { AsyncPipe } from '@angular/common';
|
||||
|
||||
@Component({
|
||||
selector: 'adf-filesize-cell',
|
||||
imports: [CommonModule, FileSizePipe],
|
||||
imports: [FileSizePipe, AsyncPipe],
|
||||
template: `
|
||||
<ng-container *ngIf="value$ | async | adfFileSize as fileSize">
|
||||
<span [title]="tooltip">{{ fileSize }}</span>
|
||||
</ng-container>
|
||||
@let value = value$ | async;
|
||||
<span [title]="title()" class="adf-datatable-cell-value">{{ value | adfFileSize }}</span>
|
||||
`,
|
||||
encapsulation: ViewEncapsulation.None,
|
||||
host: { class: 'adf-filesize-cell' }
|
||||
host: { class: 'adf-filesize-cell' },
|
||||
providers: [FileSizePipe]
|
||||
})
|
||||
export class FileSizeCellComponent extends DataTableCellComponent implements OnInit {
|
||||
private readonly fileSizePipe = inject(FileSizePipe);
|
||||
|
||||
ngOnInit(): void {
|
||||
super.ngOnInit();
|
||||
}
|
||||
|
||||
protected override computeTitle(value: any): string {
|
||||
if (value != null) {
|
||||
return this.fileSizePipe.transform(value);
|
||||
}
|
||||
return '';
|
||||
}
|
||||
}
|
||||
|
||||
@@ -20,12 +20,13 @@ import { AppConfigService } from '../app-config/app-config.service';
|
||||
import { UserPreferencesService, UserPreferenceValues } from '../common/services/user-preferences.service';
|
||||
import { DatePipe } from '@angular/common';
|
||||
import { differenceInDays, formatDistance } from 'date-fns';
|
||||
import * as Locales from 'date-fns/locale';
|
||||
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
||||
import { DateFnsUtils } from '../common/utils/date-fns-utils';
|
||||
|
||||
@Pipe({
|
||||
standalone: true,
|
||||
name: 'adfTimeAgo'
|
||||
name: 'adfTimeAgo',
|
||||
pure: false
|
||||
})
|
||||
export class TimeAgoPipe implements PipeTransform {
|
||||
static DEFAULT_LOCALE = 'en-US';
|
||||
@@ -34,7 +35,10 @@ export class TimeAgoPipe implements PipeTransform {
|
||||
defaultLocale: string;
|
||||
defaultDateTimeFormat: string;
|
||||
|
||||
constructor(public userPreferenceService: UserPreferencesService, public appConfig: AppConfigService) {
|
||||
constructor(
|
||||
public userPreferenceService: UserPreferencesService,
|
||||
public appConfig: AppConfigService
|
||||
) {
|
||||
this.userPreferenceService
|
||||
.select(UserPreferenceValues.Locale)
|
||||
.pipe(takeUntilDestroyed())
|
||||
@@ -52,7 +56,8 @@ export class TimeAgoPipe implements PipeTransform {
|
||||
const datePipe: DatePipe = new DatePipe(actualLocale);
|
||||
return datePipe.transform(value, this.defaultDateTimeFormat);
|
||||
} else {
|
||||
return formatDistance(new Date(value), new Date(), { addSuffix: true, locale: Locales[actualLocale] });
|
||||
const dateFnsLocale = DateFnsUtils.getLocaleFromString(actualLocale);
|
||||
return formatDistance(new Date(value), new Date(), { addSuffix: true, locale: dateFnsLocale });
|
||||
}
|
||||
}
|
||||
return '';
|
||||
|
||||
Reference in New Issue
Block a user