1 /*
2 * Copyright (c) 2022-2025 Huawei Device Co., Ltd.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16 #include "js_ui_appearance.h"
17
18 #include <string>
19 #include "js_native_api.h"
20 #include "ui_appearance_log.h"
21
22 namespace OHOS {
23 namespace ArkUi::UiAppearance {
24 namespace {
25 static constexpr size_t ARGC_WITH_ONE = 1;
26 static constexpr size_t ARGC_WITH_TWO = 2;
27 static constexpr size_t MAX_FONT_SCALE = 5;
28 static constexpr size_t MIN_FONT_SCALE = 0;
29 const std::string PERMISSION_ERR_MSG =
30 "An attempt was made to update configuration forbidden by permission: ohos.permission.UPDATE_CONFIGURATION.";
31 const std::string INVALID_ARG_MSG = "The type of 'mode' must be DarkMode.";
32
ParseErrCode(const int32_t errCode)33 std::string ParseErrCode(const int32_t errCode)
34 {
35 switch (errCode) {
36 case UiAppearanceAbilityErrCode::PERMISSION_ERR:
37 return "Permission denied. ";
38 case UiAppearanceAbilityErrCode::INVALID_ARG:
39 return "Parameter error. ";
40 case UiAppearanceAbilityErrCode::SYS_ERR:
41 return "Internal error. ";
42 default:
43 return "";
44 }
45 }
46
NapiThrow(napi_env env,const std::string & message,int32_t errCode)47 void NapiThrow(napi_env env, const std::string& message, int32_t errCode)
48 {
49 napi_value code = nullptr;
50 std::string strCode = std::to_string(errCode);
51 napi_create_string_utf8(env, strCode.c_str(), strCode.length(), &code);
52
53 napi_value msg = nullptr;
54 std::string strMsg = ParseErrCode(errCode) + message;
55 LOGI("napi throw errCode %{public}d, strMsg %{public}s", errCode, strMsg.c_str());
56 napi_create_string_utf8(env, strMsg.c_str(), strMsg.length(), &msg);
57
58 napi_value error = nullptr;
59 napi_create_error(env, code, msg, &error);
60 napi_throw(env, error);
61 }
62 } // namespace
63
OnExecute(napi_env env,void * data)64 void JsUiAppearance::OnExecute(napi_env env, void* data)
65 {
66 LOGI("OnExecute begin.");
67 AsyncContext* asyncContext = static_cast<AsyncContext*>(data);
68 if (asyncContext == nullptr) {
69 NapiThrow(env, "asyncContext is null.", UiAppearanceAbilityErrCode::SYS_ERR);
70 return;
71 }
72 auto resCode = UiAppearanceAbilityClient::GetInstance()->SetDarkMode(asyncContext->mode);
73 asyncContext->status = static_cast<UiAppearanceAbilityErrCode>(resCode);
74 if (asyncContext->status == UiAppearanceAbilityErrCode::PERMISSION_ERR) {
75 asyncContext->errMsg = PERMISSION_ERR_MSG;
76 } else if (asyncContext->status == UiAppearanceAbilityErrCode::INVALID_ARG) {
77 asyncContext->errMsg = INVALID_ARG_MSG;
78 } else {
79 asyncContext->errMsg = "";
80 }
81 }
82
OnSetFontScale(napi_env env,void * data)83 void JsUiAppearance::OnSetFontScale(napi_env env, void* data)
84 {
85 LOGI("OnSetFontScale begin.");
86 AsyncContext* asyncContext = static_cast<AsyncContext*>(data);
87 if (asyncContext == nullptr) {
88 NapiThrow(env, "asyncContext is null.", UiAppearanceAbilityErrCode::SYS_ERR);
89 return;
90 }
91 int32_t resCode = 0;
92 if (asyncContext->jsFontScale <= MIN_FONT_SCALE || asyncContext->jsFontScale > MAX_FONT_SCALE) {
93 resCode = UiAppearanceAbilityErrCode::INVALID_ARG;
94 } else {
95 resCode = UiAppearanceAbilityClient::GetInstance()->SetFontScale(asyncContext->fontScale);
96 }
97
98 asyncContext->status = static_cast<UiAppearanceAbilityErrCode>(resCode);
99 if (asyncContext->status == UiAppearanceAbilityErrCode::PERMISSION_ERR) {
100 asyncContext->errMsg = PERMISSION_ERR_MSG;
101 } else if (asyncContext->status == UiAppearanceAbilityErrCode::INVALID_ARG) {
102 asyncContext->errMsg = "fontScale must between 0 and 5";
103 } else {
104 asyncContext->errMsg = "";
105 }
106 }
107
OnSetFontWeightScale(napi_env env,void * data)108 void JsUiAppearance::OnSetFontWeightScale(napi_env env, void* data)
109 {
110 LOGI("OnSetFontWeightScale begin.");
111 AsyncContext* asyncContext = static_cast<AsyncContext*>(data);
112 if (asyncContext == nullptr) {
113 NapiThrow(env, "asyncContext is null.", UiAppearanceAbilityErrCode::SYS_ERR);
114 return;
115 }
116 int32_t resCode = 0;
117 if (asyncContext->jsFontWeightScale <= MIN_FONT_SCALE ||
118 asyncContext->jsFontWeightScale > MAX_FONT_SCALE) {
119 resCode = UiAppearanceAbilityErrCode::INVALID_ARG;
120 } else {
121 resCode = UiAppearanceAbilityClient::GetInstance()
122 ->SetFontWeightScale(asyncContext->fontWeightScale);
123 }
124
125 asyncContext->status = static_cast<UiAppearanceAbilityErrCode>(resCode);
126 if (asyncContext->status == UiAppearanceAbilityErrCode::PERMISSION_ERR) {
127 asyncContext->errMsg = PERMISSION_ERR_MSG;
128 } else if (asyncContext->status == UiAppearanceAbilityErrCode::INVALID_ARG) {
129 asyncContext->errMsg = "fontWeightScale must between 0 and 5";
130 } else {
131 asyncContext->errMsg = "";
132 }
133 }
134
OnComplete(napi_env env,napi_status status,void * data)135 void JsUiAppearance::OnComplete(napi_env env, napi_status status, void* data)
136 {
137 LOGI("OnComplete begin.");
138 napi_handle_scope scope = nullptr;
139 napi_open_handle_scope(env, &scope);
140 if (scope == nullptr) {
141 NapiThrow(env, "open handle scope failed.", UiAppearanceAbilityErrCode::SYS_ERR);
142 return;
143 }
144 AsyncContext* asyncContext = static_cast<AsyncContext*>(data);
145 if (asyncContext == nullptr) {
146 NapiThrow(env, "asyncContext is null.", UiAppearanceAbilityErrCode::SYS_ERR);
147 return;
148 }
149
150 if (asyncContext->status == UiAppearanceAbilityErrCode::SUCCEEDED) {
151 napi_value result = nullptr;
152 napi_get_null(env, &result);
153 if (asyncContext->deferred) { // promise
154 napi_resolve_deferred(env, asyncContext->deferred, result);
155 } else { // AsyncCallback
156 napi_value callback = nullptr;
157 napi_get_reference_value(env, asyncContext->callbackRef, &callback);
158 napi_value ret;
159 napi_call_function(env, nullptr, callback, 1, &result, &ret);
160 }
161 } else {
162 napi_value code = nullptr;
163 std::string strCode = std::to_string(asyncContext->status);
164 napi_create_string_utf8(env, strCode.c_str(), strCode.length(), &code);
165
166 napi_value msg = nullptr;
167 std::string strMsg = ParseErrCode(asyncContext->status) + asyncContext->errMsg;
168 LOGI("napi throw errCode %{public}d, strMsg %{public}s", asyncContext->status, strMsg.c_str());
169 napi_create_string_utf8(env, strMsg.c_str(), strMsg.length(), &msg);
170
171 napi_value error = nullptr;
172 napi_create_error(env, code, msg, &error);
173 if (asyncContext->deferred) { // promise
174 napi_reject_deferred(env, asyncContext->deferred, error);
175 } else { // AsyncCallback
176 napi_value callback = nullptr;
177 napi_get_reference_value(env, asyncContext->callbackRef, &callback);
178 napi_value ret;
179 napi_call_function(env, nullptr, callback, 1, &error, &ret);
180 }
181 }
182 napi_delete_async_work(env, asyncContext->work);
183 delete asyncContext;
184 napi_close_handle_scope(env, scope);
185 }
186
CheckArgs(napi_env env,size_t argc,napi_value * argv)187 napi_status JsUiAppearance::CheckArgs(napi_env env, size_t argc, napi_value* argv)
188 {
189 if (argc != ARGC_WITH_ONE && argc != ARGC_WITH_TWO) {
190 NapiThrow(
191 env, "the number of parameters can only be 1 or 2.", UiAppearanceAbilityErrCode::INVALID_ARG);
192 return napi_invalid_arg;
193 }
194
195 napi_valuetype valueType = napi_undefined;
196 switch (argc) {
197 case ARGC_WITH_TWO:
198 napi_typeof(env, argv[1], &valueType);
199 if (valueType != napi_function) {
200 NapiThrow(env, "the second parameter must be a function.",
201 UiAppearanceAbilityErrCode::INVALID_ARG);
202 return napi_invalid_arg;
203 }
204 [[fallthrough]];
205 case ARGC_WITH_ONE:
206 napi_typeof(env, argv[0], &valueType);
207 if (valueType != napi_number) {
208 NapiThrow(
209 env, "the first parameter must be DarkMode.", UiAppearanceAbilityErrCode::INVALID_ARG);
210 return napi_invalid_arg;
211 }
212 break;
213 default:
214 return napi_invalid_arg;
215 }
216 return napi_ok;
217 }
218
CheckFontScaleArgs(napi_env env,size_t argc,napi_value * argv)219 napi_status JsUiAppearance::CheckFontScaleArgs(napi_env env, size_t argc, napi_value* argv)
220 {
221 if (argc != ARGC_WITH_ONE && argc != ARGC_WITH_TWO) {
222 NapiThrow(
223 env, "the number of parameters can only be 1 or 2.", UiAppearanceAbilityErrCode::INVALID_ARG);
224 return napi_invalid_arg;
225 }
226
227 napi_valuetype valueType = napi_undefined;
228 switch (argc) {
229 case ARGC_WITH_TWO:
230 napi_typeof(env, argv[1], &valueType);
231 if (valueType != napi_function) {
232 NapiThrow(env, "the second parameter must be a function.",
233 UiAppearanceAbilityErrCode::INVALID_ARG);
234 return napi_invalid_arg;
235 }
236 [[fallthrough]];
237 case ARGC_WITH_ONE:
238 napi_typeof(env, argv[0], &valueType);
239 if (valueType != napi_number) {
240 NapiThrow(
241 env, "the first parameter must be Number.", UiAppearanceAbilityErrCode::INVALID_ARG);
242 return napi_invalid_arg;
243 }
244 break;
245 default:
246 return napi_invalid_arg;
247 }
248 return napi_ok;
249 }
250
ConvertJsDarkMode2Enum(int32_t jsVal)251 DarkMode JsUiAppearance::ConvertJsDarkMode2Enum(int32_t jsVal)
252 {
253 switch (jsVal) {
254 case 0:
255 return DarkMode::ALWAYS_DARK;
256 case 1:
257 return DarkMode::ALWAYS_LIGHT;
258 default:
259 return DarkMode::UNKNOWN;
260 }
261 }
262
JSSetDarkMode(napi_env env,napi_callback_info info)263 static napi_value JSSetDarkMode(napi_env env, napi_callback_info info)
264 {
265 LOGI("JSSetDarkMode begin.");
266
267 size_t argc = ARGC_WITH_TWO;
268 napi_value argv[ARGC_WITH_TWO] = { 0 };
269 napi_status napiStatus = napi_ok;
270 napi_value result = nullptr;
271 napi_get_undefined(env, &result);
272
273 napiStatus = napi_get_cb_info(env, info, &argc, argv, nullptr, nullptr);
274 if (napiStatus != napi_ok) {
275 NapiThrow(env, "get callback info failed.", UiAppearanceAbilityErrCode::INVALID_ARG);
276 return result;
277 }
278 napiStatus = JsUiAppearance::CheckArgs(env, argc, argv);
279 if (napiStatus != napi_ok) {
280 NapiThrow(env, "parameter parsing error.", UiAppearanceAbilityErrCode::INVALID_ARG);
281 return result;
282 }
283
284 auto asyncContext = new (std::nothrow) AsyncContext();
285 if (asyncContext == nullptr) {
286 NapiThrow(env, "create AsyncContext failed.", UiAppearanceAbilityErrCode::SYS_ERR);
287 return result;
288 }
289
290 napi_get_value_int32(env, argv[0], &asyncContext->jsSetArg);
291 asyncContext->mode = JsUiAppearance::ConvertJsDarkMode2Enum(asyncContext->jsSetArg);
292 if (argc == ARGC_WITH_TWO) {
293 napi_create_reference(env, argv[1], 1, &asyncContext->callbackRef);
294 }
295 if (asyncContext->callbackRef == nullptr) {
296 napi_create_promise(env, &asyncContext->deferred, &result);
297 }
298
299 napi_value resource = nullptr;
300 napi_create_string_utf8(env, "JSSetDarkMode", NAPI_AUTO_LENGTH, &resource);
301 napi_create_async_work(env, nullptr, resource, JsUiAppearance::OnExecute, JsUiAppearance::OnComplete,
302 reinterpret_cast<void*>(asyncContext), &asyncContext->work);
303 napi_queue_async_work(env, asyncContext->work);
304
305 return result;
306 }
307
JSGetDarkMode(napi_env env,napi_callback_info info)308 static napi_value JSGetDarkMode(napi_env env, napi_callback_info info)
309 {
310 LOGI("JSGetDarkMode begin.");
311
312 napi_value result = nullptr;
313 napi_get_undefined(env, &result);
314 size_t argc = 0;
315 NAPI_CALL(env, napi_get_cb_info(env, info, &argc, nullptr, nullptr, nullptr));
316 if (argc != 0) {
317 NapiThrow(env, "requires no parameter.", UiAppearanceAbilityErrCode::INVALID_ARG);
318 return result;
319 }
320
321 auto mode = UiAppearanceAbilityClient::GetInstance()->GetDarkMode();
322 if (mode == UiAppearanceAbilityErrCode::SYS_ERR) {
323 NapiThrow(env, "get dark-mode failed.", UiAppearanceAbilityErrCode::SYS_ERR);
324 return result;
325 }
326 if (mode == UiAppearanceAbilityErrCode::PERMISSION_ERR) {
327 NapiThrow(env,
328 "An attempt was made to get configuration forbidden by permission: ohos.permission.UPDATE_CONFIGURATION.",
329 UiAppearanceAbilityErrCode::PERMISSION_ERR);
330 return result;
331 }
332 NAPI_CALL(env, napi_create_int32(env, mode, &result));
333 return result;
334 }
335
JSGetFontScale(napi_env env,napi_callback_info info)336 static napi_value JSGetFontScale(napi_env env, napi_callback_info info)
337 {
338 LOGI("JSGetFontScale begin.");
339 napi_value result = nullptr;
340 napi_get_undefined(env, &result);
341 size_t argc = 0;
342 NAPI_CALL(env, napi_get_cb_info(env, info, &argc, nullptr, nullptr, nullptr));
343 if (argc != 0) {
344 NapiThrow(env, "requires no parameter.", UiAppearanceAbilityErrCode::INVALID_ARG);
345 return result;
346 }
347
348 std::string fontScale;
349 auto ret = UiAppearanceAbilityClient::GetInstance()->GetFontScale(fontScale);
350 if (ret == UiAppearanceAbilityErrCode::SYS_ERR) {
351 NapiThrow(env, "get font-scale failed.", UiAppearanceAbilityErrCode::SYS_ERR);
352 return result;
353 }
354 if (ret == UiAppearanceAbilityErrCode::PERMISSION_ERR) {
355 NapiThrow(env,
356 "An attempt was made to get configuration forbidden by permission: ohos.permission.UPDATE_CONFIGURATION.",
357 UiAppearanceAbilityErrCode::PERMISSION_ERR);
358 return result;
359 }
360 double fontScaleNumber = std::stod(fontScale);
361 NAPI_CALL(env, napi_create_double(env, fontScaleNumber, &result));
362 return result;
363 }
364
JSSetFontScale(napi_env env,napi_callback_info info)365 static napi_value JSSetFontScale(napi_env env, napi_callback_info info)
366 {
367 LOGI("JSSetFontScale begin.");
368
369 size_t argc = ARGC_WITH_TWO;
370 napi_value argv[ARGC_WITH_TWO] = { 0 };
371 napi_status napiStatus = napi_ok;
372 napi_value result = nullptr;
373 napi_get_undefined(env, &result);
374
375 napiStatus = napi_get_cb_info(env, info, &argc, argv, nullptr, nullptr);
376 if (napiStatus != napi_ok) {
377 NapiThrow(env, "get callback info failed.", UiAppearanceAbilityErrCode::INVALID_ARG);
378 return result;
379 }
380 napiStatus = JsUiAppearance::CheckFontScaleArgs(env, argc, argv);
381 if (napiStatus != napi_ok) {
382 NapiThrow(env, "parameter parsing error.", UiAppearanceAbilityErrCode::INVALID_ARG);
383 return result;
384 }
385 auto asyncContext = new (std::nothrow) AsyncContext();
386 if (asyncContext == nullptr) {
387 NapiThrow(env, "create AsyncContext failed.", UiAppearanceAbilityErrCode::SYS_ERR);
388 return result;
389 }
390 napi_get_value_double(env, argv[0], &asyncContext->jsFontScale);
391
392 asyncContext->fontScale = std::to_string(asyncContext->jsFontScale);
393 if (argc == ARGC_WITH_TWO) {
394 napi_create_reference(env, argv[1], 1, &asyncContext->callbackRef);
395 }
396 if (asyncContext->callbackRef == nullptr) {
397 napi_create_promise(env, &asyncContext->deferred, &result);
398 }
399
400 napi_value resource = nullptr;
401 napi_create_string_utf8(env, "JSSetFontScale", NAPI_AUTO_LENGTH, &resource);
402 napi_create_async_work(env, nullptr, resource, JsUiAppearance::OnSetFontScale, JsUiAppearance::OnComplete,
403 reinterpret_cast<void*>(asyncContext), &asyncContext->work);
404 napi_queue_async_work(env, asyncContext->work);
405
406 return result;
407 }
408
JSGetFontWeightScale(napi_env env,napi_callback_info info)409 static napi_value JSGetFontWeightScale(napi_env env, napi_callback_info info)
410 {
411 LOGI("JSGetFontWeightScale begin.");
412 napi_value result = nullptr;
413 napi_get_undefined(env, &result);
414 size_t argc = 0;
415 NAPI_CALL(env, napi_get_cb_info(env, info, &argc, nullptr, nullptr, nullptr));
416 if (argc != 0) {
417 NapiThrow(env, "requires no parameter.", UiAppearanceAbilityErrCode::INVALID_ARG);
418 return result;
419 }
420
421 std::string fontWeightScale;
422 auto ret = UiAppearanceAbilityClient::GetInstance()->GetFontWeightScale(fontWeightScale);
423 if (ret == UiAppearanceAbilityErrCode::SYS_ERR) {
424 NapiThrow(env, "get font-Weight-scale failed.", UiAppearanceAbilityErrCode::SYS_ERR);
425 return result;
426 }
427 if (ret == UiAppearanceAbilityErrCode::PERMISSION_ERR) {
428 NapiThrow(env,
429 "An attempt was made to get configuration forbidden by permission: ohos.permission.UPDATE_CONFIGURATION.",
430 UiAppearanceAbilityErrCode::PERMISSION_ERR);
431 return result;
432 }
433 double fontWeightScaleNumber = std::stod(fontWeightScale);
434 NAPI_CALL(env, napi_create_double(env, fontWeightScaleNumber, &result));
435 return result;
436 }
437
JSSetFontWeightScale(napi_env env,napi_callback_info info)438 static napi_value JSSetFontWeightScale(napi_env env, napi_callback_info info)
439 {
440 LOGI("JSSetFontWeightScale begin.");
441
442 size_t argc = ARGC_WITH_TWO;
443 napi_value argv[ARGC_WITH_TWO] = { 0 };
444 napi_status napiStatus = napi_ok;
445 napi_value result = nullptr;
446 napi_get_undefined(env, &result);
447
448 napiStatus = napi_get_cb_info(env, info, &argc, argv, nullptr, nullptr);
449 if (napiStatus != napi_ok) {
450 NapiThrow(env, "get callback info failed.", UiAppearanceAbilityErrCode::INVALID_ARG);
451 return result;
452 }
453 napiStatus = JsUiAppearance::CheckFontScaleArgs(env, argc, argv);
454 if (napiStatus != napi_ok) {
455 NapiThrow(env, "parameter parsing error.", UiAppearanceAbilityErrCode::INVALID_ARG);
456 return result;
457 }
458 auto asyncContext = new (std::nothrow) AsyncContext();
459 if (asyncContext == nullptr) {
460 NapiThrow(env, "create AsyncContext failed.", UiAppearanceAbilityErrCode::SYS_ERR);
461 return result;
462 }
463 napi_get_value_double(env, argv[0], &asyncContext->jsFontWeightScale);
464
465 asyncContext->fontWeightScale = std::to_string(asyncContext->jsFontWeightScale);
466 if (argc == ARGC_WITH_TWO) {
467 napi_create_reference(env, argv[1], 1, &asyncContext->callbackRef);
468 }
469 if (asyncContext->callbackRef == nullptr) {
470 napi_create_promise(env, &asyncContext->deferred, &result);
471 }
472
473 napi_value resource = nullptr;
474 napi_create_string_utf8(env, "JSSetFontWeightScale", NAPI_AUTO_LENGTH, &resource);
475 napi_create_async_work(env, nullptr, resource, JsUiAppearance::OnSetFontWeightScale, JsUiAppearance::OnComplete,
476 reinterpret_cast<void*>(asyncContext), &asyncContext->work);
477 napi_queue_async_work(env, asyncContext->work);
478
479 return result;
480 }
481
482 EXTERN_C_START
UiAppearanceExports(napi_env env,napi_value exports)483 static napi_value UiAppearanceExports(napi_env env, napi_value exports)
484 {
485 napi_value DarkMode = nullptr;
486 napi_value alwaysDark = nullptr;
487 napi_value alwaysLight = nullptr;
488 NAPI_CALL(env, napi_create_int32(env, 0, &alwaysDark));
489 NAPI_CALL(env, napi_create_int32(env, 1, &alwaysLight));
490 NAPI_CALL(env, napi_create_object(env, &DarkMode));
491 NAPI_CALL(env, napi_set_named_property(env, DarkMode, "ALWAYS_DARK", alwaysDark));
492 NAPI_CALL(env, napi_set_named_property(env, DarkMode, "ALWAYS_LIGHT", alwaysLight));
493 napi_property_descriptor properties[] = {
494 DECLARE_NAPI_FUNCTION("setDarkMode", JSSetDarkMode),
495 DECLARE_NAPI_FUNCTION("getDarkMode", JSGetDarkMode),
496 DECLARE_NAPI_FUNCTION("getFontScale", JSGetFontScale),
497 DECLARE_NAPI_FUNCTION("setFontScale", JSSetFontScale),
498 DECLARE_NAPI_FUNCTION("getFontWeightScale", JSGetFontWeightScale),
499 DECLARE_NAPI_FUNCTION("setFontWeightScale", JSSetFontWeightScale),
500 DECLARE_NAPI_STATIC_PROPERTY("DarkMode", DarkMode),
501 };
502 NAPI_CALL(env, napi_define_properties(env, exports, sizeof(properties) / sizeof(properties[0]), properties));
503 return exports;
504 }
505 EXTERN_C_END
506
507 static napi_module ui_appearance_module = {
508 .nm_version = 1,
509 .nm_flags = 0,
510 .nm_filename = nullptr,
511 .nm_register_func = UiAppearanceExports,
512 .nm_modname = "uiAppearance", // relative to the module name while import.
513 .nm_priv = nullptr,
514 .reserved = { 0 },
515 };
516
UiAppearanceRegister()517 extern "C" __attribute__((constructor)) void UiAppearanceRegister()
518 {
519 napi_module_register(&ui_appearance_module);
520 }
521 } // namespace ArkUi::UiAppearance
522 } // namespace OHOS
523