@umituz/react-native-localization 3.7.10 → 3.7.11

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@umituz/react-native-localization",
3
- "version": "3.7.10",
3
+ "version": "3.7.11",
4
4
  "type": "module",
5
5
  "description": "Generic localization system for React Native apps with i18n support",
6
6
  "main": "./src/index.ts",
@@ -11,8 +11,17 @@ export function extractUsedKeys(srcDir) {
11
11
  if (!srcDir) return keys;
12
12
 
13
13
  const absoluteSrcDir = path.resolve(process.cwd(), srcDir);
14
- if (!fs.existsSync(absoluteSrcDir)) {
15
- return keys;
14
+ if (!fs.existsSync(absoluteSrcDir)) return keys;
15
+
16
+ // Step 1: Find all ScenarioId possible values once
17
+ const scenarioValues = new Set();
18
+ const scenarioFile = path.resolve(absoluteSrcDir, 'domains/scenarios/domain/Scenario.ts');
19
+ if (fs.existsSync(scenarioFile)) {
20
+ const content = fs.readFileSync(scenarioFile, 'utf8');
21
+ const matches = content.matchAll(/([A-Z0-9_]+)\s*=\s*['"`]([a-z0-9_]+)['"`]/g);
22
+ for (const match of matches) {
23
+ scenarioValues.add(match[2]);
24
+ }
16
25
  }
17
26
 
18
27
  function walk(dir) {
@@ -23,16 +32,37 @@ export function extractUsedKeys(srcDir) {
23
32
 
24
33
  if (stat.isDirectory()) {
25
34
  const skipDirs = ['node_modules', '.expo', '.git', 'build', 'ios', 'android', 'assets'];
26
- if (!skipDirs.includes(file)) {
27
- walk(fullPath);
28
- }
35
+ if (!skipDirs.includes(file)) walk(fullPath);
29
36
  } else if (/\.(ts|tsx|js|jsx)$/.test(file)) {
30
37
  const content = fs.readFileSync(fullPath, 'utf8');
31
- // Regex for t('key') or t("key") or i18n.t('key')
32
- const regex = /(?:^|\W)t\(['"]([^'"]+)['"]\)/g;
38
+
39
+ // Pattern 1: Standard t('key') calls
40
+ const tRegex = /(?:^|\W)t\(['"`]([^'"`]+)['"`]\)/g;
33
41
  let match;
34
- while ((match = regex.exec(content)) !== null) {
35
- keys.add(match[1]);
42
+ while ((match = tRegex.exec(content)) !== null) {
43
+ if (!match[1].includes('${')) keys.add(match[1]);
44
+ }
45
+
46
+ // Pattern 2: Scenario ID usage
47
+ // If we see ScenarioId.XXXX or just the string value matching a scenario
48
+ scenarioValues.forEach(val => {
49
+ if (content.includes(val)) {
50
+ keys.add(`scenario.${val}.title`);
51
+ keys.add(`scenario.${val}.description`);
52
+ keys.add(`scenario.${val}.details`);
53
+ keys.add(`scenario.${val}.tip`);
54
+ }
55
+ });
56
+
57
+ // Pattern 3: Dot-notation keys in configs/literals (e.g. "common.button.save")
58
+ // We look for at least two segments (one dot)
59
+ const dotRegex = /['"`]([a-z][a-z0-9_]*\.[a-z0-9_.]*[a-z0-9_])['"`]/gi;
60
+ while ((match = dotRegex.exec(content)) !== null) {
61
+ const key = match[1];
62
+ const isFile = /\.(ts|tsx|js|jsx|png|jpg|jpeg|svg|json)$/i.test(key);
63
+ if (!isFile && !key.includes(' ') && !key.includes('${')) {
64
+ keys.add(key);
65
+ }
36
66
  }
37
67
  }
38
68
  }
@@ -1,56 +1,51 @@
1
- import https from 'https';
2
- import { shouldSkipWord } from './translation-config.js';
3
-
4
1
  /**
5
- * Translator
6
- * Google Translate API integration and translation logic
2
+ * Translation Utilities
3
+ * Handles call to translation APIs
7
4
  */
8
5
 
9
- export function delay(ms) {
10
- return new Promise(resolve => setTimeout(resolve, ms));
11
- }
6
+ import { getTargetLanguage, isSingleWord as checkSingleWord, shouldSkipWord } from './translation-config.js';
12
7
 
13
- export async function translateText(text, targetLang) {
14
- return new Promise((resolve) => {
15
- if (shouldSkipWord(text)) {
16
- resolve(text);
17
- return;
18
- }
8
+ let lastCallTime = 0;
9
+ const MIN_DELAY = 100; // ms
19
10
 
11
+ async function translateText(text, targetLang) {
12
+ if (!text || typeof text !== 'string') return text;
13
+ if (shouldSkipWord(text)) return text;
14
+
15
+ // Rate limiting
16
+ const now = Date.now();
17
+ const waitTime = Math.max(0, MIN_DELAY - (now - lastCallTime));
18
+ if (waitTime > 0) await new Promise(resolve => setTimeout(resolve, waitTime));
19
+ lastCallTime = Date.now();
20
+
21
+ try {
20
22
  const encodedText = encodeURIComponent(text);
21
23
  const url = `https://translate.googleapis.com/translate_a/single?client=gtx&sl=en&tl=${targetLang}&dt=t&q=${encodedText}`;
22
-
23
- https
24
- .get(url, res => {
25
- let data = '';
26
- res.on('data', chunk => {
27
- data += chunk;
28
- });
29
- res.on('end', () => {
30
- try {
31
- const parsed = JSON.parse(data);
32
- const translated = parsed[0]
33
- .map(item => item[0])
34
- .join('')
35
- .trim();
36
- resolve(translated || text);
37
- } catch (error) {
38
- resolve(text);
39
- }
40
- });
41
- })
42
- .on('error', () => {
43
- resolve(text);
44
- });
45
- });
24
+
25
+ const response = await fetch(url);
26
+ if (!response.ok) return text;
27
+
28
+ const data = await response.json();
29
+ return data && data[0] && data[0][0] && data[0][0][0] ? data[0][0][0] : text;
30
+ } catch (error) {
31
+ if (__DEV__) console.error(` ❌ Translation error for "${text}":`, error.message);
32
+ return text;
33
+ }
46
34
  }
47
35
 
48
36
  function needsTranslation(value, enValue) {
49
- if (typeof enValue !== 'string') return false;
37
+ if (typeof enValue !== 'string' || !enValue.trim()) return false;
50
38
  if (shouldSkipWord(enValue)) return false;
39
+
40
+ // If value is missing or same as technical key
51
41
  if (!value || typeof value !== 'string') return true;
52
42
 
43
+ // Heuristic: If English value looks like a technical key (e.g. "scenario.xxx.title")
44
+ // and the target value is exactly the same, it definitely needs translation.
45
+ const isTechnicalKey = enValue.includes('.') && !enValue.includes(' ');
46
+
53
47
  if (value === enValue) {
48
+ if (isTechnicalKey) return true;
54
49
  const isSingleWord = !enValue.includes(' ') && enValue.length < 20;
55
50
  return !isSingleWord;
56
51
  }
@@ -59,39 +54,28 @@ function needsTranslation(value, enValue) {
59
54
  }
60
55
 
61
56
  export async function translateObject(enObj, targetObj, targetLang, path = '', stats = { count: 0, newKeys: [] }) {
62
- for (const key in enObj) {
63
- const currentPath = path ? `${path}.${key}` : key;
57
+ const keys = Object.keys(enObj);
58
+
59
+ for (const key of keys) {
64
60
  const enValue = enObj[key];
65
61
  const targetValue = targetObj[key];
62
+ const currentPath = path ? `${path}.${key}` : key;
66
63
 
67
- if (Array.isArray(enValue)) {
68
- if (!Array.isArray(targetValue)) targetObj[key] = [];
69
- for (let i = 0; i < enValue.length; i++) {
70
- if (typeof enValue[i] === 'string' && needsTranslation(targetObj[key][i], enValue[i])) {
71
- const translated = await translateText(enValue[i], targetLang);
72
- if (translated !== enValue[i]) {
73
- const isNewKey = targetObj[key][i] === enValue[i];
74
- console.log(` ${isNewKey ? '🆕 NEW' : '🔄'} ${currentPath}[${i}]: "${enValue[i].substring(0, 40)}"`);
75
- targetObj[key][i] = translated;
76
- stats.count++;
77
- if (isNewKey) stats.newKeys.push(`${currentPath}[${i}]`);
78
- }
79
- await delay(200);
80
- }
81
- }
82
- } else if (typeof enValue === 'object' && enValue !== null) {
64
+ if (typeof enValue === 'object' && enValue !== null) {
83
65
  if (!targetObj[key] || typeof targetObj[key] !== 'object') targetObj[key] = {};
84
66
  await translateObject(enValue, targetObj[key], targetLang, currentPath, stats);
85
67
  } else if (typeof enValue === 'string' && needsTranslation(targetValue, enValue)) {
86
68
  const translated = await translateText(enValue, targetLang);
69
+
87
70
  const isNewKey = targetValue === undefined;
88
- if (translated !== enValue || isNewKey) {
89
- console.log(` ${isNewKey ? '🆕 NEW' : '🔄'} ${currentPath}: "${enValue.substring(0, 40)}"`);
71
+ // Force increment if it looks like a technical key that we just "translated"
72
+ // even if the API returned the same string (placeholder)
73
+ const isTechnicalKey = enValue.includes('.') && !enValue.includes(' ');
74
+
75
+ if (translated !== enValue || isNewKey || (isTechnicalKey && translated === enValue)) {
90
76
  targetObj[key] = translated;
91
77
  stats.count++;
92
- if (isNewKey) stats.newKeys.push(currentPath);
93
78
  }
94
- await delay(200);
95
79
  }
96
80
  }
97
81
  }