@mp-consulting/homebridge-lg-thinq 1.0.0
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/.claude/settings.local.json +15 -0
- package/CHANGELOG.md +98 -0
- package/LICENSE +176 -0
- package/README.md +114 -0
- package/config.schema.json +399 -0
- package/dist/__tests__/baseDevice.spec.d.ts +1 -0
- package/dist/__tests__/baseDevice.spec.js +96 -0
- package/dist/__tests__/baseDevice.spec.js.map +1 -0
- package/dist/__tests__/deviceControl.coercion.spec.d.ts +1 -0
- package/dist/__tests__/deviceControl.coercion.spec.js +53 -0
- package/dist/__tests__/deviceControl.coercion.spec.js.map +1 -0
- package/dist/__tests__/helper.spec.d.ts +1 -0
- package/dist/__tests__/helper.spec.js +74 -0
- package/dist/__tests__/helper.spec.js.map +1 -0
- package/dist/baseDevice.d.ts +40 -0
- package/dist/baseDevice.js +85 -0
- package/dist/baseDevice.js.map +1 -0
- package/dist/baseDevice.spec.d.ts +1 -0
- package/dist/baseDevice.spec.js +107 -0
- package/dist/baseDevice.spec.js.map +1 -0
- package/dist/characteristics/TotalConsumption.d.ts +2 -0
- package/dist/characteristics/TotalConsumption.js +17 -0
- package/dist/characteristics/TotalConsumption.js.map +1 -0
- package/dist/characteristics/index.d.ts +2 -0
- package/dist/characteristics/index.js +7 -0
- package/dist/characteristics/index.js.map +1 -0
- package/dist/cli.d.ts +2 -0
- package/dist/cli.js +89 -0
- package/dist/cli.js.map +1 -0
- package/dist/devices/AeroTower.d.ts +24 -0
- package/dist/devices/AeroTower.js +113 -0
- package/dist/devices/AeroTower.js.map +1 -0
- package/dist/devices/AirConditioner.d.ts +425 -0
- package/dist/devices/AirConditioner.js +1253 -0
- package/dist/devices/AirConditioner.js.map +1 -0
- package/dist/devices/AirPurifier.d.ts +50 -0
- package/dist/devices/AirPurifier.js +281 -0
- package/dist/devices/AirPurifier.js.map +1 -0
- package/dist/devices/Dehumidifier.d.ts +28 -0
- package/dist/devices/Dehumidifier.js +175 -0
- package/dist/devices/Dehumidifier.js.map +1 -0
- package/dist/devices/Dishwasher.d.ts +64 -0
- package/dist/devices/Dishwasher.js +740 -0
- package/dist/devices/Dishwasher.js.map +1 -0
- package/dist/devices/Microwave.d.ts +128 -0
- package/dist/devices/Microwave.js +1939 -0
- package/dist/devices/Microwave.js.map +1 -0
- package/dist/devices/Oven.d.ts +148 -0
- package/dist/devices/Oven.js +1850 -0
- package/dist/devices/Oven.js.map +1 -0
- package/dist/devices/RangeHood.d.ts +16 -0
- package/dist/devices/RangeHood.js +99 -0
- package/dist/devices/RangeHood.js.map +1 -0
- package/dist/devices/Refrigerator.d.ts +50 -0
- package/dist/devices/Refrigerator.js +325 -0
- package/dist/devices/Refrigerator.js.map +1 -0
- package/dist/devices/Styler.d.ts +27 -0
- package/dist/devices/Styler.js +76 -0
- package/dist/devices/Styler.js.map +1 -0
- package/dist/devices/WasherDryer.d.ts +39 -0
- package/dist/devices/WasherDryer.js +170 -0
- package/dist/devices/WasherDryer.js.map +1 -0
- package/dist/devices/WasherDryer2.d.ts +9 -0
- package/dist/devices/WasherDryer2.js +16 -0
- package/dist/devices/WasherDryer2.js.map +1 -0
- package/dist/errors/AuthenticationError.d.ts +2 -0
- package/dist/errors/AuthenticationError.js +3 -0
- package/dist/errors/AuthenticationError.js.map +1 -0
- package/dist/errors/ManualProcessNeeded.d.ts +3 -0
- package/dist/errors/ManualProcessNeeded.js +4 -0
- package/dist/errors/ManualProcessNeeded.js.map +1 -0
- package/dist/errors/MonitorError.d.ts +2 -0
- package/dist/errors/MonitorError.js +3 -0
- package/dist/errors/MonitorError.js.map +1 -0
- package/dist/errors/NotConnectedError.d.ts +3 -0
- package/dist/errors/NotConnectedError.js +4 -0
- package/dist/errors/NotConnectedError.js.map +1 -0
- package/dist/errors/TokenError.d.ts +2 -0
- package/dist/errors/TokenError.js +3 -0
- package/dist/errors/TokenError.js.map +1 -0
- package/dist/errors/TokenExpiredError.d.ts +3 -0
- package/dist/errors/TokenExpiredError.js +4 -0
- package/dist/errors/TokenExpiredError.js.map +1 -0
- package/dist/errors/index.d.ts +6 -0
- package/dist/errors/index.js +7 -0
- package/dist/errors/index.js.map +1 -0
- package/dist/helper.d.ts +24 -0
- package/dist/helper.js +66 -0
- package/dist/helper.js.map +1 -0
- package/dist/helper.spec.d.ts +1 -0
- package/dist/helper.spec.js +74 -0
- package/dist/helper.spec.js.map +1 -0
- package/dist/index.d.ts +6 -0
- package/dist/index.js +9 -0
- package/dist/index.js.map +1 -0
- package/dist/lib/API.d.ts +141 -0
- package/dist/lib/API.js +362 -0
- package/dist/lib/API.js.map +1 -0
- package/dist/lib/API.spec.d.ts +1 -0
- package/dist/lib/API.spec.js +55 -0
- package/dist/lib/API.spec.js.map +1 -0
- package/dist/lib/Auth.d.ts +99 -0
- package/dist/lib/Auth.js +348 -0
- package/dist/lib/Auth.js.map +1 -0
- package/dist/lib/Auth.spec.d.ts +1 -0
- package/dist/lib/Auth.spec.js +111 -0
- package/dist/lib/Auth.spec.js.map +1 -0
- package/dist/lib/Device.d.ts +88 -0
- package/dist/lib/Device.js +95 -0
- package/dist/lib/Device.js.map +1 -0
- package/dist/lib/Device.spec.d.ts +1 -0
- package/dist/lib/Device.spec.js +53 -0
- package/dist/lib/Device.spec.js.map +1 -0
- package/dist/lib/DeviceModel.d.ts +164 -0
- package/dist/lib/DeviceModel.js +321 -0
- package/dist/lib/DeviceModel.js.map +1 -0
- package/dist/lib/DeviceModel.spec.d.ts +1 -0
- package/dist/lib/DeviceModel.spec.js +90 -0
- package/dist/lib/DeviceModel.spec.js.map +1 -0
- package/dist/lib/Gateway.d.ts +18 -0
- package/dist/lib/Gateway.js +25 -0
- package/dist/lib/Gateway.js.map +1 -0
- package/dist/lib/Gateway.spec.d.ts +1 -0
- package/dist/lib/Gateway.spec.js +35 -0
- package/dist/lib/Gateway.spec.js.map +1 -0
- package/dist/lib/Persist.d.ts +101 -0
- package/dist/lib/Persist.js +245 -0
- package/dist/lib/Persist.js.map +1 -0
- package/dist/lib/Persist.spec.d.ts +1 -0
- package/dist/lib/Persist.spec.js +90 -0
- package/dist/lib/Persist.spec.js.map +1 -0
- package/dist/lib/Session.d.ts +80 -0
- package/dist/lib/Session.js +100 -0
- package/dist/lib/Session.js.map +1 -0
- package/dist/lib/Session.spec.d.ts +1 -0
- package/dist/lib/Session.spec.js +43 -0
- package/dist/lib/Session.spec.js.map +1 -0
- package/dist/lib/ThinQ.d.ts +28 -0
- package/dist/lib/ThinQ.js +373 -0
- package/dist/lib/ThinQ.js.map +1 -0
- package/dist/lib/__tests__/API.spec.d.ts +1 -0
- package/dist/lib/__tests__/API.spec.js +55 -0
- package/dist/lib/__tests__/API.spec.js.map +1 -0
- package/dist/lib/__tests__/Auth.spec.d.ts +1 -0
- package/dist/lib/__tests__/Auth.spec.js +110 -0
- package/dist/lib/__tests__/Auth.spec.js.map +1 -0
- package/dist/lib/__tests__/Device.spec.d.ts +1 -0
- package/dist/lib/__tests__/Device.spec.js +53 -0
- package/dist/lib/__tests__/Device.spec.js.map +1 -0
- package/dist/lib/__tests__/DeviceModel.spec.d.ts +1 -0
- package/dist/lib/__tests__/DeviceModel.spec.js +90 -0
- package/dist/lib/__tests__/DeviceModel.spec.js.map +1 -0
- package/dist/lib/__tests__/Gateway.spec.d.ts +1 -0
- package/dist/lib/__tests__/Gateway.spec.js +35 -0
- package/dist/lib/__tests__/Gateway.spec.js.map +1 -0
- package/dist/lib/__tests__/Persist.spec.d.ts +1 -0
- package/dist/lib/__tests__/Persist.spec.js +90 -0
- package/dist/lib/__tests__/Persist.spec.js.map +1 -0
- package/dist/lib/__tests__/Session.spec.d.ts +1 -0
- package/dist/lib/__tests__/Session.spec.js +43 -0
- package/dist/lib/__tests__/Session.spec.js.map +1 -0
- package/dist/lib/constants.d.ts +95 -0
- package/dist/lib/constants.js +106 -0
- package/dist/lib/constants.js.map +1 -0
- package/dist/lib/request.d.ts +2 -0
- package/dist/lib/request.js +66 -0
- package/dist/lib/request.js.map +1 -0
- package/dist/platform.d.ts +41 -0
- package/dist/platform.js +229 -0
- package/dist/platform.js.map +1 -0
- package/dist/settings.d.ts +8 -0
- package/dist/settings.js +9 -0
- package/dist/settings.js.map +1 -0
- package/dist/status/BaseStatus.d.ts +48 -0
- package/dist/status/BaseStatus.js +59 -0
- package/dist/status/BaseStatus.js.map +1 -0
- package/dist/types/snapshots.d.ts +142 -0
- package/dist/types/snapshots.js +6 -0
- package/dist/types/snapshots.js.map +1 -0
- package/dist/utils/__tests__/normalize.spec.d.ts +1 -0
- package/dist/utils/__tests__/normalize.spec.js +95 -0
- package/dist/utils/__tests__/normalize.spec.js.map +1 -0
- package/dist/utils/normalize.d.ts +22 -0
- package/dist/utils/normalize.js +51 -0
- package/dist/utils/normalize.js.map +1 -0
- package/dist/v1/__tests__/prepareControlData.spec.d.ts +1 -0
- package/dist/v1/__tests__/prepareControlData.spec.js +48 -0
- package/dist/v1/__tests__/prepareControlData.spec.js.map +1 -0
- package/dist/v1/devices/AC.d.ts +13 -0
- package/dist/v1/devices/AC.js +112 -0
- package/dist/v1/devices/AC.js.map +1 -0
- package/dist/v1/devices/AirPurifier.d.ts +15 -0
- package/dist/v1/devices/AirPurifier.js +57 -0
- package/dist/v1/devices/AirPurifier.js.map +1 -0
- package/dist/v1/devices/RangeHood.d.ts +6 -0
- package/dist/v1/devices/RangeHood.js +12 -0
- package/dist/v1/devices/RangeHood.js.map +1 -0
- package/dist/v1/devices/Refrigerator.d.ts +17 -0
- package/dist/v1/devices/Refrigerator.js +69 -0
- package/dist/v1/devices/Refrigerator.js.map +1 -0
- package/dist/v1/devices/Washer.d.ts +10 -0
- package/dist/v1/devices/Washer.js +23 -0
- package/dist/v1/devices/Washer.js.map +1 -0
- package/dist/v1/devices/index.d.ts +6 -0
- package/dist/v1/devices/index.js +7 -0
- package/dist/v1/devices/index.js.map +1 -0
- package/dist/v1/helper.d.ts +14 -0
- package/dist/v1/helper.js +111 -0
- package/dist/v1/helper.js.map +1 -0
- package/dist/v1/transforms/AirPurifierState.d.ts +2 -0
- package/dist/v1/transforms/AirPurifierState.js +9 -0
- package/dist/v1/transforms/AirPurifierState.js.map +1 -0
- package/dist/v1/transforms/AirState.d.ts +9 -0
- package/dist/v1/transforms/AirState.js +55 -0
- package/dist/v1/transforms/AirState.js.map +1 -0
- package/dist/v1/transforms/HoodState.d.ts +17 -0
- package/dist/v1/transforms/HoodState.js +20 -0
- package/dist/v1/transforms/HoodState.js.map +1 -0
- package/dist/v1/transforms/RefState.d.ts +6 -0
- package/dist/v1/transforms/RefState.js +29 -0
- package/dist/v1/transforms/RefState.js.map +1 -0
- package/dist/v1/transforms/WasherDryer.d.ts +49 -0
- package/dist/v1/transforms/WasherDryer.js +56 -0
- package/dist/v1/transforms/WasherDryer.js.map +1 -0
- package/docs/authorization.md +40 -0
- package/docs/device-configuration.md +68 -0
- package/homebridge-ui/public/index.html +120 -0
- package/homebridge-ui/public/js/app.js +300 -0
- package/homebridge-ui/public/js/countries.js +233 -0
- package/homebridge-ui/public/styles.css +185 -0
- package/homebridge-ui/server.js +103 -0
- package/jest.config.ts +39 -0
- package/package.json +83 -0
- package/sample/README.md +10 -0
- package/sample/airconditioner-model.json +3080 -0
- package/sample/airconditioner-snapshot.json +49 -0
- package/sample/airconditioner.json +157 -0
- package/sample/dishwasher-model.json +869 -0
- package/sample/dishwasher.json +125 -0
- package/sample/washer_dryer-model.json +1294 -0
- package/sample/washer_dryer.json +126 -0
|
@@ -0,0 +1,300 @@
|
|
|
1
|
+
(async () => {
|
|
2
|
+
'use strict';
|
|
3
|
+
|
|
4
|
+
// State
|
|
5
|
+
let credentials = { refresh_token: null, country: null, language: null };
|
|
6
|
+
let devices = [];
|
|
7
|
+
|
|
8
|
+
// DOM helper
|
|
9
|
+
const $ = id => document.getElementById(id);
|
|
10
|
+
const steps = { login: $('step-login'), devices: $('step-devices'), complete: $('step-complete') };
|
|
11
|
+
|
|
12
|
+
// Constants
|
|
13
|
+
const DEFAULT_COUNTRY = 'US';
|
|
14
|
+
const DEFAULT_LANGUAGE = 'en-US';
|
|
15
|
+
|
|
16
|
+
// Load existing config
|
|
17
|
+
const pluginConfig = await homebridge.getPluginConfig();
|
|
18
|
+
const config = pluginConfig[0] || {};
|
|
19
|
+
|
|
20
|
+
// Populate country select
|
|
21
|
+
const populateCountrySelect = () => {
|
|
22
|
+
const select = $('country_language');
|
|
23
|
+
select.innerHTML = '';
|
|
24
|
+
|
|
25
|
+
COUNTRIES.forEach(({ country, language, label }) => {
|
|
26
|
+
const option = document.createElement('option');
|
|
27
|
+
option.dataset.country = country;
|
|
28
|
+
option.dataset.language = language;
|
|
29
|
+
option.textContent = label;
|
|
30
|
+
select.appendChild(option);
|
|
31
|
+
});
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
// Find country option
|
|
35
|
+
const findCountryOption = (country, language) => {
|
|
36
|
+
const select = $('country_language');
|
|
37
|
+
let option = select.querySelector(`option[data-country="${country}"][data-language="${language}"]`);
|
|
38
|
+
if (!option) {
|
|
39
|
+
option = select.querySelector(`option[data-country="${DEFAULT_COUNTRY}"][data-language="${DEFAULT_LANGUAGE}"]`);
|
|
40
|
+
}
|
|
41
|
+
return option;
|
|
42
|
+
};
|
|
43
|
+
|
|
44
|
+
// Pre-fill form from config
|
|
45
|
+
const prefillForm = () => {
|
|
46
|
+
if (config.username) $('username').value = config.username;
|
|
47
|
+
if (config.password) $('password').value = config.password;
|
|
48
|
+
if (config.thinq1) $('thinq1').checked = config.thinq1;
|
|
49
|
+
if (config.username && config.password) $('rememberCredentials').checked = true;
|
|
50
|
+
|
|
51
|
+
// Set country/language
|
|
52
|
+
const country = config.country || DEFAULT_COUNTRY;
|
|
53
|
+
const language = config.language || DEFAULT_LANGUAGE;
|
|
54
|
+
const option = findCountryOption(country, language);
|
|
55
|
+
if (option) option.selected = true;
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
// Show/hide step
|
|
59
|
+
const showStep = step => {
|
|
60
|
+
Object.entries(steps).forEach(([k, el]) => {
|
|
61
|
+
el.classList.toggle('hidden', k !== step);
|
|
62
|
+
});
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
// Escape HTML for XSS prevention
|
|
66
|
+
const escapeHtml = text => {
|
|
67
|
+
const div = document.createElement('div');
|
|
68
|
+
div.textContent = text;
|
|
69
|
+
return div.innerHTML;
|
|
70
|
+
};
|
|
71
|
+
|
|
72
|
+
// Render device list
|
|
73
|
+
const renderDevices = () => {
|
|
74
|
+
const list = $('device-list');
|
|
75
|
+
if (!devices.length) {
|
|
76
|
+
list.innerHTML = '<div class="text-center text-muted py-4">No devices found</div>';
|
|
77
|
+
return;
|
|
78
|
+
}
|
|
79
|
+
list.innerHTML = devices.map(d => `
|
|
80
|
+
<div class="device-item">
|
|
81
|
+
<div class="device-info">
|
|
82
|
+
<div class="device-name">${escapeHtml(d.name)}</div>
|
|
83
|
+
<div class="device-id">ID: ${escapeHtml(d.id)}</div>
|
|
84
|
+
<div class="device-type">${escapeHtml(d.type || 'Unknown')}</div>
|
|
85
|
+
</div>
|
|
86
|
+
<div class="device-status">
|
|
87
|
+
<span class="status-badge ${d.online !== false ? 'status-online' : 'status-offline'}">${d.online !== false ? 'Online' : 'Offline'}</span>
|
|
88
|
+
</div>
|
|
89
|
+
</div>
|
|
90
|
+
`).join('');
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
// Load devices from API
|
|
94
|
+
const loadDevices = async () => {
|
|
95
|
+
$('refresh-spinner').classList.remove('hidden');
|
|
96
|
+
$('btn-refresh').disabled = true;
|
|
97
|
+
|
|
98
|
+
try {
|
|
99
|
+
const res = await homebridge.request('/get-all-devices', {
|
|
100
|
+
country: credentials.country,
|
|
101
|
+
language: credentials.language,
|
|
102
|
+
refresh_token: credentials.refresh_token,
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
if (res.success) {
|
|
106
|
+
devices = res.devices || [];
|
|
107
|
+
$('device-count').textContent = devices.length;
|
|
108
|
+
renderDevices();
|
|
109
|
+
} else {
|
|
110
|
+
homebridge.toast.error(res.error || 'Failed to load devices');
|
|
111
|
+
}
|
|
112
|
+
} catch (e) {
|
|
113
|
+
homebridge.toast.error(e.message || 'Failed to load devices');
|
|
114
|
+
} finally {
|
|
115
|
+
$('refresh-spinner').classList.add('hidden');
|
|
116
|
+
$('btn-refresh').disabled = false;
|
|
117
|
+
}
|
|
118
|
+
};
|
|
119
|
+
|
|
120
|
+
// Check for existing session
|
|
121
|
+
const checkSession = async () => {
|
|
122
|
+
try {
|
|
123
|
+
if (config.refresh_token) {
|
|
124
|
+
credentials = {
|
|
125
|
+
refresh_token: config.refresh_token,
|
|
126
|
+
country: config.country || DEFAULT_COUNTRY,
|
|
127
|
+
language: config.language || DEFAULT_LANGUAGE,
|
|
128
|
+
};
|
|
129
|
+
|
|
130
|
+
$('active-session-notice').classList.remove('hidden');
|
|
131
|
+
$('login-form').classList.add('hidden');
|
|
132
|
+
|
|
133
|
+
await loadDevices();
|
|
134
|
+
showStep('devices');
|
|
135
|
+
return true;
|
|
136
|
+
}
|
|
137
|
+
} catch {
|
|
138
|
+
// No session, show login
|
|
139
|
+
}
|
|
140
|
+
return false;
|
|
141
|
+
};
|
|
142
|
+
|
|
143
|
+
// Get selected country/language
|
|
144
|
+
const getSelectedCountryLanguage = () => {
|
|
145
|
+
const select = $('country_language');
|
|
146
|
+
const selected = select.options[select.selectedIndex];
|
|
147
|
+
return {
|
|
148
|
+
country: selected.dataset.country,
|
|
149
|
+
language: selected.dataset.language,
|
|
150
|
+
};
|
|
151
|
+
};
|
|
152
|
+
|
|
153
|
+
// Initialize
|
|
154
|
+
populateCountrySelect();
|
|
155
|
+
prefillForm();
|
|
156
|
+
await checkSession();
|
|
157
|
+
|
|
158
|
+
// Event: Login with different account
|
|
159
|
+
$('btn-new-login')?.addEventListener('click', () => {
|
|
160
|
+
$('active-session-notice').classList.add('hidden');
|
|
161
|
+
$('login-form').classList.remove('hidden');
|
|
162
|
+
});
|
|
163
|
+
|
|
164
|
+
// Event: Login
|
|
165
|
+
$('btn-login').addEventListener('click', async () => {
|
|
166
|
+
const username = $('username').value.trim();
|
|
167
|
+
const password = $('password').value;
|
|
168
|
+
const { country, language } = getSelectedCountryLanguage();
|
|
169
|
+
|
|
170
|
+
if (!username || !password) {
|
|
171
|
+
homebridge.toast.error('Please enter your username and password');
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
$('login-spinner').classList.remove('hidden');
|
|
176
|
+
$('btn-login').disabled = true;
|
|
177
|
+
|
|
178
|
+
try {
|
|
179
|
+
const res = await homebridge.request('/login-by-user-pass', {
|
|
180
|
+
country,
|
|
181
|
+
language,
|
|
182
|
+
username,
|
|
183
|
+
password,
|
|
184
|
+
});
|
|
185
|
+
|
|
186
|
+
if (res.success) {
|
|
187
|
+
credentials = {
|
|
188
|
+
refresh_token: res.token,
|
|
189
|
+
country,
|
|
190
|
+
language,
|
|
191
|
+
};
|
|
192
|
+
|
|
193
|
+
// Update config
|
|
194
|
+
const newConfig = {
|
|
195
|
+
...config,
|
|
196
|
+
platform: 'LGThinQ',
|
|
197
|
+
name: config.name || 'LG ThinQ',
|
|
198
|
+
country,
|
|
199
|
+
language,
|
|
200
|
+
refresh_token: res.token,
|
|
201
|
+
auth_mode: 'token',
|
|
202
|
+
};
|
|
203
|
+
|
|
204
|
+
// Remember credentials if checked
|
|
205
|
+
if ($('rememberCredentials').checked) {
|
|
206
|
+
newConfig.username = username;
|
|
207
|
+
newConfig.password = password;
|
|
208
|
+
} else {
|
|
209
|
+
delete newConfig.username;
|
|
210
|
+
delete newConfig.password;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
await homebridge.updatePluginConfig([newConfig]);
|
|
214
|
+
await homebridge.savePluginConfig();
|
|
215
|
+
|
|
216
|
+
homebridge.toast.success('Successfully connected to LG ThinQ!');
|
|
217
|
+
|
|
218
|
+
await loadDevices();
|
|
219
|
+
showStep('devices');
|
|
220
|
+
} else {
|
|
221
|
+
homebridge.toast.error(res.error || 'Login failed');
|
|
222
|
+
}
|
|
223
|
+
} catch (e) {
|
|
224
|
+
homebridge.toast.error(e.message || 'Login failed');
|
|
225
|
+
} finally {
|
|
226
|
+
$('login-spinner').classList.add('hidden');
|
|
227
|
+
$('btn-login').disabled = false;
|
|
228
|
+
}
|
|
229
|
+
});
|
|
230
|
+
|
|
231
|
+
// Event: Refresh devices
|
|
232
|
+
$('btn-refresh').addEventListener('click', loadDevices);
|
|
233
|
+
|
|
234
|
+
// Event: Advanced settings
|
|
235
|
+
$('btn-schema').addEventListener('click', () => homebridge.showSchemaForm());
|
|
236
|
+
|
|
237
|
+
// Event: Save configuration
|
|
238
|
+
$('btn-save').addEventListener('click', async () => {
|
|
239
|
+
homebridge.showSpinner();
|
|
240
|
+
|
|
241
|
+
try {
|
|
242
|
+
const { country, language } = getSelectedCountryLanguage();
|
|
243
|
+
|
|
244
|
+
const newConfig = {
|
|
245
|
+
...config,
|
|
246
|
+
platform: 'LGThinQ',
|
|
247
|
+
name: config.name || 'LG ThinQ',
|
|
248
|
+
country,
|
|
249
|
+
language,
|
|
250
|
+
refresh_token: credentials.refresh_token,
|
|
251
|
+
thinq1: $('thinq1').checked,
|
|
252
|
+
auth_mode: 'token',
|
|
253
|
+
};
|
|
254
|
+
|
|
255
|
+
// Remember credentials if checked
|
|
256
|
+
if ($('rememberCredentials').checked) {
|
|
257
|
+
newConfig.username = $('username').value.trim();
|
|
258
|
+
newConfig.password = $('password').value;
|
|
259
|
+
} else {
|
|
260
|
+
delete newConfig.username;
|
|
261
|
+
delete newConfig.password;
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
await homebridge.updatePluginConfig([newConfig]);
|
|
265
|
+
await homebridge.savePluginConfig();
|
|
266
|
+
|
|
267
|
+
homebridge.toast.success('Configuration saved!');
|
|
268
|
+
showStep('complete');
|
|
269
|
+
} catch (e) {
|
|
270
|
+
homebridge.toast.error(e.message || 'Failed to save configuration');
|
|
271
|
+
} finally {
|
|
272
|
+
homebridge.hideSpinner();
|
|
273
|
+
}
|
|
274
|
+
});
|
|
275
|
+
|
|
276
|
+
// Event: Restart Homebridge
|
|
277
|
+
$('btn-restart').addEventListener('click', () => {
|
|
278
|
+
if (confirm('Are you sure you want to restart Homebridge?')) {
|
|
279
|
+
homebridge.request('/restart');
|
|
280
|
+
homebridge.closeSettings();
|
|
281
|
+
}
|
|
282
|
+
});
|
|
283
|
+
|
|
284
|
+
// Event: Config changes from schema form
|
|
285
|
+
homebridge.addEventListener('configChanged', e => Object.assign(config, e.data));
|
|
286
|
+
|
|
287
|
+
// Event: Country/language change
|
|
288
|
+
$('country_language').addEventListener('change', async () => {
|
|
289
|
+
const { country, language } = getSelectedCountryLanguage();
|
|
290
|
+
config.country = country;
|
|
291
|
+
config.language = language;
|
|
292
|
+
await homebridge.updatePluginConfig([{ ...config, country, language }]);
|
|
293
|
+
});
|
|
294
|
+
|
|
295
|
+
// Event: ThinQ1 checkbox change
|
|
296
|
+
$('thinq1').addEventListener('change', async () => {
|
|
297
|
+
config.thinq1 = $('thinq1').checked;
|
|
298
|
+
await homebridge.updatePluginConfig([{ ...config, thinq1: config.thinq1 }]);
|
|
299
|
+
});
|
|
300
|
+
})();
|
|
@@ -0,0 +1,233 @@
|
|
|
1
|
+
const COUNTRIES = [
|
|
2
|
+
{ country: "AE", language: "ar-AE", label: "United Arab Emirates/العربية" },
|
|
3
|
+
{ country: "AE", language: "en-AE", label: "United Arab Emirates/English" },
|
|
4
|
+
{ country: "AG", language: "en-AG", label: "Antigua and Barbuda/English" },
|
|
5
|
+
{ country: "AL", language: "en-AL", label: "Albania/English" },
|
|
6
|
+
{ country: "AL", language: "sq-AL", label: "Albania/Shqip" },
|
|
7
|
+
{ country: "AM", language: "en-AM", label: "Armenia/English" },
|
|
8
|
+
{ country: "AM", language: "fr-AM", label: "Armenia/français" },
|
|
9
|
+
{ country: "AO", language: "en-AO", label: "Angola/English" },
|
|
10
|
+
{ country: "AR", language: "es-AR", label: "Argentina/Spanish" },
|
|
11
|
+
{ country: "AT", language: "de-AT", label: "Austria/Deutsch" },
|
|
12
|
+
{ country: "AT", language: "en-AT", label: "Austria/English" },
|
|
13
|
+
{ country: "AU", language: "en-AU", label: "Australia/English" },
|
|
14
|
+
{ country: "AZ", language: "en-AZ", label: "Azerbaijan/English" },
|
|
15
|
+
{ country: "BA", language: "hr-BA", label: "Bosnia/Hrvatski" },
|
|
16
|
+
{ country: "BA", language: "sr-BA", label: "Bosnia/Srpski" },
|
|
17
|
+
{ country: "BB", language: "en-BB", label: "Barbados/English" },
|
|
18
|
+
{ country: "BD", language: "en-BD", label: "Bangladesh/English" },
|
|
19
|
+
{ country: "BE", language: "en-BE", label: "Belgium/English" },
|
|
20
|
+
{ country: "BE", language: "fr-BE", label: "Belgium/français" },
|
|
21
|
+
{ country: "BE", language: "nl-BE", label: "Belgium/Nederlands" },
|
|
22
|
+
{ country: "BF", language: "fr-BF", label: "Burkina Faso/français" },
|
|
23
|
+
{ country: "BG", language: "bg-BG", label: "Bulgaria/Български" },
|
|
24
|
+
{ country: "BG", language: "en-BG", label: "Bulgaria/English" },
|
|
25
|
+
{ country: "BH", language: "ar-BH", label: "Bahrain/العربية" },
|
|
26
|
+
{ country: "BH", language: "en-BH", label: "Bahrain/English" },
|
|
27
|
+
{ country: "BJ", language: "fr-BJ", label: "Benin/français" },
|
|
28
|
+
{ country: "BO", language: "es-BO", label: "Bolivia/Spanish" },
|
|
29
|
+
{ country: "BR", language: "pt-BR", label: "Brazil/Português" },
|
|
30
|
+
{ country: "BS", language: "en-BS", label: "Bahamas/English" },
|
|
31
|
+
{ country: "BY", language: "ru-BY", label: "Belarus/Русский" },
|
|
32
|
+
{ country: "BZ", language: "en-BZ", label: "Belize/English" },
|
|
33
|
+
{ country: "CA", language: "en-CA", label: "Canada/English" },
|
|
34
|
+
{ country: "CA", language: "fr-CA", label: "Canada/français" },
|
|
35
|
+
{ country: "CD", language: "fr-CD", label: "DR Congo/français" },
|
|
36
|
+
{ country: "CF", language: "fr-CF", label: "Central African Republic/français" },
|
|
37
|
+
{ country: "CG", language: "fr-CG", label: "Congo Brazzaville/français" },
|
|
38
|
+
{ country: "CH", language: "de-CH", label: "Switzerland/Deutsch" },
|
|
39
|
+
{ country: "CH", language: "en-CH", label: "Switzerland/English" },
|
|
40
|
+
{ country: "CH", language: "fr-CH", label: "Switzerland/français" },
|
|
41
|
+
{ country: "CI", language: "fr-CI", label: "Cote dKIvoire(Republic of Ivory Coast)/français" },
|
|
42
|
+
{ country: "CL", language: "en-CL", label: "Chile/English" },
|
|
43
|
+
{ country: "CL", language: "es-CL", label: "Chile/Spanish" },
|
|
44
|
+
{ country: "CM", language: "fr-CM", label: "Cameroon/français" },
|
|
45
|
+
{ country: "CN", language: "en-CN", label: "China/English" },
|
|
46
|
+
{ country: "CN", language: "zh-CN", label: "China/中国语" },
|
|
47
|
+
{ country: "CO", language: "en-CO", label: "Colombia/English" },
|
|
48
|
+
{ country: "CO", language: "es-CO", label: "Colombia/Spanish" },
|
|
49
|
+
{ country: "CR", language: "es-CR", label: "Costa Rica/Spanish" },
|
|
50
|
+
{ country: "CU", language: "es-CU", label: "Cuba/Spanish" },
|
|
51
|
+
{ country: "CV", language: "pt-CV", label: "Cape Verde/Português" },
|
|
52
|
+
{ country: "CY", language: "el-CY", label: "Cyprus/Ελληνικά" },
|
|
53
|
+
{ country: "CY", language: "en-CY", label: "Cyprus/English" },
|
|
54
|
+
{ country: "CZ", language: "cs-CZ", label: "Czech Republic/Česky" },
|
|
55
|
+
{ country: "CZ", language: "en-CZ", label: "Czech Republic/English" },
|
|
56
|
+
{ country: "DE", language: "de-DE", label: "Germany/Deutsch" },
|
|
57
|
+
{ country: "DE", language: "en-DE", label: "Germany/English" },
|
|
58
|
+
{ country: "DJ", language: "fr-DJ", label: "Djibouti/francais" },
|
|
59
|
+
{ country: "DK", language: "da-DK", label: "Denmark/Dansk" },
|
|
60
|
+
{ country: "DK", language: "en-DK", label: "Denmark/English" },
|
|
61
|
+
{ country: "DM", language: "en-DM", label: "Dominica/English" },
|
|
62
|
+
{ country: "DO", language: "es-DO", label: "Dominican Rep./Spanish" },
|
|
63
|
+
{ country: "DZ", language: "ar-DZ", label: "Algeria/العربية" },
|
|
64
|
+
{ country: "DZ", language: "en-DZ", label: "Algeria/English" },
|
|
65
|
+
{ country: "DZ", language: "fr-DZ", label: "Algeria/francais" },
|
|
66
|
+
{ country: "EC", language: "en-EC", label: "Ecuador/English" },
|
|
67
|
+
{ country: "EC", language: "es-EC", label: "Ecuador/Spanish" },
|
|
68
|
+
{ country: "EE", language: "en-EE", label: "Estonia/English" },
|
|
69
|
+
{ country: "EE", language: "et-EE", label: "Estonia/Eesti" },
|
|
70
|
+
{ country: "EE", language: "ru-EE", label: "Estonia/Русский" },
|
|
71
|
+
{ country: "EG", language: "ar-EG", label: "Egypt/العربية" },
|
|
72
|
+
{ country: "EG", language: "en-EG", label: "Egypt/English" },
|
|
73
|
+
{ country: "ES", language: "en-ES", label: "Spain/English" },
|
|
74
|
+
{ country: "ES", language: "es-ES", label: "Spain/Spanish" },
|
|
75
|
+
{ country: "ET", language: "en-ET", label: "Ethiopia/English" },
|
|
76
|
+
{ country: "FI", language: "en-FI", label: "Finland/English" },
|
|
77
|
+
{ country: "FI", language: "fi-FI", label: "Finland/Suomi" },
|
|
78
|
+
{ country: "FR", language: "fr-FR", label: "France/français" },
|
|
79
|
+
{ country: "GA", language: "fr-GA", label: "Gabon/français" },
|
|
80
|
+
{ country: "GB", language: "en-GB", label: "UK/English" },
|
|
81
|
+
{ country: "GD", language: "en-GD", label: "Grenada/English" },
|
|
82
|
+
{ country: "GE", language: "en-GE", label: "Georgia/English" },
|
|
83
|
+
{ country: "GH", language: "en-GH", label: "Ghana/English" },
|
|
84
|
+
{ country: "GM", language: "en-GM", label: "Gambia/English" },
|
|
85
|
+
{ country: "GN", language: "fr-GN", label: "Guinea Conakry/français" },
|
|
86
|
+
{ country: "GQ", language: "es-GQ", label: "Guinea Equatorial/Spanish" },
|
|
87
|
+
{ country: "GQ", language: "fr-GQ", label: "Guinea Equatorial/français" },
|
|
88
|
+
{ country: "GR", language: "el-GR", label: "Greece/Ελληνικά" },
|
|
89
|
+
{ country: "GR", language: "en-GR", label: "Greece/English" },
|
|
90
|
+
{ country: "GT", language: "es-GT", label: "Guatemala/Spanish" },
|
|
91
|
+
{ country: "GY", language: "en-GY", label: "Guyana/English" },
|
|
92
|
+
{ country: "HK", language: "en-HK", label: "Hong Kong/English" },
|
|
93
|
+
{ country: "HK", language: "zh-HK", label: "Hong Kong/中国语" },
|
|
94
|
+
{ country: "HN", language: "es-HN", label: "Honduras/Spanish" },
|
|
95
|
+
{ country: "HR", language: "en-HR", label: "Croatia/English" },
|
|
96
|
+
{ country: "HR", language: "hr-HR", label: "Croatia/Hrvatski" },
|
|
97
|
+
{ country: "HT", language: "fr-HT", label: "Haiti/français" },
|
|
98
|
+
{ country: "HU", language: "en-HU", label: "Hungary/English" },
|
|
99
|
+
{ country: "HU", language: "hu-HU", label: "Hungary/magyar" },
|
|
100
|
+
{ country: "ID", language: "en-ID", label: "Indonesia/English" },
|
|
101
|
+
{ country: "ID", language: "id-ID", label: "Indonesia/Bahasa Indonesia" },
|
|
102
|
+
{ country: "IE", language: "en-IE", label: "Ireland/English" },
|
|
103
|
+
{ country: "IL", language: "en-IL", label: "Israel/English" },
|
|
104
|
+
{ country: "IN", language: "en-IN", label: "India/English" },
|
|
105
|
+
{ country: "IQ", language: "ar-IQ", label: "Iraq/العربية" },
|
|
106
|
+
{ country: "IQ", language: "en-IQ", label: "Iraq/English" },
|
|
107
|
+
{ country: "IR", language: "en-IR", label: "Iran/English" },
|
|
108
|
+
{ country: "IS", language: "en-IS", label: "Iceland/English" },
|
|
109
|
+
{ country: "IT", language: "en-IT", label: "Italy/English" },
|
|
110
|
+
{ country: "IT", language: "it-IT", label: "Italy/Italiano" },
|
|
111
|
+
{ country: "JM", language: "en-JM", label: "Jamaica/English" },
|
|
112
|
+
{ country: "JO", language: "ar-JO", label: "Jordan/العربية" },
|
|
113
|
+
{ country: "JO", language: "en-JO", label: "Jordan/English" },
|
|
114
|
+
{ country: "JP", language: "ja-JP", label: "Japan/日本語" },
|
|
115
|
+
{ country: "KE", language: "en-KE", label: "Kenya/English" },
|
|
116
|
+
{ country: "KG", language: "ru-KG", label: "Kyrgyzstan/Русский" },
|
|
117
|
+
{ country: "KH", language: "en-KH", label: "Cambodia/English" },
|
|
118
|
+
{ country: "KN", language: "en-KN", label: "Saint Kitts and Nevis/English" },
|
|
119
|
+
{ country: "KR", language: "ko-KR", label: "Korea/한국어" },
|
|
120
|
+
{ country: "KW", language: "ar-KW", label: "Kuwait/العربية" },
|
|
121
|
+
{ country: "KW", language: "en-KW", label: "Kuwait/English" },
|
|
122
|
+
{ country: "KZ", language: "en-KZ", label: "Kazakhstan/English" },
|
|
123
|
+
{ country: "KZ", language: "kk-KZ", label: "Kazakhstan/Kazakh" },
|
|
124
|
+
{ country: "KZ", language: "ru-KZ", label: "Kazakhstan/Русский" },
|
|
125
|
+
{ country: "LB", language: "ar-LB", label: "Lebanon/العربية" },
|
|
126
|
+
{ country: "LB", language: "en-LB", label: "Lebanon/English" },
|
|
127
|
+
{ country: "LC", language: "en-LC", label: "Saint Lucia/English" },
|
|
128
|
+
{ country: "LK", language: "en-LK", label: "Sri Lanka/English" },
|
|
129
|
+
{ country: "LR", language: "en-LR", label: "Liberia/English" },
|
|
130
|
+
{ country: "LT", language: "en-LT", label: "Lithuania/English" },
|
|
131
|
+
{ country: "LT", language: "lt-LT", label: "Lithuania/Lietuvių" },
|
|
132
|
+
{ country: "LU", language: "de-LU", label: "Luxemburg/Deutsch" },
|
|
133
|
+
{ country: "LU", language: "en-LU", label: "Luxemburg/English" },
|
|
134
|
+
{ country: "LU", language: "fr-LU", label: "Luxemburg/français" },
|
|
135
|
+
{ country: "LV", language: "en-LV", label: "Latvia/English" },
|
|
136
|
+
{ country: "LV", language: "lv-LV", label: "Latvia/Latviešu" },
|
|
137
|
+
{ country: "LY", language: "ar-LY", label: "Libya/العربية" },
|
|
138
|
+
{ country: "LY", language: "en-LY", label: "Libya/English" },
|
|
139
|
+
{ country: "MA", language: "ar-MA", label: "Morocco/العربية" },
|
|
140
|
+
{ country: "MA", language: "en-MA", label: "Morocco/English" },
|
|
141
|
+
{ country: "MA", language: "fr-MA", label: "Morocco/français" },
|
|
142
|
+
{ country: "MD", language: "ro-MD", label: "Moldova/Română" },
|
|
143
|
+
{ country: "MD", language: "ru-MD", label: "Moldova/Русский" },
|
|
144
|
+
{ country: "ME", language: "en-ME", label: "Montenegro/English" },
|
|
145
|
+
{ country: "ME", language: "sr-ME", label: "Montenegro/Srpski" },
|
|
146
|
+
{ country: "MK", language: "en-MK", label: "Macedonia/English" },
|
|
147
|
+
{ country: "MK", language: "mk-MK", label: "Macedonia/Македонски" },
|
|
148
|
+
{ country: "ML", language: "fr-ML", label: "Mali/français" },
|
|
149
|
+
{ country: "MM", language: "en-MM", label: "Myanmar (Burma)/English" },
|
|
150
|
+
{ country: "MT", language: "en-MT", label: "Malta/English" },
|
|
151
|
+
{ country: "MU", language: "en-MU", label: "Mauritius/English" },
|
|
152
|
+
{ country: "MX", language: "en-MX", label: "Mexico/English" },
|
|
153
|
+
{ country: "MX", language: "es-MX", label: "Mexico/Spanish" },
|
|
154
|
+
{ country: "MY", language: "en-MY", label: "Malaysia/English" },
|
|
155
|
+
{ country: "NE", language: "fr-NE", label: "Niger/français" },
|
|
156
|
+
{ country: "NG", language: "en-NG", label: "Nigeria/English" },
|
|
157
|
+
{ country: "NI", language: "es-NI", label: "Nicaragua/Spanish" },
|
|
158
|
+
{ country: "NL", language: "en-NL", label: "Netherlands/English" },
|
|
159
|
+
{ country: "NL", language: "nl-NL", label: "Netherlands/Nederlands" },
|
|
160
|
+
{ country: "NO", language: "en-NO", label: "Norway/English" },
|
|
161
|
+
{ country: "NO", language: "no-NO", label: "Norway/Norsk" },
|
|
162
|
+
{ country: "NZ", language: "en-NZ", label: "New Zealand/English" },
|
|
163
|
+
{ country: "OM", language: "ar-OM", label: "Oman/العربية" },
|
|
164
|
+
{ country: "OM", language: "en-OM", label: "Oman/English" },
|
|
165
|
+
{ country: "PA", language: "en-PA", label: "Panama/English" },
|
|
166
|
+
{ country: "PA", language: "es-PA", label: "Panama/Spanish" },
|
|
167
|
+
{ country: "PE", language: "en-PE", label: "Peru/English" },
|
|
168
|
+
{ country: "PE", language: "es-PE", label: "Peru/Spanish" },
|
|
169
|
+
{ country: "PH", language: "en-PH", label: "Philippines/English" },
|
|
170
|
+
{ country: "PK", language: "en-PK", label: "Pakistan/English" },
|
|
171
|
+
{ country: "PL", language: "en-PL", label: "Poland/English" },
|
|
172
|
+
{ country: "PL", language: "pl-PL", label: "Poland/Polski" },
|
|
173
|
+
{ country: "PR", language: "es-PR", label: "Puerto Rico/Spanish" },
|
|
174
|
+
{ country: "PS", language: "en-PS", label: "Palestine/English" },
|
|
175
|
+
{ country: "PT", language: "en-PT", label: "Portugal/English" },
|
|
176
|
+
{ country: "PT", language: "pt-PT", label: "Portugal/Português" },
|
|
177
|
+
{ country: "PY", language: "es-PY", label: "Paraguay/Spanish" },
|
|
178
|
+
{ country: "QA", language: "ar-QA", label: "Qatar/العربية" },
|
|
179
|
+
{ country: "QA", language: "en-QA", label: "Qatar/English" },
|
|
180
|
+
{ country: "RO", language: "en-RO", label: "Romania/English" },
|
|
181
|
+
{ country: "RO", language: "ro-RO", label: "Romania/Română" },
|
|
182
|
+
{ country: "RS", language: "en-RS", label: "Serbia/English" },
|
|
183
|
+
{ country: "RS", language: "sr-RS", label: "Serbia/Srpski" },
|
|
184
|
+
{ country: "RU", language: "ru-RU", label: "Russia/Русский" },
|
|
185
|
+
{ country: "RW", language: "fr-RW", label: "Rwanda/français" },
|
|
186
|
+
{ country: "SA", language: "ar-SA", label: "Saudi arabia/العربية" },
|
|
187
|
+
{ country: "SA", language: "en-SA", label: "Saudi arabia/English" },
|
|
188
|
+
{ country: "SD", language: "en-SD", label: "Sudan/English" },
|
|
189
|
+
{ country: "SE", language: "en-SE", label: "Sweden/English" },
|
|
190
|
+
{ country: "SE", language: "sv-SE", label: "Sweden/Svenska" },
|
|
191
|
+
{ country: "SG", language: "en-SG", label: "Singapore/English" },
|
|
192
|
+
{ country: "SI", language: "en-SI", label: "Slovenia/English" },
|
|
193
|
+
{ country: "SI", language: "sl-SI", label: "Slovenia/Slovenščina" },
|
|
194
|
+
{ country: "SK", language: "en-SK", label: "Slovakia/English" },
|
|
195
|
+
{ country: "SK", language: "sk-SK", label: "Slovakia/Slovenčina" },
|
|
196
|
+
{ country: "SL", language: "en-SL", label: "Sierra Leone/English" },
|
|
197
|
+
{ country: "SN", language: "fr-SN", label: "Senegal/français" },
|
|
198
|
+
{ country: "SO", language: "en-SO", label: "Somalia/English" },
|
|
199
|
+
{ country: "SR", language: "nl-SR", label: "Republic of Suriname/Nederlands" },
|
|
200
|
+
{ country: "ST", language: "pt-ST", label: "Sao Tome and Principe/Português" },
|
|
201
|
+
{ country: "SV", language: "es-SV", label: "El salvador/Spanish" },
|
|
202
|
+
{ country: "SY", language: "ar-SY", label: "Syria/العربية" },
|
|
203
|
+
{ country: "SY", language: "en-SY", label: "Syria/English" },
|
|
204
|
+
{ country: "TD", language: "fr-TD", label: "Chad/français" },
|
|
205
|
+
{ country: "TG", language: "fr-TG", label: "Togo/français" },
|
|
206
|
+
{ country: "TH", language: "en-TH", label: "Thailand/English" },
|
|
207
|
+
{ country: "TH", language: "th-TH", label: "Thailand/ไทย" },
|
|
208
|
+
{ country: "TN", language: "ar-TN", label: "Tunisia/العربية" },
|
|
209
|
+
{ country: "TN", language: "en-TN", label: "Tunisia/English" },
|
|
210
|
+
{ country: "TR", language: "en-TR", label: "Turkey/English" },
|
|
211
|
+
{ country: "TR", language: "tr-TR", label: "Turkey/Türkçe" },
|
|
212
|
+
{ country: "TT", language: "en-TT", label: "Trinidad/English" },
|
|
213
|
+
{ country: "TW", language: "en-TW", label: "Taiwan/English" },
|
|
214
|
+
{ country: "TW", language: "zh-TW", label: "Taiwan/繁體中文" },
|
|
215
|
+
{ country: "TZ", language: "en-TZ", label: "Tanzania/English" },
|
|
216
|
+
{ country: "UA", language: "ru-UA", label: "Ukraine/Русский" },
|
|
217
|
+
{ country: "UA", language: "uk-UA", label: "Ukraine/Українська мова" },
|
|
218
|
+
{ country: "UG", language: "en-UG", label: "Uganda/English" },
|
|
219
|
+
{ country: "US", language: "en-US", label: "USA/English" },
|
|
220
|
+
{ country: "UY", language: "es-UY", label: "Uruguay/Spanish" },
|
|
221
|
+
{ country: "UZ", language: "en-UZ", label: "Uzbekistan/English" },
|
|
222
|
+
{ country: "UZ", language: "ru-UZ", label: "Uzbekistan/Русский" },
|
|
223
|
+
{ country: "UZ", language: "uz-UZ", label: "Uzbekistan/O'zbek tili" },
|
|
224
|
+
{ country: "VC", language: "en-VC", label: "Saint Vincent/English" },
|
|
225
|
+
{ country: "VE", language: "es-VE", label: "Venezuela/Spanish" },
|
|
226
|
+
{ country: "VN", language: "en-VN", label: "Vietnam/English" },
|
|
227
|
+
{ country: "VN", language: "vi-VN", label: "Vietnam/Tiếng Việt" },
|
|
228
|
+
{ country: "XK", language: "en-XK", label: "Kosovo/English" },
|
|
229
|
+
{ country: "YE", language: "ar-YE", label: "Yemen/العربية" },
|
|
230
|
+
{ country: "YE", language: "en-YE", label: "Yemen/English" },
|
|
231
|
+
{ country: "ZA", language: "en-ZA", label: "South Africa/English" },
|
|
232
|
+
{ country: "ZM", language: "en-ZM", label: "Zambia/English" },
|
|
233
|
+
];
|