forked from MapComplete/MapComplete
151 lines
No EOL
4.4 KiB
TypeScript
151 lines
No EOL
4.4 KiB
TypeScript
import {DesugaringStep} from "./Conversion";
|
|
import {Utils} from "../../../Utils";
|
|
import Translations from "../../../UI/i18n/Translations";
|
|
|
|
export class AddContextToTranslations<T> extends DesugaringStep<T> {
|
|
private readonly _prefix: string;
|
|
|
|
constructor(prefix = "") {
|
|
super("Adds a '_context' to every object that is probably a translation", ["_context"], "AddContextToTranslation");
|
|
this._prefix = prefix;
|
|
}
|
|
|
|
/**
|
|
* const theme = {
|
|
* layers: [
|
|
* {
|
|
* builtin: ["abc"],
|
|
* override: {
|
|
* title:{
|
|
* en: "Some title"
|
|
* }
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* const rewritten = new AddContextToTranslations<any>("prefix:").convert(theme, "context").result
|
|
* const expected = {
|
|
* layers: [
|
|
* {
|
|
* builtin: ["abc"],
|
|
* override: {
|
|
* title:{
|
|
* _context: "prefix:context.layers.0.override.title"
|
|
* en: "Some title"
|
|
* }
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* rewritten // => expected
|
|
*
|
|
* // should use the ID if one is present instead of the index
|
|
* const theme = {
|
|
* layers: [
|
|
* {
|
|
* tagRenderings:[
|
|
*
|
|
* {id: "some-tr",
|
|
* question:{
|
|
* en:"Question?"
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* ]
|
|
* }
|
|
* const rewritten = new AddContextToTranslations<any>("prefix:").convert(theme, "context").result
|
|
* const expected = {
|
|
* layers: [
|
|
* {
|
|
* tagRenderings:[
|
|
*
|
|
* {id: "some-tr",
|
|
* question:{
|
|
* _context: "prefix:context.layers.0.tagRenderings.some-tr.question"
|
|
* en:"Question?"
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* ]
|
|
* }
|
|
* rewritten // => expected
|
|
*
|
|
* // should preserve nulls
|
|
* const theme = {
|
|
* layers: [
|
|
* {
|
|
* builtin: ["abc"],
|
|
* override: {
|
|
* name:null
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* const rewritten = new AddContextToTranslations<any>("prefix:").convert(theme, "context").result
|
|
* const expected = {
|
|
* layers: [
|
|
* {
|
|
* builtin: ["abc"],
|
|
* override: {
|
|
* name: null
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* rewritten // => expected
|
|
*
|
|
*
|
|
* // Should ignore all if '#dont-translate' is set
|
|
* const theme = {
|
|
* "#dont-translate": "*",
|
|
* layers: [
|
|
* {
|
|
* builtin: ["abc"],
|
|
* override: {
|
|
* title:{
|
|
* en: "Some title"
|
|
* }
|
|
* }
|
|
* }
|
|
* ]
|
|
* }
|
|
* const rewritten = new AddContextToTranslations<any>("prefix:").convert(theme, "context").result
|
|
* rewritten // => theme
|
|
*
|
|
*/
|
|
convert(json: T, context: string): { result: T; errors?: string[]; warnings?: string[]; information?: string[] } {
|
|
|
|
if(json["#dont-translate"] === "*"){
|
|
return {result: json}
|
|
}
|
|
|
|
const result = Utils.WalkJson(json, (leaf, path) => {
|
|
if(leaf === undefined || leaf === null){
|
|
return leaf
|
|
}
|
|
if (typeof leaf === "object") {
|
|
|
|
// follow the path. If we encounter a number, check that there is no ID we can use instead
|
|
let breadcrumb = json;
|
|
for (let i = 0; i < path.length; i++) {
|
|
const pointer = path[i]
|
|
breadcrumb = breadcrumb[pointer]
|
|
if(pointer.match("[0-9]+") && breadcrumb["id"] !== undefined){
|
|
path[i] = breadcrumb["id"]
|
|
}
|
|
}
|
|
|
|
return {...leaf, _context: this._prefix + context + "." + path.join(".")}
|
|
} else {
|
|
return leaf
|
|
}
|
|
}, obj => obj === undefined || obj === null || Translations.isProbablyATranslation(obj))
|
|
|
|
return {
|
|
result
|
|
};
|
|
}
|
|
|
|
} |