mirror of
https://gitlab.com/openstapps/openstapps.git
synced 2025-12-15 18:56:18 +00:00
feat: add service and pipe for core translator
This commit is contained in:
82
src/app/translation/thing-translator.parser.ts
Normal file
82
src/app/translation/thing-translator.parser.ts
Normal file
@@ -0,0 +1,82 @@
|
||||
/*
|
||||
* Copyright (C) 2020-2021 StApps
|
||||
* This program is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License as published by the Free
|
||||
* Software Foundation, version 3.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along with
|
||||
* this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import {Injectable} from '@angular/core';
|
||||
import {isDefined} from '@ngx-translate/core/lib/util';
|
||||
|
||||
// tslint:disable: member-ordering prefer-function-over-method completed-docs
|
||||
|
||||
export abstract class TranslateParser {
|
||||
/**
|
||||
* Gets a value from an object by composed key
|
||||
* parser.getValue({ key1: { keyA: 'valueI' }}, 'key1.keyA') ==> 'valueI'
|
||||
*/
|
||||
abstract getValue(target: unknown, key: string): string;
|
||||
|
||||
/**
|
||||
* TODO
|
||||
*/
|
||||
abstract getValueFromKeyPath(instance: object, keypath: string): unknown;
|
||||
}
|
||||
|
||||
@Injectable()
|
||||
export class TranslateDefaultParser extends TranslateParser {
|
||||
|
||||
/**
|
||||
* TODO
|
||||
*
|
||||
*/
|
||||
getValue(target: unknown, key: string): string {
|
||||
const keys = typeof key === 'string' ? key.split('.') : [key];
|
||||
let aKey = '';
|
||||
// tslint:disable-next-line: no-any
|
||||
let newTarget = target as any;
|
||||
do {
|
||||
aKey += keys.shift();
|
||||
if (isDefined(newTarget) && isDefined(newTarget[aKey]) && (typeof newTarget[aKey] === 'object' || keys.length === 0)) {
|
||||
newTarget = newTarget[aKey];
|
||||
aKey = '';
|
||||
} else if (keys.length === 0) {
|
||||
newTarget = undefined;
|
||||
} else {
|
||||
aKey += '.';
|
||||
}
|
||||
} while (keys.length > 0);
|
||||
|
||||
return newTarget;
|
||||
}
|
||||
|
||||
|
||||
getValueFromKeyPath(instance: object, keypath: string): unknown {
|
||||
// keypath = aproperty[0].anotherproperty["arrayproperty"][42].finalproperty
|
||||
let path = keypath.replace(/(?:\"|\')'"/gmi, '');
|
||||
// path = aproperty[0].anotherproperty[.arrayproperty.][42].finalproperty
|
||||
path = path.replace(/(?:\[|\])/gmi, '.');
|
||||
// path = aproperty.0..anotherproperty..arrayproperty...42..finalproperty
|
||||
path = path.replace(/\.{2,}/gmi, '.');
|
||||
// path = aproperty.0.anotherproperty.arrayproperty.42.finalproperty
|
||||
|
||||
const keypathChain = keypath.split('.');
|
||||
|
||||
// tslint:disable-next-line: no-any
|
||||
let property = instance as any;
|
||||
|
||||
for(const key of keypathChain) {
|
||||
property = property[key] ?? undefined;
|
||||
}
|
||||
|
||||
return property;
|
||||
}
|
||||
}
|
||||
91
src/app/translation/thing-translator.pipe.ts
Normal file
91
src/app/translation/thing-translator.pipe.ts
Normal file
@@ -0,0 +1,91 @@
|
||||
/*
|
||||
* Copyright (C) 2020-2021 StApps
|
||||
* This program is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License as published by the Free
|
||||
* Software Foundation, version 3.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along with
|
||||
* this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import {Injectable, OnDestroy, Pipe, PipeTransform} from '@angular/core';
|
||||
import {LangChangeEvent, TranslateService} from '@ngx-translate/core';
|
||||
import {isThing, SCThings} from '@openstapps/core';
|
||||
import {Subscription} from 'rxjs';
|
||||
import {ThingTranslatorService} from './thing-translator.service';
|
||||
|
||||
// tslint:disable: member-ordering prefer-function-over-method completed-docs
|
||||
|
||||
|
||||
@Injectable()
|
||||
@Pipe({
|
||||
name: 'translate',
|
||||
pure: false, // required to update the value when the promise is resolved
|
||||
})
|
||||
export class ThingTranslatePipe implements PipeTransform, OnDestroy {
|
||||
value = '';
|
||||
lastKey?: string;
|
||||
lastThing: SCThings;
|
||||
onLangChange: Subscription;
|
||||
|
||||
constructor(private readonly translate: TranslateService,
|
||||
// private readonly _ref: ChangeDetectorRef,
|
||||
private readonly thingTranslate: ThingTranslatorService) {
|
||||
}
|
||||
|
||||
updateValue(key: string, thing: SCThings): void {
|
||||
this.value = String(this.thingTranslate.get(thing, key));
|
||||
}
|
||||
|
||||
transform(query: string, thing: SCThings): string {
|
||||
if (typeof query !== 'string' || query.length <= 0) {
|
||||
return query;
|
||||
}
|
||||
|
||||
if (!isThing(thing)){
|
||||
throw new SyntaxError(`Wrong parameter in ThingTranslatePipe. Expected a valid SCThing, received: ${thing}`);
|
||||
}
|
||||
|
||||
// store the query, in case it changes
|
||||
this.lastKey = query;
|
||||
|
||||
// store the params, in case they change
|
||||
this.lastThing = thing;
|
||||
|
||||
// set the value
|
||||
this.updateValue(query, thing);
|
||||
|
||||
// if there is a subscription to onLangChange, clean it
|
||||
this._dispose();
|
||||
|
||||
// subscribe to onLangChange event, in case the language changes
|
||||
if (typeof this.onLangChange === 'undefined' || this.onLangChange.closed ) {
|
||||
this.onLangChange = this.translate.onLangChange.subscribe((_event: LangChangeEvent) => {
|
||||
if (typeof this.lastKey === 'string') {
|
||||
this.lastKey = undefined; // we want to make sure it doesn't return the same value until it's been updated
|
||||
this.updateValue(query, thing);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
return this.value;
|
||||
}
|
||||
|
||||
/**
|
||||
* Clean any existing subscription to change events
|
||||
*/
|
||||
private _dispose(): void {
|
||||
if (this.onLangChange?.closed) {
|
||||
this.onLangChange?.unsubscribe();
|
||||
}
|
||||
}
|
||||
|
||||
ngOnDestroy(): void {
|
||||
this._dispose();
|
||||
}
|
||||
}
|
||||
83
src/app/translation/thing-translator.service.ts
Normal file
83
src/app/translation/thing-translator.service.ts
Normal file
@@ -0,0 +1,83 @@
|
||||
/*
|
||||
* Copyright (C) 2020-2021 StApps
|
||||
* This program is free software: you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License as published by the Free
|
||||
* Software Foundation, version 3.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||||
* more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License along with
|
||||
* this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import {Injectable, OnDestroy} from '@angular/core';
|
||||
import {LangChangeEvent, TranslateService} from '@ngx-translate/core';
|
||||
import {isDefined} from '@ngx-translate/core/lib/util';
|
||||
import {SCLanguage, SCThings, SCThingTranslator, SCTranslations} from '@openstapps/core';
|
||||
import {Subscription} from 'rxjs';
|
||||
import {TranslateParser} from './thing-translator.parser';
|
||||
|
||||
// export const DEFAULT_LANGUAGE = new InjectionToken<string>('DEFAULT_LANGUAGE');
|
||||
|
||||
// tslint:disable: member-ordering prefer-function-over-method newline-per-chained-call completed-docs
|
||||
|
||||
@Injectable({
|
||||
providedIn: 'root',
|
||||
})
|
||||
export class ThingTranslatorService implements OnDestroy {
|
||||
|
||||
onLangChange: Subscription;
|
||||
translator: SCThingTranslator;
|
||||
|
||||
/**
|
||||
*
|
||||
* @param translateService Instance of Angular TranslateService
|
||||
* @param parser An instance of the parser currently used
|
||||
* @param language TODO
|
||||
*/
|
||||
constructor(private readonly translateService: TranslateService,
|
||||
public parser: TranslateParser,
|
||||
language?: keyof SCTranslations<SCLanguage>) {
|
||||
|
||||
this.translator = new SCThingTranslator(language ?? this.translateService.currentLang as keyof SCTranslations<SCLanguage>);
|
||||
|
||||
/** set the default language from configuration */
|
||||
this.onLangChange = this.translateService.onLangChange.subscribe((event: LangChangeEvent) => {
|
||||
this.translator.language = event.lang as keyof SCTranslations<SCLanguage>;
|
||||
});
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the parsed result of the translations
|
||||
*/
|
||||
// tslint:disable-next-line: no-any
|
||||
getParsedResult(target: object, key: string): any {
|
||||
return this.parser.getValueFromKeyPath(target, key);
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the translated value of a key (or an array of keys)
|
||||
* @returns the translated key, or an object of translated keys
|
||||
*/
|
||||
public get(thing: SCThings, keyPath: string | string[]): string | number | boolean {
|
||||
if (!isDefined(keyPath) || keyPath.length === 0) {
|
||||
throw new Error(`Parameter "key" required`);
|
||||
}
|
||||
if (keyPath instanceof Array) {
|
||||
return this.getParsedResult(thing, keyPath.join('.'));
|
||||
}
|
||||
|
||||
return this.getParsedResult(thing, keyPath);
|
||||
}
|
||||
|
||||
// tslint:disable-next-line: completed-docs
|
||||
ngOnDestroy() {
|
||||
if (!this.onLangChange.closed) {
|
||||
this.onLangChange.unsubscribe();
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user