message_broadcast.dart 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. import 'dart:io';
  2. import 'package:dio/dio.dart';
  3. import 'package:easy_localization/easy_localization.dart';
  4. import 'package:file_picker/file_picker.dart';
  5. import 'package:flutter/foundation.dart';
  6. import 'package:flutter/material.dart';
  7. import 'package:telnow_mobile_new/src/api/api_auth_provider.dart';
  8. import 'package:telnow_mobile_new/src/layouts/components/template.dart';
  9. import 'package:telnow_mobile_new/src/utils/U.dart';
  10. class MobMessageBroadcastPage extends StatefulWidget {
  11. final user;
  12. final recipient;
  13. const MobMessageBroadcastPage(this.user, this.recipient, {super.key});
  14. @override
  15. State<MobMessageBroadcastPage> createState() => _MobMessageBroadcastPageState();
  16. }
  17. class _MobMessageBroadcastPageState extends State<MobMessageBroadcastPage> {
  18. final ApiAuthProvider _apiAuthProvider = ApiAuthProvider();
  19. bool _isDisabled = true;
  20. List data = [];
  21. String listOfTenant = '';
  22. String? textContent;
  23. String _filePath = '';
  24. String _fileName = '';
  25. String _fileExt = '';
  26. PlatformFile? fileX;
  27. var _prevImg;
  28. File? file;
  29. int count = 0;
  30. bool _isSending = false;
  31. _buttonController(value) {
  32. textContent = value;
  33. bool activate = true;
  34. if (value.length == 0) {
  35. activate = false;
  36. }
  37. setState(() {
  38. _isDisabled = !activate;
  39. });
  40. }
  41. _sendMyMessage() async {
  42. var recTenant = listOfTenant;
  43. var params = {'recipientTenant': recTenant, 'text': textContent};
  44. var data;
  45. if (file != null || fileX != null) {
  46. var _file;
  47. _file = kIsWeb ? await MultipartFile.fromBytes(fileX!.bytes!, filename: fileX!.name) : await MultipartFile.fromFile(_filePath);
  48. FormData formData = FormData.fromMap({
  49. 'file': _file,
  50. });
  51. data = formData;
  52. }
  53. setState(() {
  54. _isSending = true;
  55. });
  56. var res = await _apiAuthProvider.postData('/api/messages/broadcast', params, data, context);
  57. // print("res : ${res.toString()}");
  58. if (res != null && mounted) {
  59. Navigator.pop(context, 'pop');
  60. }
  61. else{
  62. setState(() {
  63. _isSending = false;
  64. });
  65. }
  66. }
  67. @override
  68. void initState() {
  69. // TODO: implement initState
  70. data = widget.recipient;
  71. if (data.isNotEmpty) {
  72. // int? totalLength = 96;
  73. data.sort((a, b) => (a['name'].length).compareTo(b['name'].length));
  74. listOfTenant = data.map((e) => e['code']).join(',');
  75. // data.forEach((e) {
  76. // totalLength = (totalLength! - e['name'].length) as int?;
  77. // totalLength! >= 0 ? count++ : null;
  78. // listOfTenant += ',' + e['code'];
  79. // });
  80. }
  81. // print(data);
  82. super.initState();
  83. }
  84. @override
  85. Widget build(BuildContext context) {
  86. return Scaffold(
  87. backgroundColor: Colors.white,
  88. appBar: appBarTemplate(context: context, title: 'composeMessage'.tr(), action: [
  89. !_isSending ? Padding(
  90. padding: EdgeInsets.only(right: 20),
  91. child: GestureDetector(
  92. child: U.iconsax('bold/send-1', size: 30, color: _isDisabled ? primaryColor.withOpacity(0.5):primaryColor),
  93. onTap: _isDisabled ? null : ()=>_sendMyMessage(),
  94. ),
  95. ):Container()
  96. ]),
  97. body: !_isSending ? Column(
  98. children: [
  99. divider(),
  100. Expanded(
  101. child: Container(
  102. alignment: Alignment.topCenter,
  103. width: U.bodyWidth(context),
  104. padding: EdgeInsets.fromLTRB(15, 15, 15, 0),
  105. child: SingleChildScrollView(
  106. child: Column(
  107. crossAxisAlignment: CrossAxisAlignment.start,
  108. children: [
  109. widget.recipient != null ? Row(
  110. crossAxisAlignment: CrossAxisAlignment.start,
  111. children: [
  112. Padding(
  113. padding: EdgeInsets.only(top: 4),
  114. child: Text('to'.tr()),
  115. ),
  116. data.isNotEmpty?Expanded(
  117. child: Wrap(
  118. children: List.generate(data.length, (i) {
  119. return Container(
  120. decoration: BoxDecoration(color: Color(U.getColor(data[i]['code'])), borderRadius: BorderRadius.all(Radius.circular(2))),
  121. padding: EdgeInsets.fromLTRB(8, 4, 8, 4),
  122. margin: EdgeInsets.fromLTRB(8, 0, 0, 8),
  123. child: Text(data[i]['name'], style: TextStyle(color: Colors.white)),
  124. );
  125. }),
  126. )
  127. ):Container(
  128. decoration: BoxDecoration(color: Color(U.getColor('allInformants')), borderRadius: BorderRadius.all(Radius.circular(2))),
  129. padding: EdgeInsets.fromLTRB(8, 4, 8, 4),
  130. margin: EdgeInsets.fromLTRB(8, 0, 0, 8),
  131. child: Text(
  132. "allInformants".tr(),
  133. style: TextStyle(color: Colors.white),
  134. ),
  135. )
  136. ],
  137. ) : Container(),
  138. SizedBox(height: 16),
  139. Text('message'.tr()),
  140. SizedBox(height: 8),
  141. TextField(
  142. maxLines: null,
  143. maxLength: 256,
  144. decoration: InputDecoration(
  145. border: OutlineInputBorder(),
  146. focusedBorder: OutlineInputBorder(
  147. borderSide: const BorderSide(color: Colors.black45),
  148. )),
  149. onChanged: (String value) async {
  150. _buttonController(value);
  151. },
  152. ),
  153. Align(
  154. alignment: Alignment.bottomLeft,
  155. child: Container(
  156. child: GestureDetector(
  157. child: Container(
  158. child: Row(
  159. children: [
  160. Icon(Icons.attach_file_outlined),
  161. SizedBox(
  162. width: 4,
  163. ),
  164. Text(_filePath == '' ? 'addAttachment'.tr() : 'editAttachment'.tr())
  165. ],
  166. ),
  167. ),
  168. onTap: () async {
  169. var allowedExt = ['jpg', 'pdf', 'png', 'jpeg'];
  170. _filePath = '';
  171. _fileName = '';
  172. FilePickerResult? result = await FilePicker.platform.pickFiles(
  173. type: FileType.custom,
  174. allowedExtensions: allowedExt,
  175. withReadStream: false,
  176. );
  177. if (result != null) {
  178. PlatformFile _file = result.files.first;
  179. if (!allowedExt.contains(_file.extension)) {
  180. final snackBar = SnackBar(
  181. content: Row(
  182. children: [
  183. Icon(
  184. Icons.warning,
  185. color: Colors.yellow,
  186. size: 18,
  187. ),
  188. SizedBox(
  189. width: 8,
  190. ),
  191. Text(
  192. 'unsupportedFile'.tr(),
  193. style: TextStyle(),
  194. overflow: TextOverflow.fade,
  195. maxLines: 2,
  196. )
  197. ],
  198. ));
  199. ScaffoldMessenger.of(context).showSnackBar(snackBar);
  200. } else {
  201. _fileName = _file.name;
  202. _fileExt = _file.extension!;
  203. if (!kIsWeb) {
  204. file = File(result.files.single.path!);
  205. _filePath = _file.path!;
  206. } else {
  207. fileX = result.files.single;
  208. _prevImg = Image.memory(fileX!.bytes!);
  209. }
  210. }
  211. }
  212. setState(() {});
  213. }),
  214. ),
  215. ),
  216. _fileName != '' ? Padding(
  217. padding: EdgeInsets.only(top: 12),
  218. child: Text(_fileName, overflow: TextOverflow.ellipsis),
  219. ) : Container(),
  220. _filePath != '' || fileX != null ? Padding(
  221. padding: EdgeInsets.only(top: 12, bottom: 12),
  222. child: Stack(
  223. children: [
  224. _fileExt != 'pdf' ? !kIsWeb ? Image.file(File(_filePath)) : _prevImg != null ? _prevImg : Container() : Padding(
  225. padding: EdgeInsets.only(top: 6, left: 8),
  226. child: Row(
  227. children: [
  228. Icon(Icons.picture_as_pdf),
  229. SizedBox(width: 6),
  230. Text('pdfFile'.tr())
  231. ],
  232. ),
  233. ),
  234. GestureDetector(
  235. child: Container(
  236. width: double.infinity,
  237. height: 36,
  238. padding: EdgeInsets.only(top: 6, right: 8),
  239. decoration: BoxDecoration(color: Colors.transparent.withOpacity(0.5), gradient: LinearGradient(end: Alignment.bottomCenter, begin: Alignment.center, colors: [Color(0xffDADADA), Colors.white])),
  240. child: Align(
  241. child: Row(
  242. mainAxisAlignment: MainAxisAlignment.end,
  243. children: [
  244. Text('delete'.tr()),
  245. Icon(Icons.clear)
  246. ],
  247. ),
  248. alignment: Alignment.topRight,
  249. ),
  250. ),
  251. onTap: () {
  252. // print('delete please!');
  253. setState(() {
  254. _filePath = '';
  255. _fileName = '';
  256. fileX = null;
  257. });
  258. },
  259. )
  260. ],
  261. ),
  262. ) : Container()
  263. ],
  264. ),
  265. )
  266. ),
  267. )
  268. ],
  269. ):loadingTemplate(),
  270. );
  271. }
  272. }