UNPKG

angular-l10n

Version:

Angular library to translate texts, dates and numbers

165 lines 21.3 kB
import { Directive, Input, ElementRef, Renderer2, inject } from '@angular/core'; import { Subject } from 'rxjs'; import { takeUntil } from 'rxjs/operators'; import { getTargetNode } from './bfs'; import { L10nTranslationService } from '../services/l10n-translation.service'; import * as i0 from "@angular/core"; export class L10nDirective { constructor() { this.el = inject(ElementRef); this.renderer = inject(Renderer2); this.translation = inject(L10nTranslationService); this.attributes = []; this.destroy = new Subject(); } set innerHTML(content) { // Handle TrustedHTML this.content = content.toString(); } ngAfterViewInit() { if (this.el && this.el.nativeElement) { this.element = this.el.nativeElement; this.renderNode = getTargetNode(this.el.nativeElement); this.text = this.getText(); this.attributes = this.getAttributes(); this.addTextListener(); if (this.language) { this.replaceText(); this.replaceAttributes(); } else { this.addTranslationListener(); } } } ngOnChanges() { if (this.text) { if (this.nodeValue == null || this.nodeValue === '') { if (this.value) { this.text = this.value; } else if (this.content) { this.text = this.content; } } this.replaceText(); } if (this.attributes && this.attributes.length > 0) { this.replaceAttributes(); } } ngOnDestroy() { this.destroy.next(true); this.removeTextListener(); } getText() { let text = ''; if (this.element && this.element.childNodes.length > 0) { text = this.getNodeValue(); } else if (this.value) { text = this.value; } else if (this.content) { text = this.content; } return text; } getNodeValue() { this.nodeValue = this.renderNode != null && this.renderNode.nodeValue != null ? this.renderNode.nodeValue : ''; return this.nodeValue ? this.nodeValue.trim() : ''; } getAttributes() { const attributes = []; if (this.element && this.element.attributes) { for (const attr of Array.from(this.element.attributes)) { if (attr && attr.name) { const [, name = ''] = attr.name.match(/^l10n-(.+)$/) || []; if (name) { const targetAttr = Array.from(this.element.attributes).find(a => a.name === name); if (targetAttr) attributes.push({ name: targetAttr.name, value: targetAttr.value }); } } } } return attributes; } addTextListener() { if (typeof MutationObserver !== 'undefined') { this.textObserver = new MutationObserver(() => { if (this.element) { this.renderNode = getTargetNode(this.element); this.text = this.getText(); this.replaceText(); } }); if (this.renderNode) { this.textObserver.observe(this.renderNode, { subtree: true, characterData: true }); } } } removeTextListener() { if (this.textObserver) { this.textObserver.disconnect(); } } addTranslationListener() { this.translation.onChange().pipe(takeUntil(this.destroy)).subscribe({ next: () => { this.replaceText(); this.replaceAttributes(); } }); } replaceText() { if (this.text) { this.setText(this.getValue(this.text)); } } replaceAttributes() { if (this.attributes.length > 0) { this.setAttributes(this.getAttributesValues()); } } setText(value) { if (value) { if (this.nodeValue && this.text) { this.removeTextListener(); this.renderer.setValue(this.renderNode, this.nodeValue.replace(this.text, value)); this.addTextListener(); } else if (this.value) { this.renderer.setAttribute(this.element, 'value', value); } else if (this.content) { this.renderer.setProperty(this.element, 'innerHTML', value); } } } setAttributes(data) { for (const attr of this.attributes) { this.renderer.setAttribute(this.element, attr.name, data[attr.value]); } } getAttributesValues() { const values = this.attributes.map(attr => attr.value); const data = {}; for (const value of values) { data[value] = this.getValue(value); } return data; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.4", ngImport: i0, type: L10nDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.0.4", type: L10nDirective, inputs: { value: "value", innerHTML: "innerHTML", language: "language" }, usesOnChanges: true, ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.4", ngImport: i0, type: L10nDirective, decorators: [{ type: Directive }], propDecorators: { value: [{ type: Input }], innerHTML: [{ type: Input }], language: [{ type: Input }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"l10n-directive.js","sourceRoot":"","sources":["../../../../../projects/angular-l10n/src/lib/models/l10n-directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, Input, AfterViewInit, OnChanges, OnDestroy, ElementRef, Renderer2, inject, Injectable } from '@angular/core';\r\nimport { Subject } from 'rxjs';\r\nimport { takeUntil } from 'rxjs/operators';\r\n\r\nimport { getTargetNode } from './bfs';\r\nimport { L10nTranslationService } from '../services/l10n-translation.service';\r\n\r\n@Directive()\r\nexport abstract class L10nDirective implements AfterViewInit, OnChanges, OnDestroy {\r\n\r\n    @Input() public value?: string;\r\n\r\n    @Input() set innerHTML(content: any) {\r\n        // Handle TrustedHTML\r\n        this.content = content.toString();\r\n    }\r\n\r\n    @Input() public language?: string;\r\n\r\n    protected el = inject(ElementRef);\r\n    protected renderer = inject(Renderer2);\r\n    protected translation = inject(L10nTranslationService);\r\n\r\n    private content?: string;\r\n\r\n    private text?: string;\r\n    private attributes: any[] = [];\r\n\r\n    private element?: HTMLElement;\r\n    private renderNode?: HTMLElement;\r\n    private nodeValue?: string;\r\n\r\n    private textObserver?: MutationObserver;\r\n\r\n    private destroy = new Subject<boolean>();\r\n\r\n    public ngAfterViewInit(): void {\r\n        if (this.el && this.el.nativeElement) {\r\n            this.element = this.el.nativeElement;\r\n            this.renderNode = getTargetNode(this.el.nativeElement);\r\n            this.text = this.getText();\r\n            this.attributes = this.getAttributes();\r\n            this.addTextListener();\r\n\r\n            if (this.language) {\r\n                this.replaceText();\r\n                this.replaceAttributes();\r\n            } else {\r\n                this.addTranslationListener();\r\n            }\r\n        }\r\n    }\r\n\r\n    public ngOnChanges(): void {\r\n        if (this.text) {\r\n            if (this.nodeValue == null || this.nodeValue === '') {\r\n                if (this.value) {\r\n                    this.text = this.value;\r\n                } else if (this.content) {\r\n                    this.text = this.content;\r\n                }\r\n            }\r\n            this.replaceText();\r\n        }\r\n        if (this.attributes && this.attributes.length > 0) {\r\n            this.replaceAttributes();\r\n        }\r\n    }\r\n\r\n    public ngOnDestroy(): void {\r\n        this.destroy.next(true);\r\n        this.removeTextListener();\r\n    }\r\n\r\n    protected abstract getValue(text: string): string;\r\n\r\n    private getText(): string {\r\n        let text = '';\r\n        if (this.element && this.element.childNodes.length > 0) {\r\n            text = this.getNodeValue();\r\n        } else if (this.value) {\r\n            text = this.value;\r\n        } else if (this.content) {\r\n            text = this.content;\r\n        }\r\n        return text;\r\n    }\r\n\r\n    private getNodeValue(): string {\r\n        this.nodeValue = this.renderNode != null && this.renderNode.nodeValue != null ? this.renderNode.nodeValue : '';\r\n        return this.nodeValue ? this.nodeValue.trim() : '';\r\n    }\r\n\r\n    private getAttributes(): any[] {\r\n        const attributes: any[] = [];\r\n        if (this.element && this.element.attributes) {\r\n            for (const attr of Array.from(this.element.attributes)) {\r\n                if (attr && attr.name) {\r\n                    const [, name = ''] = attr.name.match(/^l10n-(.+)$/) || [];\r\n                    if (name) {\r\n                        const targetAttr = Array.from(this.element.attributes).find(a => a.name === name);\r\n                        if (targetAttr) attributes.push({ name: targetAttr.name, value: targetAttr.value });\r\n                    }\r\n                }\r\n            }\r\n        }\r\n        return attributes;\r\n    }\r\n\r\n    private addTextListener(): void {\r\n        if (typeof MutationObserver !== 'undefined') {\r\n            this.textObserver = new MutationObserver(() => {\r\n                if (this.element) {\r\n                    this.renderNode = getTargetNode(this.element);\r\n                    this.text = this.getText();\r\n                    this.replaceText();\r\n                }\r\n            });\r\n            if (this.renderNode) {\r\n                this.textObserver.observe(this.renderNode, { subtree: true, characterData: true });\r\n            }\r\n        }\r\n    }\r\n\r\n    private removeTextListener(): void {\r\n        if (this.textObserver) {\r\n            this.textObserver.disconnect();\r\n        }\r\n    }\r\n\r\n    private addTranslationListener(): void {\r\n        this.translation.onChange().pipe(takeUntil(this.destroy)).subscribe({\r\n            next: () => {\r\n                this.replaceText();\r\n                this.replaceAttributes();\r\n            }\r\n        });\r\n    }\r\n\r\n    private replaceText(): void {\r\n        if (this.text) {\r\n            this.setText(this.getValue(this.text));\r\n        }\r\n    }\r\n\r\n    private replaceAttributes(): void {\r\n        if (this.attributes.length > 0) {\r\n            this.setAttributes(this.getAttributesValues());\r\n        }\r\n    }\r\n\r\n    private setText(value: string): void {\r\n        if (value) {\r\n            if (this.nodeValue && this.text) {\r\n                this.removeTextListener();\r\n                this.renderer.setValue(this.renderNode, this.nodeValue.replace(this.text, value));\r\n                this.addTextListener();\r\n            } else if (this.value) {\r\n                this.renderer.setAttribute(this.element, 'value', value);\r\n            } else if (this.content) {\r\n                this.renderer.setProperty(this.element, 'innerHTML', value);\r\n            }\r\n        }\r\n    }\r\n\r\n    private setAttributes(data: any): void {\r\n        for (const attr of this.attributes) {\r\n            this.renderer.setAttribute(this.element, attr.name, data[attr.value]);\r\n        }\r\n    }\r\n\r\n    private getAttributesValues(): any {\r\n        const values = this.attributes.map(attr => attr.value);\r\n        const data: any = {};\r\n        for (const value of values) {\r\n            data[value] = this.getValue(value);\r\n        }\r\n        return data;\r\n    }\r\n\r\n}\r\n"]}