menu_account.dart 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  1. import 'package:app_settings/app_settings.dart';
  2. import 'package:auto_route/auto_route.dart';
  3. import 'package:easy_localization/easy_localization.dart';
  4. import 'package:flutter/material.dart';
  5. import 'package:page_transition/page_transition.dart';
  6. import 'package:provider/provider.dart';
  7. import 'package:qr_flutter/qr_flutter.dart';
  8. import 'package:telnow_mobile_new/src/api/api_auth_provider.dart';
  9. import 'package:telnow_mobile_new/src/layouts/functions/account.dart';
  10. import 'package:telnow_mobile_new/src/layouts/mobile/password.dart';
  11. import 'package:telnow_mobile_new/src/layouts/components/template.dart';
  12. import 'package:telnow_mobile_new/src/utils/U.dart';
  13. import 'package:telnow_mobile_new/src/utils/provider.dart';
  14. import 'package:permission_handler/permission_handler.dart';
  15. class MobAccountPage extends StatefulWidget {
  16. const MobAccountPage({super.key});
  17. @override
  18. State<MobAccountPage> createState() => _MobAccountPageState();
  19. }
  20. class _MobAccountPageState extends State<MobAccountPage> {
  21. final AccountFunction accFunc = AccountFunction();
  22. bool isDeniedNotifPermission = false;
  23. bool serDis = false;
  24. bool dnd = false;
  25. bool autoTranslate = false;
  26. List lang = [];
  27. @override
  28. void initState() {
  29. Provider.of<UserModule>(context, listen: false).reset();
  30. accFunc.getUser(context);
  31. setToggle();
  32. checkPermission(0);
  33. // TODO: implement initState
  34. super.initState();
  35. }
  36. checkPermission(seconds){
  37. Future.delayed(Duration(seconds: seconds)).then((value)async {
  38. bool denied = await Permission.notification.isDenied;
  39. if(!U.getNotifPermission() && !denied) U.setNotifPermissionDismissed(false);
  40. U.setNotifPermission(!denied);
  41. setState(() => isDeniedNotifPermission = denied);
  42. });
  43. }
  44. setToggle() async{
  45. var license = await U.getLicense();
  46. setState(() {
  47. dnd = Provider.of<UserModule>(context, listen: false).dndStatus();
  48. serDis = U.servantDisplay();
  49. autoTranslate = U.autoTranslate();
  50. lang = license['languages'] != null ? license['languages'].split(',') : [];
  51. });
  52. }
  53. @override
  54. Widget build(BuildContext context) {
  55. var codeOflang = {
  56. "id": 'bahasa'.tr(),
  57. "en": 'english'.tr(),
  58. "ja": 'japanese'.tr(),
  59. "zh": 'chinese'.tr(),
  60. "ko": 'korean'.tr(),
  61. };
  62. return Provider.of<UserModule>(context).user().isNotEmpty ? Scaffold(
  63. backgroundColor: backgroundColor,
  64. appBar: AppBar(
  65. elevation: 0,
  66. bottomOpacity: 0,
  67. backgroundColor: backgroundColor,
  68. titleSpacing: 16,
  69. title: Text('account'.tr(), style: TextStyle(color: textColor, fontSize: 17, fontWeight: FontWeight.w500), overflow: TextOverflow.ellipsis),
  70. ),
  71. body: Column(
  72. children: [
  73. divider(),
  74. Expanded(
  75. child: Container(
  76. alignment: Alignment.topCenter,
  77. width: U.bodyWidth(context),
  78. child: SingleChildScrollView(
  79. child: Column(
  80. crossAxisAlignment: CrossAxisAlignment.start,
  81. children: [
  82. Container(
  83. padding: EdgeInsets.all(16),
  84. child: Row(
  85. children: [
  86. CircleAvatar(
  87. child: Text(Provider.of<UserModule>(context).user()['name'][0].toString().toUpperCase(), style: TextStyle(color: Colors.white)),
  88. backgroundColor: primaryColor,
  89. radius: 25,
  90. ),
  91. SizedBox(width: 16),
  92. Expanded(child: Text(Provider.of<UserModule>(context).user()['name'], style: TextStyle(fontSize: 16, fontWeight: FontWeight.w500, color: textColor), maxLines: 2, overflow: TextOverflow.ellipsis))
  93. ],
  94. ),
  95. ),
  96. separator(),
  97. Container(
  98. padding: EdgeInsets.all(16),
  99. child: Column(
  100. crossAxisAlignment: CrossAxisAlignment.start,
  101. children: [
  102. Text('info_label'.tr(), style: TextStyle(color: textColor, fontSize: 16, fontWeight: FontWeight.w500)),
  103. SizedBox(height: 20),
  104. textHorizontal('userId'.tr(), Provider.of<UserModule>(context).user()['relatedTo'] != null ? Provider.of<UserModule>(context).user()['relatedTo'] : Provider.of<UserModule>(context).user()['userId'], size: 16, opacity: 0.75),
  105. SizedBox(height: 12),
  106. textHorizontal('location'.tr(), Provider.of<UserModule>(context).user()['location'] != null ? Provider.of<UserModule>(context).user()['location'] : '-', size: 16, opacity: 0.75),
  107. SizedBox(height: 12),
  108. textHorizontal('servantGroup'.tr(), Provider.of<UserModule>(context).user()['requestTypeListDescriptionMobile'] != null ? Provider.of<UserModule>(context).user()['requestTypeListDescriptionMobile'].toLowerCase() == 'semua' ? 'all'.tr() : Provider.of<UserModule>(context).user()['requestTypeListDescriptionMobile'] : '-', size: 16, opacity: 0.75),
  109. ],
  110. ),
  111. ),
  112. Provider.of<UserModule>(context).houseKeeping()?separator():Container(),
  113. Provider.of<UserModule>(context).houseKeeping()?Container(
  114. padding: EdgeInsets.all(16),
  115. child: Column(
  116. crossAxisAlignment: CrossAxisAlignment.start,
  117. children: [
  118. Text('doNotDisturb'.tr(), style: TextStyle(color: textColor, fontSize: 16, fontWeight: FontWeight.w500)),
  119. SizedBox(height: 20),
  120. Row(
  121. children: [
  122. Expanded(child: Text('set_dnd_status'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16))),
  123. Text(dnd?'active_dnd'.tr():'inactive_dnd'.tr(), style: TextStyle(color: textColor, fontSize: 14)),
  124. SizedBox(width: 5),
  125. GestureDetector(
  126. child: Container(
  127. decoration: BoxDecoration(border: Border.all(color: Provider.of<UserModule>(context).user()['checkedIn']?primaryColor:Colors.black38, width: 1.5), borderRadius: BorderRadius.all(Radius.circular(50))),
  128. child: Row(
  129. children: [
  130. !dnd?Container(
  131. width: 20, height: 20, decoration: BoxDecoration(color: Provider.of<UserModule>(context).user()['checkedIn']?primaryColor:Colors.black38, borderRadius: BorderRadius.all(Radius.circular(20))),
  132. ):Container(width: 20, height: 20),
  133. dnd?Container(
  134. width: 20, height: 20, decoration: BoxDecoration(color: Provider.of<UserModule>(context).user()['checkedIn']?primaryColor:Colors.black38, borderRadius: BorderRadius.all(Radius.circular(20))),
  135. ):Container(width: 20, height: 20),
  136. ],
  137. ),
  138. ),
  139. onTap: Provider.of<UserModule>(context).user()['checkedIn']?(){
  140. dialogConfirm(context: context, title: 'set_dnd_status'.tr(), text: dnd?'msg_change_inactive'.tr():'msg_change_active'.tr(), actionYes: ()async{
  141. bool res = await accFunc.setDndStatus(context, dnd);
  142. if(res){
  143. setState(() => dnd = !dnd);
  144. }
  145. });
  146. }:null,
  147. ),
  148. ],
  149. )
  150. ],
  151. ),
  152. ):Container(),
  153. !Provider.of<UserModule>(context).user()['_profile']['isRoom']?separator():Container(),
  154. !Provider.of<UserModule>(context).user()['_profile']['isRoom']?Container(
  155. padding: EdgeInsets.all(16),
  156. child: Column(
  157. crossAxisAlignment: CrossAxisAlignment.start,
  158. children: [
  159. Text('display_menu'.tr(), style: TextStyle(color: textColor, fontSize: 16, fontWeight: FontWeight.w500)),
  160. SizedBox(height: 20),
  161. Row(
  162. children: [
  163. Expanded(child: Text(serDis?'ser_group'.tr():'req_group'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16))),
  164. SizedBox(width: 5),
  165. GestureDetector(
  166. child: Container(
  167. decoration: BoxDecoration(border: Border.all(color: primaryColor, width: 1.5), borderRadius: BorderRadius.all(Radius.circular(50))),
  168. child: Row(
  169. children: [
  170. !serDis?Container(
  171. width: 20, height: 20, decoration: BoxDecoration(color: primaryColor, borderRadius: BorderRadius.all(Radius.circular(20))),
  172. ):Container(width: 20, height: 20),
  173. serDis?Container(
  174. width: 20, height: 20, decoration: BoxDecoration(color: primaryColor, borderRadius: BorderRadius.all(Radius.circular(20))),
  175. ):Container(width: 20, height: 20),
  176. ],
  177. ),
  178. ),
  179. onTap: (){
  180. setState(() {
  181. if(serDis){
  182. U.setServantDisplay(false);
  183. serDis = false;
  184. }
  185. else{
  186. U.setServantDisplay(true);
  187. serDis = true;
  188. }
  189. });
  190. },
  191. ),
  192. ],
  193. )
  194. ],
  195. ),
  196. ):Container(),
  197. separator(),
  198. Container(
  199. padding: EdgeInsets.fromLTRB(16, 16, 16, 0),
  200. child: Column(
  201. crossAxisAlignment: CrossAxisAlignment.start,
  202. children: [
  203. Text('setting'.tr(), style: TextStyle(color: textColor, fontSize: 16, fontWeight: FontWeight.w500)),
  204. SizedBox(height: 6),
  205. GestureDetector(
  206. child: Container(
  207. color: Colors.white,
  208. padding: EdgeInsets.symmetric(vertical: 16),
  209. child: Row(
  210. children: [
  211. U.iconsax('bold/global', color: textColor.withValues(alpha: 0.75)),
  212. SizedBox(width: 16),
  213. Expanded(child: Text('language'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16), overflow: TextOverflow.ellipsis)),
  214. Text(codeOflang[context.locale.toString()]!, style: TextStyle(color: textColor)),
  215. SizedBox(width: 12),
  216. U.iconsax('arrow-right-3', size: 16, color: textColor.withValues(alpha: 0.75)),
  217. ],
  218. ),
  219. ),
  220. onTap: ()=>changeLang(context, Provider.of<UserModule>(context, listen: false).user()),
  221. ),
  222. divider(),
  223. Container(
  224. color: Colors.white,
  225. padding: EdgeInsets.symmetric(vertical: 16),
  226. child: Row(
  227. children: [
  228. Icon(Icons.g_translate_rounded, color: textColor.withValues(alpha: 0.75)),
  229. SizedBox(width: 16),
  230. Expanded(child: Text('auto_translate'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16), overflow: TextOverflow.ellipsis)),
  231. Text(autoTranslate?'ON':'OFF', style: TextStyle(color: textColor)),
  232. SizedBox(width: 5),
  233. GestureDetector(
  234. child: Container(
  235. decoration: BoxDecoration(border: Border.all(color: autoTranslate?primaryColor:Colors.black38, width: 1.5), borderRadius: BorderRadius.all(Radius.circular(50))),
  236. child: Row(
  237. children: [
  238. !autoTranslate?Container(
  239. width: 20, height: 20, decoration: BoxDecoration(color: Colors.black38, borderRadius: BorderRadius.all(Radius.circular(20))),
  240. ):Container(width: 20, height: 20),
  241. autoTranslate?Container(
  242. width: 20, height: 20, decoration: BoxDecoration(color: primaryColor, borderRadius: BorderRadius.all(Radius.circular(20))),
  243. ):Container(width: 20, height: 20),
  244. ],
  245. ),
  246. ),
  247. onTap: (){
  248. setState(() {
  249. if(autoTranslate){
  250. U.setAutoTranslate(false);
  251. autoTranslate = false;
  252. }
  253. else{
  254. U.setAutoTranslate(true);
  255. autoTranslate = true;
  256. }
  257. });
  258. },
  259. ),
  260. ],
  261. ),
  262. ),
  263. divider(),
  264. GestureDetector(
  265. child: Container(
  266. color: Colors.white,
  267. padding: EdgeInsets.symmetric(vertical: 16),
  268. child: Row(
  269. children: [
  270. U.iconsax('bold/key', color: textColor.withValues(alpha: 0.75)),
  271. SizedBox(width: 16),
  272. Text('password'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16)),
  273. Expanded(child: Text('settingPasswordText'.tr(), style: TextStyle(color: textColor), overflow: TextOverflow.ellipsis, textAlign: TextAlign.end)),
  274. SizedBox(width: 12),
  275. U.iconsax('arrow-right-3', size: 16, color: textColor.withValues(alpha: 0.75)),
  276. ],
  277. ),
  278. ),
  279. onTap: ()=>Navigator.push(context, PageTransition(type: PageTransitionType.rightToLeft, child: MobPasswordPage(user: Provider.of<UserModule>(context, listen: false).user()))).then((value) {
  280. value??false;
  281. if (value) showSuccess(context, 'messagePassChanged'.tr());
  282. }),
  283. ),
  284. divider(),
  285. GestureDetector(
  286. child: Container(
  287. color: Colors.white,
  288. padding: EdgeInsets.symmetric(vertical: 16),
  289. child: Row(
  290. children: [
  291. U.iconsax('bold/logout', color: textColor.withValues(alpha: 0.75)),
  292. SizedBox(width: 16),
  293. Expanded(child: Text('logout'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16), overflow: TextOverflow.ellipsis)),
  294. U.iconsax('arrow-right-3', size: 16, color: textColor.withValues(alpha: 0.75)),
  295. ],
  296. ),
  297. ),
  298. onTap: (){
  299. showDialog(
  300. context: context,
  301. builder: (BuildContext context) {
  302. return AlertDialog(
  303. title: Text("logout".tr()),
  304. content: Text("textLogout".tr()),
  305. actions: <Widget>[
  306. TextButton(
  307. child: Text("buttonNo".tr()),
  308. onPressed: () {
  309. Navigator.of(context).pop();
  310. },
  311. ),
  312. TextButton(
  313. onPressed: () => accFunc.logoutAction(context),
  314. child: Text("buttonYes".tr())),
  315. ],
  316. );
  317. },
  318. );
  319. },
  320. ),
  321. ],
  322. ),
  323. ),
  324. separator(),
  325. GestureDetector(
  326. child: Container(
  327. color: Colors.white,
  328. margin: EdgeInsets.symmetric(horizontal: 16),
  329. padding: EdgeInsets.symmetric(vertical: 16),
  330. child: Row(
  331. children: [
  332. U.iconsax('bold/scan', color: textColor.withValues(alpha: 0.75)),
  333. SizedBox(width: 16),
  334. Expanded(child: Text('scan_qr'.tr(), style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 16), overflow: TextOverflow.ellipsis)),
  335. U.iconsax('arrow-right-3', size: 16, color: textColor.withValues(alpha: 0.75)),
  336. ],
  337. ),
  338. ),
  339. onTap: (){
  340. var size = MediaQuery.of(context).size.width - (MediaQuery.of(context).size.width/3);
  341. var pid = U.getPidFromUrl(context.router.currentUrl);
  342. var qr_data = 'https://telnow.telmessenger.com/#/app/${pid}%23${U.getBaseUrl()}';
  343. showDialog(
  344. context: context,
  345. builder: (BuildContext context) {
  346. return AlertDialog(
  347. title: Text("show_qr".tr(), style: TextStyle(fontSize: 16, color: Colors.black), textAlign: TextAlign.center),
  348. content: Column(
  349. mainAxisSize: MainAxisSize.min,
  350. crossAxisAlignment: CrossAxisAlignment.center,
  351. children: [
  352. SizedBox(
  353. width: size,
  354. height: size,
  355. child: QrImageView(
  356. data: qr_data,
  357. version: QrVersions.auto,
  358. gapless: true,
  359. ),
  360. ),
  361. SizedBox(height: 15),
  362. Text("show_qr_desc".tr(), style: TextStyle(fontSize: 14, color: Colors.black), textAlign: TextAlign.center)
  363. ],
  364. ),
  365. );
  366. },
  367. );
  368. },
  369. ),
  370. separator(),
  371. isDeniedNotifPermission ? Container(
  372. padding: EdgeInsets.all(16),
  373. child: GestureDetector(
  374. child: Column(
  375. crossAxisAlignment: CrossAxisAlignment.start,
  376. children: [
  377. Text("not_allow_permission".tr()),
  378. SizedBox(height: 8,),
  379. Text("tap_here".tr(), style: TextStyle(color: Color(0xFF198AF2)),),
  380. ],
  381. ),
  382. onTap: () => AppSettings.openAppSettings(type: AppSettingsType.notification).then((value) async {
  383. // print("after open setting");
  384. // checkPermission(2);
  385. }),
  386. ),
  387. ) : Container(),
  388. isDeniedNotifPermission ? Container() : separator(),
  389. Padding(
  390. padding: const EdgeInsets.symmetric(vertical: 24),
  391. child: Center(child: Column(
  392. children: [
  393. Text("${"version".tr()} ${ApiAuthProvider().displayVersion}"),
  394. Text("Build ${ApiAuthProvider().buildNumber}", style: TextStyle(fontSize: 13),),
  395. ],
  396. )),
  397. )
  398. ],
  399. ),
  400. ),
  401. ),
  402. )
  403. ],
  404. ),
  405. ) : Provider.of<UserModule>(context).resetData() ? RefreshPage(() {
  406. Provider.of<UserModule>(context, listen: false).setResetData(false);
  407. accFunc.getUser(context);
  408. }) : loadingTemplate();
  409. }
  410. changeLang(context, user) {
  411. showModalBottomSheet(
  412. context: context,
  413. backgroundColor: Colors.white,
  414. builder: (BuildContext context) {
  415. return Column(
  416. mainAxisSize: MainAxisSize.min,
  417. children: <Widget>[
  418. Padding(
  419. padding: EdgeInsets.symmetric(vertical: 16),
  420. child: Center(
  421. child: Text('chooseLanguage'.tr(), style: TextStyle(color: textColor)),
  422. ),
  423. ),
  424. divider(),
  425. SizedBox(height: 16),
  426. listOfLang(user),
  427. SizedBox(height: 16)
  428. ],
  429. );
  430. }
  431. );
  432. }
  433. listOfLang(user){
  434. return Column(
  435. crossAxisAlignment: CrossAxisAlignment.start,
  436. children: [
  437. ListTile(
  438. title: Text('bahasa'.tr(), style: TextStyle(color: textColor, fontSize: 14)),
  439. trailing: Icon(context.locale.toString()=='id'?Icons.radio_button_checked:Icons.radio_button_off, color: primaryColor, size: 18),
  440. onTap: context.locale.toString() != 'id' ? () async {
  441. accFunc.switchLang(context, 'id', user);
  442. } : null,
  443. ),
  444. ListTile(
  445. title: Text('english'.tr(), style: TextStyle(color: textColor, fontSize: 14)),
  446. trailing: Icon(context.locale.toString()=='en'?Icons.radio_button_checked:Icons.radio_button_off, color: primaryColor, size: 18),
  447. onTap: context.locale.toString() != 'en' ? () async {
  448. accFunc.switchLang(context, 'en', user);
  449. } : null,
  450. ),
  451. lang.contains('ja') ? ListTile(
  452. title: Text('japanese'.tr(), style: TextStyle(color: textColor, fontSize: 14)),
  453. trailing: Icon(context.locale.toString()=='ja'?Icons.radio_button_checked:Icons.radio_button_off, color: primaryColor, size: 18),
  454. onTap: context.locale.toString() != 'ja' ? () async {
  455. accFunc.switchLang(context, 'ja', user);
  456. } : null,
  457. ) : Container(),
  458. lang.contains('zh') ? ListTile(
  459. title: Text('chinese'.tr(), style: TextStyle(color: textColor, fontSize: 14)),
  460. trailing: Icon(context.locale.toString()=='zh'?Icons.radio_button_checked:Icons.radio_button_off, color: primaryColor, size: 18),
  461. onTap: context.locale.toString() != 'zh' ? () async {
  462. accFunc.switchLang(context, 'zh', user);
  463. } : null,
  464. ) : Container(),
  465. lang.contains('ko') ? ListTile(
  466. title: Text('korean'.tr(), style: TextStyle(color: textColor, fontSize: 14)),
  467. trailing: Icon(context.locale.toString()=='ko'?Icons.radio_button_checked:Icons.radio_button_off, color: primaryColor, size: 18),
  468. onTap: context.locale.toString() != 'ko' ? () async {
  469. accFunc.switchLang(context, 'ko', user);
  470. } : null,
  471. ) : Container()
  472. ],
  473. );
  474. }
  475. }