menu_history.dart 47 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823
  1. import 'package:auto_route/auto_route.dart';
  2. import 'package:easy_localization/easy_localization.dart';
  3. import 'package:easy_refresh/easy_refresh.dart';
  4. import 'package:flutter/material.dart';
  5. import 'package:lazy_load_scrollview/lazy_load_scrollview.dart';
  6. import 'package:provider/provider.dart';
  7. import 'package:telnow_mobile_new/src/api/api_auth_provider.dart';
  8. import 'package:telnow_mobile_new/src/layouts/functions/history.dart';
  9. import 'package:telnow_mobile_new/src/layouts/components/template.dart';
  10. import 'package:telnow_mobile_new/src/layouts/web/history_detail.dart';
  11. import 'package:telnow_mobile_new/src/utils/U.dart';
  12. import 'package:telnow_mobile_new/src/utils/provider.dart';
  13. class WebHistoryPage extends StatefulWidget {
  14. const WebHistoryPage({super.key});
  15. @override
  16. State<WebHistoryPage> createState() => _WebHistoryPageState();
  17. }
  18. class _WebHistoryPageState extends State<WebHistoryPage> with TickerProviderStateMixin {
  19. final HistoryFunction hisFunc = HistoryFunction();
  20. late AnimationController _animationController;
  21. var rating = [
  22. {'key': 1, 'image': "assets/image/icon/very_dissatisfied.png", 'label': 'disatisfied'.tr()},
  23. {'key': 2, 'image': "assets/image/icon/dissatisfied.png", 'label': 'lessSatisfied'.tr()},
  24. {'key': 3, 'image': "assets/image/icon/neutral.png", 'label': 'satisfied'.tr()},
  25. {'key': 4, 'image': "assets/image/icon/satisfied.png", 'label': 'verySatisfied'.tr()},
  26. {'key': 5, 'image': "assets/image/icon/very_satisfied.png", 'label': 'reallyPleased'.tr()},
  27. ];
  28. @override
  29. void initState() {
  30. // print(U.newServerVersion(1716279633));
  31. Provider.of<HistoryModule>(context, listen: false).reset();
  32. _animationController = new AnimationController(vsync: this, duration: Duration(seconds: 1));
  33. _animationController.repeat(reverse: true);
  34. WidgetsBinding.instance.addPostFrameCallback((_) {
  35. hisFunc.getActiveForum(context);
  36. });
  37. // TODO: implement initState
  38. super.initState();
  39. }
  40. @override
  41. void dispose() {
  42. _animationController.dispose();
  43. super.dispose();
  44. }
  45. @override
  46. Widget build(BuildContext context) {
  47. return Provider.of<UserModule>(context).resetData() ? RefreshPage(() {
  48. Provider.of<UserModule>(context, listen: false).setResetData(false);
  49. hisFunc.getActiveForum(context);
  50. }) : Scaffold(
  51. backgroundColor: backgroundColor,
  52. appBar: PreferredSize(preferredSize: Size.fromHeight(0), child: AppBar(elevation: 0, backgroundColor: primaryColor)),
  53. body: Column(
  54. children: [
  55. Container(
  56. padding: EdgeInsets.symmetric(vertical: 25, horizontal: 100),
  57. child: Row(
  58. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  59. children: [
  60. Text('history'.tr(), style: TextStyle(color: textColor, fontSize: 17, fontWeight: FontWeight.w500), overflow: TextOverflow.ellipsis),
  61. Provider.of<HistoryModule>(context).multiSelectMode() ? GestureDetector(
  62. child: Container(
  63. padding: EdgeInsets.fromLTRB(16, 9, 16, 9),
  64. child: Row(
  65. children: [
  66. Text('delete'.tr(), style: TextStyle(color: primaryColor, fontSize: 14)),
  67. SizedBox(width: 5),
  68. U.iconsax('trash', color: primaryColor, size: 20)
  69. ],
  70. ),
  71. decoration: BoxDecoration(color: primaryColor.withValues(alpha: 0.1), border: Border.all(color: primaryColor), borderRadius: BorderRadius.all(Radius.circular(50))),
  72. ),
  73. onTap: () => hisFunc.deleteData(context),
  74. ) : GestureDetector(
  75. child: Text('buttonBack'.tr(), style: TextStyle(color: primaryColor, fontSize: 14)),
  76. onTap: (){
  77. var pid = U.getPidFromUrl(context.router.currentUrl);
  78. context.router.removeLast();
  79. context.navigateToPath("/app/$pid/menu");
  80. },
  81. )
  82. ],
  83. ),
  84. ),
  85. divider(),
  86. Container(
  87. padding: EdgeInsets.symmetric(vertical: 25, horizontal: 100),
  88. child: Column(
  89. crossAxisAlignment: CrossAxisAlignment.start,
  90. children: [
  91. Row(
  92. children: [
  93. GestureDetector(
  94. child: Container(
  95. padding: EdgeInsets.symmetric(vertical: 5, horizontal: 16),
  96. child: Text('ongoing'.tr(), style: TextStyle(color: Provider.of<HistoryModule>(context).selectedFilter() == 1 ? textColor : textColor.withValues(alpha: 0.65), fontSize: 16)),
  97. decoration: BoxDecoration(color: Colors.white, border: Border(bottom: BorderSide(color: Provider.of<HistoryModule>(context).selectedFilter() == 1 ? primaryColor : primaryColor.withValues(alpha: 0.3), width: Provider.of<HistoryModule>(context).selectedFilter() == 1 ? 2 : 1))),
  98. ),
  99. onTap: () {
  100. if (!Provider.of<HistoryModule>(context, listen: false).isLoadHistory()) {
  101. Provider.of<HistoryModule>(context, listen: false).setPressAttention(0);
  102. Provider.of<HistoryModule>(context, listen: false).setSelectedFilter(1);
  103. hisFunc.onRefresh(context);
  104. }
  105. },
  106. ),
  107. GestureDetector(
  108. child: Container(
  109. padding: EdgeInsets.symmetric(vertical: 5, horizontal: 16),
  110. child: Text('done'.tr(), style: TextStyle(color: Provider.of<HistoryModule>(context).selectedFilter() == 0 ? textColor : textColor.withValues(alpha: 0.65), fontSize: 16)),
  111. decoration: BoxDecoration(color: Colors.white, border: Border(bottom: BorderSide(color: Provider.of<HistoryModule>(context).selectedFilter() == 0 ? primaryColor : primaryColor.withValues(alpha: 0.3), width: Provider.of<HistoryModule>(context).selectedFilter() == 0 ? 2 : 1))),
  112. ),
  113. onTap: () {
  114. if (!Provider.of<HistoryModule>(context, listen: false).isLoadHistory()) {
  115. Provider.of<HistoryModule>(context, listen: false).setPressAttention(0);
  116. Provider.of<HistoryModule>(context, listen: false).setSelectedFilter(0);
  117. hisFunc.onRefresh(context);
  118. }
  119. },
  120. ),
  121. ],
  122. ),
  123. SizedBox(height: 20),
  124. Provider.of<HistoryModule>(context).selectedFilter() == 0 ? doneFilter() : ongoingFilter(),
  125. ],
  126. ),
  127. ),
  128. Expanded(
  129. child: Provider.of<UserModule>(context).user().isNotEmpty ? Container(
  130. width: double.infinity,
  131. padding: EdgeInsets.symmetric(horizontal: 100),
  132. child: LazyLoadScrollView(
  133. onEndOfPage: () => hisFunc.getMission(context),
  134. scrollOffset: 500,
  135. child: EasyRefresh(
  136. header: MaterialHeader(clamping: true, color: primaryColor),
  137. onRefresh: ()=>hisFunc.onRefresh(context),
  138. child: Provider.of<HistoryModule>(context).selectedFilter() == 0 ? doneContainer() : ongoingContainer(),
  139. ),
  140. ),
  141. ) : Provider.of<UserModule>(context).resetData() ? Container() : loadingTemplate(),
  142. )
  143. ],
  144. ),
  145. );
  146. }
  147. Widget ongoingFilter() {
  148. return SingleChildScrollView(
  149. scrollDirection: Axis.horizontal,
  150. child: Row(
  151. children: List.generate(hisFunc.ongoingList.length, (i) {
  152. return GestureDetector(
  153. child: Container(
  154. margin: EdgeInsets.only(right: 16),
  155. padding: EdgeInsets.symmetric(vertical: 8, horizontal: 16),
  156. child: Text(hisFunc.ongoingList[i]['title'], style: TextStyle(color: Provider.of<HistoryModule>(context).checkPressAttention(hisFunc.ongoingList[i]['value']) ? Colors.white : textColor, fontSize: 16)),
  157. decoration: BoxDecoration(
  158. color: Provider.of<HistoryModule>(context).checkPressAttention(hisFunc.ongoingList[i]['value']) ? primaryColor : Color(0xffF2F8F7),
  159. border: Border.all(color: Provider.of<HistoryModule>(context).checkPressAttention(hisFunc.ongoingList[i]['value']) ? primaryColor : Color(0xffBEC1C1)),
  160. borderRadius: BorderRadius.all(Radius.circular(50))),
  161. ),
  162. onTap: () {
  163. if (!Provider.of<HistoryModule>(context, listen: false).isLoadHistory()) {
  164. if (Provider.of<HistoryModule>(context, listen: false).checkPressAttention(hisFunc.ongoingList[i]['value'])) {
  165. Provider.of<HistoryModule>(context, listen: false).setPressAttention(0);
  166. hisFunc.onRefresh(context);
  167. } else {
  168. Provider.of<HistoryModule>(context, listen: false).setPressAttention(hisFunc.ongoingList[i]['value']);
  169. hisFunc.onRefresh(context);
  170. }
  171. }
  172. },
  173. );
  174. }),
  175. ),
  176. );
  177. }
  178. Widget ongoingContainer() {
  179. return SingleChildScrollView(
  180. child: Column(
  181. children: [
  182. U.newServerVersion(1709864293) && !U.getInternetStatus() ? Column(
  183. children: List.generate(Provider.of<HistoryModule>(context).dataPending().length, (i) {
  184. return GestureDetector(
  185. child: Container(
  186. width: double.infinity,
  187. padding: EdgeInsets.all(15),
  188. margin: EdgeInsets.only(bottom: 15),
  189. decoration: BoxDecoration(color: Colors.white, border: Border.all(color: textColor.withValues(alpha: 0.15)), borderRadius: BorderRadius.all(Radius.circular(12))),
  190. child: Column(
  191. children: [
  192. Row(
  193. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  194. children: [
  195. Text(convertDate(Provider.of<HistoryModule>(context).dataPending()[i]['datetimeRequest'], context.locale.toString()), style: TextStyle(color: primaryColor, fontSize: 14)),
  196. renderStatus('sending'.tr(), Color(0xff02C539).withValues(alpha: 0.2))
  197. ],
  198. ),
  199. SizedBox(height: 10),
  200. divider(),
  201. SizedBox(height: 10),
  202. Row(
  203. children: [
  204. imageTiles(imageUrl: Provider.of<HistoryModule>(context).dataPending()[i]['image'] ?? "null"),
  205. SizedBox(width: 25),
  206. Expanded(
  207. child: Column(
  208. crossAxisAlignment: CrossAxisAlignment.start,
  209. children: [
  210. Text(Provider.of<HistoryModule>(context).dataPending()[i]['title'], style: TextStyle(color: textColor, fontWeight: FontWeight.w500), overflow: TextOverflow.ellipsis),
  211. SizedBox(height: 10),
  212. Text(Provider.of<HistoryModule>(context).dataPending()[i][U.langColumn(context, 'sub')], style: TextStyle(color: textColor), overflow: TextOverflow.ellipsis),
  213. SizedBox(height: 5),
  214. Text(Provider.of<HistoryModule>(context).dataPending()[i][U.langColumn(context, 'desc')], style: TextStyle(color: textColor), overflow: TextOverflow.ellipsis),
  215. dashed(top: 10, bottom: 10),
  216. Row(
  217. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  218. children: [
  219. Text('location'.tr()+': '+Provider.of<HistoryModule>(context).dataPending()[i]['location'], style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  220. Row(
  221. children: [
  222. Container(
  223. width: 25, height: 25,
  224. decoration: BoxDecoration(image: DecorationImage(image: AssetImage('assets/image/general/Hourglass.png'), fit: BoxFit.cover)),
  225. ),
  226. SizedBox(width: 8),
  227. Text('pending_info_tiles'.tr(), style: TextStyle(color: textColor, fontSize: 14))
  228. ],
  229. )
  230. ],
  231. )
  232. ],
  233. ),
  234. )
  235. ],
  236. )
  237. ],
  238. ),
  239. ),
  240. onTap: () {
  241. if(!U.getInternetStatus()){
  242. // navigateTo(context, MobHistoryDetailPendingPage(data: Provider.of<HistoryModule>(context, listen: false).dataPending()[i])).then((val) {
  243. // val = val??false;
  244. // if(val) hisFunc.onRefresh(context);
  245. // });
  246. }
  247. else{
  248. hisFunc.onRefresh(context);
  249. }
  250. },
  251. );
  252. }),
  253. ) : Container(),
  254. Column(
  255. children: List.generate(Provider.of<HistoryModule>(context).dataMisi().length, (i) {
  256. return GestureDetector(
  257. child: Container(
  258. width: double.infinity,
  259. padding: EdgeInsets.all(15),
  260. margin: EdgeInsets.only(bottom: 15),
  261. decoration: BoxDecoration(color: Colors.white, border: Border.all(color: textColor.withValues(alpha: 0.15)), borderRadius: BorderRadius.all(Radius.circular(12))),
  262. child: Column(
  263. children: [
  264. Row(
  265. children: [
  266. Text(convertDate(Provider.of<HistoryModule>(context).dataMisi()[i]['datetimeRequest'], context.locale.toString()), style: TextStyle(color: primaryColor, fontSize: 14)),
  267. SizedBox(width: 15),
  268. Expanded(
  269. child: Text('ticketNumber'.tr()+': '+Provider.of<HistoryModule>(context).dataMisi()[i]['ticketNo'], style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  270. ),
  271. Provider.of<HistoryModule>(context).dataMisi()[i]['currentState'] == 'DIMULAI' ? Container(
  272. child: Row(
  273. children: [
  274. Container(
  275. width: 25, height: 25,
  276. decoration: BoxDecoration(
  277. border: Border.all(color: primaryColor), borderRadius: BorderRadius.all(Radius.circular(25)),
  278. image: DecorationImage(image: AssetImage('assets/image/general/Avatar.jpg'), fit: BoxFit.cover)
  279. ),
  280. ),
  281. SizedBox(width: 8),
  282. Text(Provider.of<HistoryModule>(context).dataMisi()[i]['servantNameStart'] + 'isWorking'.tr(), style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis)
  283. ],
  284. ),
  285. ) : Container(),
  286. Provider.of<HistoryModule>(context).dataMisi()[i]['_activeHoldRequest'] != null?renderStatus('hold'.tr(), Color(0xffD3D3D3)):Provider.of<HistoryModule>(context).dataMisi()[i]['currentState'] == 'DIMULAI'?renderStatus('onProgress'.tr(), Color(0xffCCA600).withValues(alpha: 0.2)):renderStatus('queued'.tr(), Color(0xff02C539).withValues(alpha: 0.2))
  287. ],
  288. ),
  289. SizedBox(height: 10),
  290. divider(),
  291. SizedBox(height: 10),
  292. Row(
  293. children: [
  294. imageTiles(imageUrl: Provider.of<HistoryModule>(context).dataMisi()[i]['_requestImage'] ?? "null", width: 150, height: 120),
  295. SizedBox(width: 25),
  296. Expanded(
  297. child: Column(
  298. crossAxisAlignment: CrossAxisAlignment.start,
  299. children: [
  300. Text(Provider.of<HistoryModule>(context).dataMisi()[i][U.langColumn(context, 'requestGroupDescription')], style: TextStyle(color: textColor, fontSize: 16), overflow: TextOverflow.ellipsis),
  301. SizedBox(height: 10),
  302. Text(Provider.of<HistoryModule>(context).dataMisi()[i][U.langColumn(context, 'requestSubject')], style: TextStyle(color: textColor, fontWeight: FontWeight.w600), overflow: TextOverflow.ellipsis),
  303. SizedBox(height: 5),
  304. Text(Provider.of<HistoryModule>(context).dataMisi()[i][U.langColumn(context, '_subjectDescription')]??'', style: TextStyle(color: textColor), overflow: TextOverflow.ellipsis),
  305. dashed(top: 10, bottom: 10),
  306. renderRequested(Provider.of<HistoryModule>(context).dataMisi()[i]),
  307. Row(
  308. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  309. children: [
  310. Text('location'.tr()+': '+Provider.of<HistoryModule>(context).dataMisi()[i]['ipphoneExtLocation'], style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  311. Provider.of<HistoryModule>(context).dataMisi()[i]['_activeHoldRequest'] != null ? Container(
  312. child: Row(
  313. children: [
  314. Container(
  315. width: 25, height: 25,
  316. decoration: BoxDecoration(image: DecorationImage(image: AssetImage('assets/image/general/Watch.png'), fit: BoxFit.cover)),
  317. ),
  318. SizedBox(width: 8),
  319. Text(Provider.of<HistoryModule>(context).dataMisi()[i]['_activeHoldRequest']['description'], style: TextStyle(color: textColor, fontSize: 14))
  320. ],
  321. ),
  322. ) : !Provider.of<HistoryModule>(context).dataMisi()[i]['autoResponse'] && Provider.of<HistoryModule>(context).dataMisi()[i]['_hasForum'] && Provider.of<HistoryModule>(context).dataMisi()[i]['_forumMsg'] != null ? Container(
  323. width: double.infinity,
  324. margin: EdgeInsets.only(top: 16),
  325. child: Row(
  326. children: [
  327. FadeTransition(opacity: _animationController, child: U.iconsax('messages-3', color: primaryColor)),
  328. SizedBox(width: 8),
  329. Text('“${Provider.of<HistoryModule>(context).dataMisi()[i]['_forumMsg']}”', style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 14, fontStyle: FontStyle.italic))
  330. ],
  331. ),
  332. ) : Container(),
  333. ],
  334. ),
  335. ],
  336. ),
  337. )
  338. ],
  339. ),
  340. ],
  341. ),
  342. ),
  343. onTap: () => navigateTo(context, WebHistoryDetailPage(index: i)).then((val) {
  344. val = val??false;
  345. if(val) Provider.of<HistoryModule>(context, listen: false).cancelMisi(i);
  346. }),
  347. );
  348. }),
  349. ),
  350. !Provider.of<HistoryModule>(context).stopLoadHistory() || (Provider.of<HistoryModule>(context).isLoadHistory() && Provider.of<HistoryModule>(context).page() > 0)
  351. ? Container(margin: EdgeInsets.only(top: Provider.of<HistoryModule>(context).page() > 0 ? 0 : 50), child: loadingTemplate())
  352. : Provider.of<HistoryModule>(context).dataMisi().isEmpty && Provider.of<HistoryModule>(context).dataPending().isEmpty
  353. ? NoDataPage()
  354. : Container(),
  355. ],
  356. ),
  357. );
  358. }
  359. Widget doneFilter() {
  360. return Row(
  361. children: List.generate(hisFunc.doneList.length, (i) {
  362. return GestureDetector(
  363. child: Container(
  364. margin: EdgeInsets.only(right: 16),
  365. padding: EdgeInsets.symmetric(vertical: 8, horizontal: 16),
  366. decoration: BoxDecoration(
  367. color: Provider.of<HistoryModule>(context).checkPressAttention(hisFunc.doneList[i]['value']) ? primaryColor : Color(0xffF2F8F7),
  368. border: Border.all(color: Provider.of<HistoryModule>(context).checkPressAttention(hisFunc.doneList[i]['value']) ? primaryColor : Color(0xffBEC1C1)),
  369. borderRadius: BorderRadius.all(Radius.circular(50))),
  370. child: Text(hisFunc.doneList[i]['title'], style: TextStyle(color: Provider.of<HistoryModule>(context).checkPressAttention(hisFunc.doneList[i]['value']) ? Colors.white : textColor, fontSize: 16)),
  371. ),
  372. onTap: () {
  373. if (!Provider.of<HistoryModule>(context, listen: false).isLoadHistory()) {
  374. if (Provider.of<HistoryModule>(context, listen: false).checkPressAttention(hisFunc.doneList[i]['value'])) {
  375. Provider.of<HistoryModule>(context, listen: false).setPressAttention(0);
  376. hisFunc.onRefresh(context);
  377. } else {
  378. Provider.of<HistoryModule>(context, listen: false).setPressAttention(hisFunc.doneList[i]['value']);
  379. hisFunc.onRefresh(context);
  380. }
  381. }
  382. },
  383. );
  384. }),
  385. );
  386. }
  387. Widget doneContainer() {
  388. return SingleChildScrollView(
  389. child: Column(
  390. children: [
  391. Column(
  392. children: List.generate(Provider.of<HistoryModule>(context).dataMisi().length, (i) {
  393. return GestureDetector(
  394. child: Container(
  395. width: double.infinity,
  396. padding: EdgeInsets.all(15),
  397. margin: EdgeInsets.only(bottom: 15),
  398. decoration: BoxDecoration(color: Colors.white, border: Border.all(color: textColor.withValues(alpha: 0.15)), borderRadius: BorderRadius.all(Radius.circular(12))),
  399. child: Column(
  400. children: [
  401. Row(
  402. children: [
  403. Text(convertDate(Provider.of<HistoryModule>(context).dataMisi()[i]['datetimeRequest'], context.locale.toString()), style: TextStyle(color: primaryColor, fontSize: 14)),
  404. SizedBox(width: 15),
  405. Expanded(
  406. child: Text('ticketNumber'.tr()+': '+Provider.of<HistoryModule>(context).dataMisi()[i]['ticketNo'], style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  407. ),
  408. Provider.of<HistoryModule>(context).dataMisi()[i]['currentState'] == 'DISELESAIKAN' || Provider.of<HistoryModule>(context).dataMisi()[i]['currentState'] == 'TUNTAS' ? renderStatus('stateFinish'.tr(), primaryColor.withValues(alpha: 0.4)) : renderStatus('stateCancel'.tr(), Color(0xffD81010).withValues(alpha: 0.4))
  409. ],
  410. ),
  411. SizedBox(height: 10),
  412. divider(),
  413. SizedBox(height: 10),
  414. Row(
  415. children: [
  416. imageTiles(imageUrl: Provider.of<HistoryModule>(context).dataMisi()[i]['_requestImage'] ?? "null", width: 150, height: 120),
  417. SizedBox(width: 25),
  418. Expanded(
  419. child: Column(
  420. crossAxisAlignment: CrossAxisAlignment.start,
  421. children: [
  422. Text(Provider.of<HistoryModule>(context).dataMisi()[i][U.langColumn(context, 'requestGroupDescription')], style: TextStyle(color: textColor, fontSize: 16), overflow: TextOverflow.ellipsis),
  423. SizedBox(height: 10),
  424. Text(Provider.of<HistoryModule>(context).dataMisi()[i][U.langColumn(context, 'requestSubject')], style: TextStyle(color: textColor, fontWeight: FontWeight.w600), overflow: TextOverflow.ellipsis),
  425. SizedBox(height: 5),
  426. Text(Provider.of<HistoryModule>(context).dataMisi()[i][U.langColumn(context, '_subjectDescription')]??'', style: TextStyle(color: textColor), overflow: TextOverflow.ellipsis),
  427. dashed(top: 10, bottom: 10),
  428. renderRequested(Provider.of<HistoryModule>(context).dataMisi()[i]),
  429. Text('location'.tr()+': '+Provider.of<HistoryModule>(context).dataMisi()[i]['ipphoneExtLocation'], style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  430. ],
  431. ),
  432. )
  433. ],
  434. ),
  435. SizedBox(height: 10),
  436. divider(),
  437. SizedBox(height: 10),
  438. Row(
  439. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  440. children: [
  441. !Provider.of<HistoryModule>(context).dataMisi()[i]['autoResponse'] && Provider.of<HistoryModule>(context).dataMisi()[i]['servantIdentityStart'] != '#system' && (Provider.of<HistoryModule>(context).dataMisi()[i]['currentState'] == 'TUNTAS' || Provider.of<HistoryModule>(context).dataMisi()[i]['currentState'] == 'DISELESAIKAN') ? Container(
  442. child: hisFunc.askForRate(Provider.of<HistoryModule>(context, listen: true).dataMisi()[i]) ? GestureDetector(
  443. child: Container(
  444. padding: EdgeInsets.symmetric(vertical: 4, horizontal: 16),
  445. child: Text('rateReq'.tr(), style: TextStyle(color: secondaryColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  446. decoration: BoxDecoration(color: Colors.white, border: Border.all(color: secondaryColor), borderRadius: BorderRadius.all(Radius.circular(50))),
  447. ),
  448. onTap: () {
  449. U.newServerVersion(1716279633) ? rateMissionNew(context, Provider.of<HistoryModule>(context, listen: false).dataMisi()[i], i) : rateMission(context, Provider.of<HistoryModule>(context, listen: false).dataMisi()[i], i);
  450. },
  451. ) : !hisFunc.isAutoResponse(Provider.of<HistoryModule>(context).dataMisi()[i]) ? Row(
  452. children: [
  453. Image(image: AssetImage(rating[Provider.of<HistoryModule>(context).dataMisi()[i]['satisfactionRate'] - 1]['image'].toString()), width: 25),
  454. SizedBox(width: 5),
  455. Text(rating[Provider.of<HistoryModule>(context).dataMisi()[i]['satisfactionRate'] - 1]['label'].toString(), style: TextStyle(color: textColor, fontSize: 14), overflow: TextOverflow.ellipsis),
  456. ],
  457. ) : Container(),
  458. ) : message(Provider.of<HistoryModule>(context).dataMisi()[i]['_hasForum'], Provider.of<HistoryModule>(context).dataMisi()[i]['_forumMsg'] ?? "null"),
  459. GestureDetector(
  460. child: Container(
  461. padding: EdgeInsets.symmetric(vertical: 4, horizontal: 16),
  462. child: Text('reqAgain'.tr(), style: TextStyle(color: primaryColor, fontSize: 14)),
  463. decoration: BoxDecoration(color: Colors.white, border: Border.all(color: primaryColor), borderRadius: BorderRadius.all(Radius.circular(50))),
  464. ),
  465. onTap: () => hisFunc.requestAgainAction(context, i),
  466. )
  467. ],
  468. ),
  469. ],
  470. ),
  471. ),
  472. onTap: () {
  473. if (Provider.of<HistoryModule>(context, listen: false).multiSelectMode()) {
  474. hisFunc.selectedController(context, i);
  475. } else {
  476. navigateTo(context, WebHistoryDetailPage(index: i));
  477. }
  478. },
  479. onLongPress: () => hisFunc.selectedController(context, i),
  480. );
  481. })
  482. ),
  483. !Provider.of<HistoryModule>(context).stopLoadHistory() || (Provider.of<HistoryModule>(context).isLoadHistory() && Provider.of<HistoryModule>(context).page() > 0)
  484. ? Container(margin: EdgeInsets.only(top: Provider.of<HistoryModule>(context).page() > 0 ? 0 : 50), child: loadingTemplate())
  485. : Provider.of<HistoryModule>(context).dataMisi().length == 0 ? NoDataPage() : Container(),
  486. ],
  487. ),
  488. );
  489. }
  490. Widget renderRequested(list){
  491. if(list['receptionistId'] != null){
  492. var user = Provider.of<UserModule>(context, listen: false).user();
  493. if(user['roomAttendant'] && user['userId'] != list['informantUserId'] && user['userId'] == list['receptionistId']){
  494. return Column(
  495. crossAxisAlignment: CrossAxisAlignment.start,
  496. children: [
  497. Text('${'requestedFor'.tr()}: ${list['informantName']??'-'}' ,style: TextStyle(color: textColor, fontSize: 13), overflow: TextOverflow.ellipsis),
  498. SizedBox(height: 5),
  499. ],
  500. );
  501. }
  502. if(user['userId'] == list['informantUserId'] && user['userId'] != list['receptionistId']){
  503. return Column(
  504. crossAxisAlignment: CrossAxisAlignment.start,
  505. children: [
  506. Text('${'requestedBy'.tr()}: ${list['receptionistName']??''}' ,style: TextStyle(color: textColor, fontSize: 13), overflow: TextOverflow.ellipsis),
  507. SizedBox(height: 5),
  508. ],
  509. );
  510. }
  511. }
  512. return Container();
  513. }
  514. Widget message(bool _hasForum, String _forumMsg, {double top = 0}) {
  515. return _hasForum ? Container(
  516. margin: EdgeInsets.only(top: top),
  517. child: Row(
  518. children: [
  519. FadeTransition(opacity: _animationController, child: U.iconsax('messages-3', color: primaryColor)),
  520. SizedBox(width: 8),
  521. Text('“$_forumMsg”', style: TextStyle(color: textColor.withValues(alpha: 0.75), fontSize: 14, fontStyle: FontStyle.italic), overflow: TextOverflow.ellipsis),
  522. ],
  523. ),
  524. ) : Container();
  525. }
  526. renderStatus(text, color){
  527. return Container(
  528. padding: EdgeInsets.symmetric(vertical: 3, horizontal: 10),
  529. margin: EdgeInsets.only(left: 10),
  530. child: Text(text, style: TextStyle(fontSize: 12, color: textColor)),
  531. decoration: BoxDecoration(
  532. color: color, borderRadius: BorderRadius.all(Radius.circular(3))
  533. ),
  534. );
  535. }
  536. rateMission(BuildContext context, list, index) {
  537. showDialog(
  538. context: context,
  539. builder: (contexts) {
  540. int tempRating = 0;
  541. String description = '';
  542. return AlertDialog(
  543. backgroundColor: Colors.transparent,
  544. contentPadding: EdgeInsets.all(0),
  545. content: StatefulBuilder(
  546. builder: (contextt, setStates) {
  547. return Container(
  548. decoration: BoxDecoration(color: secondaryColor, borderRadius: BorderRadius.all(Radius.circular(25))),
  549. child: Column(
  550. mainAxisSize: MainAxisSize.min,
  551. crossAxisAlignment: CrossAxisAlignment.start,
  552. children: [
  553. Padding(
  554. padding: EdgeInsets.symmetric(vertical: 12, horizontal: 16),
  555. child: Column(
  556. crossAxisAlignment: CrossAxisAlignment.start,
  557. children: [
  558. Text(list[U.langColumn(context, 'requestSubject')], style: TextStyle(color: Colors.white, fontSize: 14, fontWeight: FontWeight.w600)),
  559. SizedBox(height: 4),
  560. Row(
  561. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  562. children: [
  563. Text('ticketNumber'.tr()+': '+list['ticketNo'], style: TextStyle(color: Colors.white, fontSize: 12, fontWeight: FontWeight.w300)),
  564. Text(convertDate(list['datetimeRequest'], context.locale.toString()), style: TextStyle(color: Colors.white, fontSize: 12, fontWeight: FontWeight.w300)),
  565. ],
  566. ),
  567. ],
  568. ),
  569. ),
  570. Container(
  571. padding: EdgeInsets.fromLTRB(30, 15, 30, 15),
  572. decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.all(Radius.circular(25))),
  573. child: Column(
  574. children: [
  575. Text('satisfactionAsk'.tr(), style: TextStyle(fontSize: 16)),
  576. SizedBox(height: 12),
  577. Row(
  578. mainAxisAlignment: MainAxisAlignment.center,
  579. children: List.generate(rating.length, (i) {
  580. return Expanded(
  581. child: GestureDetector(
  582. child: Image(
  583. image: AssetImage(rating[i]['image'].toString()),
  584. color: Colors.white.withValues(alpha: tempRating == rating[i]['key'] ? 1 : 0.3),
  585. colorBlendMode: BlendMode.modulate,
  586. ),
  587. // child: Icon(rating[i]['icon'] as IconData, color: tempRating == rating[i]['key']?rating[i]['color'] as Color:Colors.grey),
  588. onTap: () {
  589. setStates(() {
  590. tempRating = rating[i]['key'] as int;
  591. description = rating[i]['label'] as String;
  592. });
  593. },
  594. ),
  595. );
  596. }),
  597. ),
  598. SizedBox(height: 6),
  599. Text(description, style: TextStyle(color: textColor, fontSize: 14, fontWeight: FontWeight.w300), textAlign: TextAlign.center),
  600. SizedBox(height: 20),
  601. Opacity(
  602. opacity: tempRating > 0 ? 1 : 0,
  603. child: Row(
  604. mainAxisAlignment: MainAxisAlignment.center,
  605. children: [
  606. Expanded(child: buttonTemplate(text: 'textCancel'.tr(), backgroundColor: Colors.black12, borderColor: Colors.black12, textColor: textColor, height: 40, action: (){
  607. navigateBack(contexts);
  608. })),
  609. SizedBox(width: 20),
  610. Expanded(child: buttonTemplate(text: 'save_rating'.tr(), backgroundColor: primaryColor, borderColor: primaryColor, height: 40, action: ()async{
  611. if (tempRating > 0) {
  612. var data = {"ticketNo": list['ticketNo'], "rateSatisfy": tempRating};
  613. var res = await ApiAuthProvider() .postData('/api/requestHistories/rateSatisfy', null, data, contexts);
  614. if (res != null) {
  615. Provider.of<HistoryModule>(context, listen: false).setSatisfaction(index, tempRating);
  616. navigateBack(contexts);
  617. }
  618. }
  619. }))
  620. ],
  621. ),
  622. )
  623. ],
  624. ),
  625. )
  626. ],
  627. ),
  628. );
  629. },
  630. ),
  631. );
  632. }
  633. );
  634. }
  635. rateMissionNew(BuildContext context, list, index) {
  636. String locale = context.locale.toString();
  637. showDialog(
  638. context: context,
  639. builder: (contexts) {
  640. int tempRating = 0;
  641. String description = '';
  642. String ratingAspect = '';
  643. TextEditingController controllerOptOther = new TextEditingController();
  644. List aspectList = list['_ratingAspect'+locale[0].toUpperCase()+locale[1]] != null && list['_ratingAspect'+locale[0].toUpperCase()+locale[1]].trim() != '' ? list['_ratingAspect'+locale[0].toUpperCase()+locale[1]].split(';') : [];
  645. if(aspectList.isNotEmpty){
  646. aspectList.add('OTHER_OPTION');
  647. }
  648. else{
  649. ratingAspect = 'OTHER_OPTION';
  650. }
  651. return AlertDialog(
  652. backgroundColor: Colors.transparent,
  653. contentPadding: EdgeInsets.all(0),
  654. content: StatefulBuilder(
  655. builder: (contextt, setStates) {
  656. return Container(
  657. decoration: BoxDecoration(color: secondaryColor, borderRadius: BorderRadius.all(Radius.circular(25))),
  658. child: Column(
  659. mainAxisSize: MainAxisSize.min,
  660. crossAxisAlignment: CrossAxisAlignment.start,
  661. children: [
  662. Padding(
  663. padding: EdgeInsets.symmetric(vertical: 12, horizontal: 16),
  664. child: Column(
  665. crossAxisAlignment: CrossAxisAlignment.start,
  666. children: [
  667. Text(list[U.langColumn(context, 'requestSubject')], style: TextStyle(color: Colors.white, fontSize: 14, fontWeight: FontWeight.w600)),
  668. SizedBox(height: 4),
  669. Row(
  670. mainAxisAlignment: MainAxisAlignment.spaceBetween,
  671. children: [
  672. Text('ticketNumber'.tr()+': '+list['ticketNo'], style: TextStyle(color: Colors.white, fontSize: 12, fontWeight: FontWeight.w300)),
  673. Text(convertDate(list['datetimeRequest'], context.locale.toString()), style: TextStyle(color: Colors.white, fontSize: 12, fontWeight: FontWeight.w300)),
  674. ],
  675. ),
  676. ],
  677. ),
  678. ),
  679. Container(
  680. padding: EdgeInsets.fromLTRB(30, 15, 30, 15),
  681. decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.all(Radius.circular(25))),
  682. child: Column(
  683. children: [
  684. Text('satisfactionAsk'.tr(), style: TextStyle(fontSize: 16)),
  685. SizedBox(height: 12),
  686. Row(
  687. mainAxisAlignment: MainAxisAlignment.center,
  688. children: List.generate(rating.length, (i) {
  689. return Expanded(
  690. child: GestureDetector(
  691. child: Image(
  692. image: AssetImage(rating[i]['image'].toString()),
  693. color: Colors.white.withValues(alpha: tempRating == rating[i]['key'] ? 1 : 0.3),
  694. colorBlendMode: BlendMode.modulate,
  695. ),
  696. onTap: () {
  697. setStates(() {
  698. tempRating = rating[i]['key'] as int;
  699. description = rating[i]['label'] as String;
  700. });
  701. },
  702. ),
  703. );
  704. }),
  705. ),
  706. SizedBox(height: 6),
  707. Text(description, style: TextStyle(color: textColor, fontSize: 14, fontWeight: FontWeight.w300), textAlign: TextAlign.center),
  708. tempRating != 0 ? SizedBox(height: 20) : Container(),
  709. tempRating != 0 ? separator() : Container(),
  710. tempRating != 0 ? SizedBox(height: 20) : Container(),
  711. tempRating != 0 ? Column(
  712. crossAxisAlignment: CrossAxisAlignment.start,
  713. children: [
  714. Text('whatMakesYou'.tr()+' '+description+'?', style: TextStyle(fontSize: 16)),
  715. SizedBox(height: 12),
  716. Wrap(
  717. runSpacing: 10, spacing: 10,
  718. children: List.generate(aspectList.length, (i){
  719. return GestureDetector(
  720. child: Container(
  721. padding: EdgeInsets.symmetric(vertical: 5, horizontal: 10),
  722. child: Text(aspectList[i]=='OTHER_OPTION'?'letMeWrite'.tr():aspectList[i], style: TextStyle(color: textColor)),
  723. decoration: BoxDecoration(
  724. color: aspectList[i] == ratingAspect ? primaryColor.withValues(alpha: 0.3) : Colors.white,
  725. border: Border.all(color: aspectList[i] == ratingAspect ? primaryColor : textColor),
  726. borderRadius: BorderRadius.all(Radius.circular(50))
  727. ),
  728. ),
  729. onTap: (){
  730. setStates(() {
  731. ratingAspect = ratingAspect == aspectList[i]?'':aspectList[i];
  732. });
  733. },
  734. );
  735. }),
  736. ),
  737. aspectList.isNotEmpty ? SizedBox(height: 10) : Container(),
  738. ratingAspect == 'OTHER_OPTION' ? TextFormField(
  739. maxLines: 1, maxLength: 50,
  740. autofocus: true,
  741. controller: controllerOptOther,
  742. style: TextStyle(fontSize: 14, color: Colors.black),
  743. decoration: InputDecoration(
  744. counterText: '',
  745. counterStyle: TextStyle(fontSize: 0),
  746. hintText: 'writeHere'.tr(),
  747. hintStyle: TextStyle(color: textColor.withValues(alpha: 0.5), fontSize: 14),
  748. filled: true, fillColor: Colors.white,
  749. isDense: true,
  750. contentPadding: EdgeInsets.all(13),
  751. prefixIcon: Padding(padding: EdgeInsets.only(left: 13, right: 13), child: U.iconsax('edit-2', color: textColor, size: 22)),
  752. enabledBorder: OutlineInputBorder(borderRadius: BorderRadius.circular(12), borderSide: BorderSide(color: Color(0xff262626).withValues(alpha: 0.5))),
  753. focusedBorder: OutlineInputBorder(borderRadius: BorderRadius.circular(12), borderSide: BorderSide(color: primaryColor)),
  754. ),
  755. ) : Container(),
  756. ],
  757. ) : Container(),
  758. tempRating != 0 ? SizedBox(height: 20) : Container(),
  759. Opacity(
  760. opacity: tempRating > 0 ? 1 : 0,
  761. child: Row(
  762. mainAxisAlignment: MainAxisAlignment.center,
  763. children: [
  764. Expanded(child: buttonTemplate(text: 'textCancel'.tr(), backgroundColor: Colors.black12, borderColor: Colors.black12, textColor: textColor, height: 40, action: (){
  765. navigateBack(contexts);
  766. })),
  767. SizedBox(width: 20),
  768. Expanded(child: buttonTemplate(text: 'save_rating'.tr(), backgroundColor: primaryColor, borderColor: primaryColor, height: 40, action: ()async{
  769. if (tempRating > 0) {
  770. var data = {"ticketNo": list['ticketNo'], "rateSatisfy": tempRating};
  771. if(ratingAspect != 'OTHER_OPTION'){
  772. data['aspect'] = ratingAspect;
  773. }
  774. else if(ratingAspect == 'OTHER_OPTION' && controllerOptOther.text.isNotEmpty){
  775. data['aspect'] = controllerOptOther.text;
  776. }
  777. var res = await ApiAuthProvider() .postData('/api/requestHistories/rateSatisfy', null, data, contexts);
  778. if (res != null) {
  779. Provider.of<HistoryModule>(context, listen: false).setSatisfaction(index, tempRating);
  780. navigateBack(contexts);
  781. }
  782. }
  783. }))
  784. ],
  785. ),
  786. )
  787. ],
  788. ),
  789. )
  790. ],
  791. ),
  792. );
  793. },
  794. ),
  795. );
  796. }
  797. );
  798. }
  799. }