@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.
Files changed (241) hide show
  1. package/.claude/settings.local.json +15 -0
  2. package/CHANGELOG.md +98 -0
  3. package/LICENSE +176 -0
  4. package/README.md +114 -0
  5. package/config.schema.json +399 -0
  6. package/dist/__tests__/baseDevice.spec.d.ts +1 -0
  7. package/dist/__tests__/baseDevice.spec.js +96 -0
  8. package/dist/__tests__/baseDevice.spec.js.map +1 -0
  9. package/dist/__tests__/deviceControl.coercion.spec.d.ts +1 -0
  10. package/dist/__tests__/deviceControl.coercion.spec.js +53 -0
  11. package/dist/__tests__/deviceControl.coercion.spec.js.map +1 -0
  12. package/dist/__tests__/helper.spec.d.ts +1 -0
  13. package/dist/__tests__/helper.spec.js +74 -0
  14. package/dist/__tests__/helper.spec.js.map +1 -0
  15. package/dist/baseDevice.d.ts +40 -0
  16. package/dist/baseDevice.js +85 -0
  17. package/dist/baseDevice.js.map +1 -0
  18. package/dist/baseDevice.spec.d.ts +1 -0
  19. package/dist/baseDevice.spec.js +107 -0
  20. package/dist/baseDevice.spec.js.map +1 -0
  21. package/dist/characteristics/TotalConsumption.d.ts +2 -0
  22. package/dist/characteristics/TotalConsumption.js +17 -0
  23. package/dist/characteristics/TotalConsumption.js.map +1 -0
  24. package/dist/characteristics/index.d.ts +2 -0
  25. package/dist/characteristics/index.js +7 -0
  26. package/dist/characteristics/index.js.map +1 -0
  27. package/dist/cli.d.ts +2 -0
  28. package/dist/cli.js +89 -0
  29. package/dist/cli.js.map +1 -0
  30. package/dist/devices/AeroTower.d.ts +24 -0
  31. package/dist/devices/AeroTower.js +113 -0
  32. package/dist/devices/AeroTower.js.map +1 -0
  33. package/dist/devices/AirConditioner.d.ts +425 -0
  34. package/dist/devices/AirConditioner.js +1253 -0
  35. package/dist/devices/AirConditioner.js.map +1 -0
  36. package/dist/devices/AirPurifier.d.ts +50 -0
  37. package/dist/devices/AirPurifier.js +281 -0
  38. package/dist/devices/AirPurifier.js.map +1 -0
  39. package/dist/devices/Dehumidifier.d.ts +28 -0
  40. package/dist/devices/Dehumidifier.js +175 -0
  41. package/dist/devices/Dehumidifier.js.map +1 -0
  42. package/dist/devices/Dishwasher.d.ts +64 -0
  43. package/dist/devices/Dishwasher.js +740 -0
  44. package/dist/devices/Dishwasher.js.map +1 -0
  45. package/dist/devices/Microwave.d.ts +128 -0
  46. package/dist/devices/Microwave.js +1939 -0
  47. package/dist/devices/Microwave.js.map +1 -0
  48. package/dist/devices/Oven.d.ts +148 -0
  49. package/dist/devices/Oven.js +1850 -0
  50. package/dist/devices/Oven.js.map +1 -0
  51. package/dist/devices/RangeHood.d.ts +16 -0
  52. package/dist/devices/RangeHood.js +99 -0
  53. package/dist/devices/RangeHood.js.map +1 -0
  54. package/dist/devices/Refrigerator.d.ts +50 -0
  55. package/dist/devices/Refrigerator.js +325 -0
  56. package/dist/devices/Refrigerator.js.map +1 -0
  57. package/dist/devices/Styler.d.ts +27 -0
  58. package/dist/devices/Styler.js +76 -0
  59. package/dist/devices/Styler.js.map +1 -0
  60. package/dist/devices/WasherDryer.d.ts +39 -0
  61. package/dist/devices/WasherDryer.js +170 -0
  62. package/dist/devices/WasherDryer.js.map +1 -0
  63. package/dist/devices/WasherDryer2.d.ts +9 -0
  64. package/dist/devices/WasherDryer2.js +16 -0
  65. package/dist/devices/WasherDryer2.js.map +1 -0
  66. package/dist/errors/AuthenticationError.d.ts +2 -0
  67. package/dist/errors/AuthenticationError.js +3 -0
  68. package/dist/errors/AuthenticationError.js.map +1 -0
  69. package/dist/errors/ManualProcessNeeded.d.ts +3 -0
  70. package/dist/errors/ManualProcessNeeded.js +4 -0
  71. package/dist/errors/ManualProcessNeeded.js.map +1 -0
  72. package/dist/errors/MonitorError.d.ts +2 -0
  73. package/dist/errors/MonitorError.js +3 -0
  74. package/dist/errors/MonitorError.js.map +1 -0
  75. package/dist/errors/NotConnectedError.d.ts +3 -0
  76. package/dist/errors/NotConnectedError.js +4 -0
  77. package/dist/errors/NotConnectedError.js.map +1 -0
  78. package/dist/errors/TokenError.d.ts +2 -0
  79. package/dist/errors/TokenError.js +3 -0
  80. package/dist/errors/TokenError.js.map +1 -0
  81. package/dist/errors/TokenExpiredError.d.ts +3 -0
  82. package/dist/errors/TokenExpiredError.js +4 -0
  83. package/dist/errors/TokenExpiredError.js.map +1 -0
  84. package/dist/errors/index.d.ts +6 -0
  85. package/dist/errors/index.js +7 -0
  86. package/dist/errors/index.js.map +1 -0
  87. package/dist/helper.d.ts +24 -0
  88. package/dist/helper.js +66 -0
  89. package/dist/helper.js.map +1 -0
  90. package/dist/helper.spec.d.ts +1 -0
  91. package/dist/helper.spec.js +74 -0
  92. package/dist/helper.spec.js.map +1 -0
  93. package/dist/index.d.ts +6 -0
  94. package/dist/index.js +9 -0
  95. package/dist/index.js.map +1 -0
  96. package/dist/lib/API.d.ts +141 -0
  97. package/dist/lib/API.js +362 -0
  98. package/dist/lib/API.js.map +1 -0
  99. package/dist/lib/API.spec.d.ts +1 -0
  100. package/dist/lib/API.spec.js +55 -0
  101. package/dist/lib/API.spec.js.map +1 -0
  102. package/dist/lib/Auth.d.ts +99 -0
  103. package/dist/lib/Auth.js +348 -0
  104. package/dist/lib/Auth.js.map +1 -0
  105. package/dist/lib/Auth.spec.d.ts +1 -0
  106. package/dist/lib/Auth.spec.js +111 -0
  107. package/dist/lib/Auth.spec.js.map +1 -0
  108. package/dist/lib/Device.d.ts +88 -0
  109. package/dist/lib/Device.js +95 -0
  110. package/dist/lib/Device.js.map +1 -0
  111. package/dist/lib/Device.spec.d.ts +1 -0
  112. package/dist/lib/Device.spec.js +53 -0
  113. package/dist/lib/Device.spec.js.map +1 -0
  114. package/dist/lib/DeviceModel.d.ts +164 -0
  115. package/dist/lib/DeviceModel.js +321 -0
  116. package/dist/lib/DeviceModel.js.map +1 -0
  117. package/dist/lib/DeviceModel.spec.d.ts +1 -0
  118. package/dist/lib/DeviceModel.spec.js +90 -0
  119. package/dist/lib/DeviceModel.spec.js.map +1 -0
  120. package/dist/lib/Gateway.d.ts +18 -0
  121. package/dist/lib/Gateway.js +25 -0
  122. package/dist/lib/Gateway.js.map +1 -0
  123. package/dist/lib/Gateway.spec.d.ts +1 -0
  124. package/dist/lib/Gateway.spec.js +35 -0
  125. package/dist/lib/Gateway.spec.js.map +1 -0
  126. package/dist/lib/Persist.d.ts +101 -0
  127. package/dist/lib/Persist.js +245 -0
  128. package/dist/lib/Persist.js.map +1 -0
  129. package/dist/lib/Persist.spec.d.ts +1 -0
  130. package/dist/lib/Persist.spec.js +90 -0
  131. package/dist/lib/Persist.spec.js.map +1 -0
  132. package/dist/lib/Session.d.ts +80 -0
  133. package/dist/lib/Session.js +100 -0
  134. package/dist/lib/Session.js.map +1 -0
  135. package/dist/lib/Session.spec.d.ts +1 -0
  136. package/dist/lib/Session.spec.js +43 -0
  137. package/dist/lib/Session.spec.js.map +1 -0
  138. package/dist/lib/ThinQ.d.ts +28 -0
  139. package/dist/lib/ThinQ.js +373 -0
  140. package/dist/lib/ThinQ.js.map +1 -0
  141. package/dist/lib/__tests__/API.spec.d.ts +1 -0
  142. package/dist/lib/__tests__/API.spec.js +55 -0
  143. package/dist/lib/__tests__/API.spec.js.map +1 -0
  144. package/dist/lib/__tests__/Auth.spec.d.ts +1 -0
  145. package/dist/lib/__tests__/Auth.spec.js +110 -0
  146. package/dist/lib/__tests__/Auth.spec.js.map +1 -0
  147. package/dist/lib/__tests__/Device.spec.d.ts +1 -0
  148. package/dist/lib/__tests__/Device.spec.js +53 -0
  149. package/dist/lib/__tests__/Device.spec.js.map +1 -0
  150. package/dist/lib/__tests__/DeviceModel.spec.d.ts +1 -0
  151. package/dist/lib/__tests__/DeviceModel.spec.js +90 -0
  152. package/dist/lib/__tests__/DeviceModel.spec.js.map +1 -0
  153. package/dist/lib/__tests__/Gateway.spec.d.ts +1 -0
  154. package/dist/lib/__tests__/Gateway.spec.js +35 -0
  155. package/dist/lib/__tests__/Gateway.spec.js.map +1 -0
  156. package/dist/lib/__tests__/Persist.spec.d.ts +1 -0
  157. package/dist/lib/__tests__/Persist.spec.js +90 -0
  158. package/dist/lib/__tests__/Persist.spec.js.map +1 -0
  159. package/dist/lib/__tests__/Session.spec.d.ts +1 -0
  160. package/dist/lib/__tests__/Session.spec.js +43 -0
  161. package/dist/lib/__tests__/Session.spec.js.map +1 -0
  162. package/dist/lib/constants.d.ts +95 -0
  163. package/dist/lib/constants.js +106 -0
  164. package/dist/lib/constants.js.map +1 -0
  165. package/dist/lib/request.d.ts +2 -0
  166. package/dist/lib/request.js +66 -0
  167. package/dist/lib/request.js.map +1 -0
  168. package/dist/platform.d.ts +41 -0
  169. package/dist/platform.js +229 -0
  170. package/dist/platform.js.map +1 -0
  171. package/dist/settings.d.ts +8 -0
  172. package/dist/settings.js +9 -0
  173. package/dist/settings.js.map +1 -0
  174. package/dist/status/BaseStatus.d.ts +48 -0
  175. package/dist/status/BaseStatus.js +59 -0
  176. package/dist/status/BaseStatus.js.map +1 -0
  177. package/dist/types/snapshots.d.ts +142 -0
  178. package/dist/types/snapshots.js +6 -0
  179. package/dist/types/snapshots.js.map +1 -0
  180. package/dist/utils/__tests__/normalize.spec.d.ts +1 -0
  181. package/dist/utils/__tests__/normalize.spec.js +95 -0
  182. package/dist/utils/__tests__/normalize.spec.js.map +1 -0
  183. package/dist/utils/normalize.d.ts +22 -0
  184. package/dist/utils/normalize.js +51 -0
  185. package/dist/utils/normalize.js.map +1 -0
  186. package/dist/v1/__tests__/prepareControlData.spec.d.ts +1 -0
  187. package/dist/v1/__tests__/prepareControlData.spec.js +48 -0
  188. package/dist/v1/__tests__/prepareControlData.spec.js.map +1 -0
  189. package/dist/v1/devices/AC.d.ts +13 -0
  190. package/dist/v1/devices/AC.js +112 -0
  191. package/dist/v1/devices/AC.js.map +1 -0
  192. package/dist/v1/devices/AirPurifier.d.ts +15 -0
  193. package/dist/v1/devices/AirPurifier.js +57 -0
  194. package/dist/v1/devices/AirPurifier.js.map +1 -0
  195. package/dist/v1/devices/RangeHood.d.ts +6 -0
  196. package/dist/v1/devices/RangeHood.js +12 -0
  197. package/dist/v1/devices/RangeHood.js.map +1 -0
  198. package/dist/v1/devices/Refrigerator.d.ts +17 -0
  199. package/dist/v1/devices/Refrigerator.js +69 -0
  200. package/dist/v1/devices/Refrigerator.js.map +1 -0
  201. package/dist/v1/devices/Washer.d.ts +10 -0
  202. package/dist/v1/devices/Washer.js +23 -0
  203. package/dist/v1/devices/Washer.js.map +1 -0
  204. package/dist/v1/devices/index.d.ts +6 -0
  205. package/dist/v1/devices/index.js +7 -0
  206. package/dist/v1/devices/index.js.map +1 -0
  207. package/dist/v1/helper.d.ts +14 -0
  208. package/dist/v1/helper.js +111 -0
  209. package/dist/v1/helper.js.map +1 -0
  210. package/dist/v1/transforms/AirPurifierState.d.ts +2 -0
  211. package/dist/v1/transforms/AirPurifierState.js +9 -0
  212. package/dist/v1/transforms/AirPurifierState.js.map +1 -0
  213. package/dist/v1/transforms/AirState.d.ts +9 -0
  214. package/dist/v1/transforms/AirState.js +55 -0
  215. package/dist/v1/transforms/AirState.js.map +1 -0
  216. package/dist/v1/transforms/HoodState.d.ts +17 -0
  217. package/dist/v1/transforms/HoodState.js +20 -0
  218. package/dist/v1/transforms/HoodState.js.map +1 -0
  219. package/dist/v1/transforms/RefState.d.ts +6 -0
  220. package/dist/v1/transforms/RefState.js +29 -0
  221. package/dist/v1/transforms/RefState.js.map +1 -0
  222. package/dist/v1/transforms/WasherDryer.d.ts +49 -0
  223. package/dist/v1/transforms/WasherDryer.js +56 -0
  224. package/dist/v1/transforms/WasherDryer.js.map +1 -0
  225. package/docs/authorization.md +40 -0
  226. package/docs/device-configuration.md +68 -0
  227. package/homebridge-ui/public/index.html +120 -0
  228. package/homebridge-ui/public/js/app.js +300 -0
  229. package/homebridge-ui/public/js/countries.js +233 -0
  230. package/homebridge-ui/public/styles.css +185 -0
  231. package/homebridge-ui/server.js +103 -0
  232. package/jest.config.ts +39 -0
  233. package/package.json +83 -0
  234. package/sample/README.md +10 -0
  235. package/sample/airconditioner-model.json +3080 -0
  236. package/sample/airconditioner-snapshot.json +49 -0
  237. package/sample/airconditioner.json +157 -0
  238. package/sample/dishwasher-model.json +869 -0
  239. package/sample/dishwasher.json +125 -0
  240. package/sample/washer_dryer-model.json +1294 -0
  241. 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
+ ];