Hibok
Nie możesz wybrać więcej, niż 25 tematów Tematy muszą się zaczynać od litery lub cyfry, mogą zawierać myślniki ('-') i mogą mieć do 35 znaków.
 
 
 
 
 
 

414 wiersze
12 KiB

  1. import 'dart:convert';
  2. import 'package:chat/data/UserData.dart';
  3. import 'package:chat/generated/i18n.dart';
  4. import 'package:chat/models/ChatMsg.dart';
  5. import 'package:chat/models/group_info_model.dart';
  6. import 'package:chat/models/group_info_server_model.dart';
  7. import 'package:chat/proto/all.pbserver.dart';
  8. import 'package:chat/data/constants.dart';
  9. import 'package:chat/utils/file_cache_mgr.dart';
  10. import 'package:convert/convert.dart';
  11. import 'package:dio/dio.dart';
  12. import 'package:crypto/crypto.dart';
  13. import 'package:flutter/material.dart';
  14. import 'package:flutter/services.dart';
  15. import 'dart:async';
  16. import 'package:http_parser/http_parser.dart';
  17. import 'package:chat/utils/MessageMgr.dart';
  18. import 'package:oktoast/oktoast.dart';
  19. import 'package:chat/utils/msgHandler.dart';
  20. import 'package:chat/utils/sql_util.dart';
  21. var signKey = 'O@MlSxWaeAlc5CYu';
  22. class UploadUtil {
  23. // 工厂模式
  24. factory UploadUtil() => _getInstance();
  25. static UploadUtil get instance => _getInstance();
  26. static UploadUtil _instance;
  27. UploadUtil._internal() {
  28. // 初始化
  29. }
  30. static UploadUtil _getInstance() {
  31. if (_instance == null) {
  32. _instance = new UploadUtil._internal();
  33. }
  34. return _instance;
  35. }
  36. Dio dio;
  37. //进度流
  38. Map<String, StreamController<double>> streamMap = {};
  39. //缓存进度,方便快速显示
  40. Map<String, double> streamLastPercentMap = {};
  41. //上传地址
  42. String uploadUrl;
  43. String rootUrl;
  44. void setUploadUrl(url) {
  45. rootUrl = 'http://' + url;
  46. uploadUrl = 'http://' + url + '/chatresfile/';
  47. BaseOptions options = BaseOptions(
  48. baseUrl: uploadUrl,
  49. headers: {},
  50. );
  51. print(uploadUrl);
  52. dio = new Dio(options);
  53. }
  54. // md5 加密
  55. String generateMd5(String data) {
  56. var content = new Utf8Encoder().convert(data);
  57. var digest = md5.convert(content);
  58. // 这里其实就是 digest.toString()
  59. return hex.encode(digest.bytes);
  60. }
  61. String getSign(Map parameter) {
  62. String k = '';
  63. var keyList = [];
  64. for (var key in parameter.keys) {
  65. keyList.add(key);
  66. }
  67. keyList.sort();
  68. for (var key in keyList) {
  69. k += '$key=${parameter[key]}&';
  70. }
  71. k += 'key=O@MlSxWaeAlc5CYu';
  72. return generateMd5(k);
  73. }
  74. String getFileType(String suffix) {
  75. if (suffix == 'png' || suffix == 'jpg' || suffix == 'jpeg') {
  76. return 'image';
  77. } else if (suffix == 'wav') {
  78. return 'audio';
  79. } else {
  80. return 'video';
  81. }
  82. }
  83. bool isImage(String suffix) {
  84. suffix = suffix.toLowerCase();
  85. return suffix == 'png' || suffix == 'jpg' || suffix == 'jpeg';
  86. }
  87. StreamController<double> getStream(String fileId) {
  88. if (streamMap[fileId] == null) {
  89. streamMap[fileId] = StreamController<double>.broadcast();
  90. }
  91. return streamMap[fileId];
  92. }
  93. Future uploadFile(MsgModel msg, {CancelToken cancelToken}) async {
  94. Response response;
  95. var data = {
  96. "upuserid": msg.from,
  97. 'channeltype': msg.channelType,
  98. "targetid": msg.sessionId,
  99. 'fileid': msg.extraFile,
  100. };
  101. print('上传参数$data');
  102. var path = msg.localFile;
  103. var name = path.substring(path.lastIndexOf("/") + 1, path.length);
  104. String suffix = name.substring(name.lastIndexOf(".") + 1, name.length);
  105. var fileType = getFileType(suffix);
  106. print('文件类型 : $fileType $suffix');
  107. data['sign'] = getSign(data);
  108. data['uploadfile'] = await MultipartFile.fromFile(path,
  109. filename: name, contentType: MediaType(fileType, suffix));
  110. FormData formData = new FormData.fromMap(data);
  111. try {
  112. print('开始上传文件${msg.extraFile} url:${uploadUrl + 'uploadchatfile'}');
  113. var streamController = getStream(msg.extraFile);
  114. response = await dio.post(uploadUrl + 'uploadchatfile', data: formData,
  115. onSendProgress: (int progress, int total) {
  116. streamController.sink.add(progress / total);
  117. streamLastPercentMap[msg.extraFile] = progress / total;
  118. }, cancelToken: cancelToken);
  119. streamController.close();
  120. streamMap.remove(msg.extraFile);
  121. print('uploadFile response $response');
  122. print('###msgContent ${msg.msgContent.length}');
  123. if (response.data['code'] == 0) {
  124. var data = response.data['data'];
  125. print('uploadFile 上传成功');
  126. msg.state = MsgState.Uploaded;
  127. MsgHandler.sendChatMsg(msg);
  128. MessageMgr().emit('Update LastMsg', msg.sessionId);
  129. return data;
  130. } else {
  131. print('uploadFile 上传失败');
  132. msg.state = MsgState.UploadFailed;
  133. MessageMgr().emit('Update LastMsg', msg.sessionId);
  134. return null;
  135. }
  136. } on DioError catch (e) {
  137. if (CancelToken.isCancel(e)) {
  138. print('post请求取消! ' + e.message);
  139. }
  140. showToast('上传文件失败');
  141. msg.state = MsgState.UploadFailed;
  142. print('post请求发生错误:$e');
  143. }
  144. MessageMgr().emit('Update LastMsg', msg.sessionId);
  145. return null;
  146. }
  147. void cancelRequests(CancelToken cancelToken) {
  148. print('取消请求$cancelToken');
  149. cancelToken.cancel("cancelled");
  150. }
  151. String getFullUrl(String imgData, int sessionid, int channelType) {
  152. var data = {
  153. "fileid": imgData,
  154. 'targetid': sessionid,
  155. "userid": UserData().basicInfo.userId,
  156. 'channeltype': channelType
  157. };
  158. var sign = getSign(data);
  159. return uploadUrl +
  160. 'downloadchatfile?targetid=$sessionid&channeltype=$channelType&userid=${UserData().basicInfo.userId}&fileid=$imgData&sign=$sign';
  161. }
  162. copyFileUrl(MsgModel msg, BuildContext context) {
  163. var url = getFullUrl(msg.extraFile, msg.sessionId, msg.channelType);
  164. ClipboardData clipboardData = new ClipboardData(text: url);
  165. Clipboard.setData(clipboardData);
  166. showToast(I18n.of(context).successful_copy);
  167. }
  168. Future downloadFile(MsgModel msg, {CancelToken cancelToken}) async {
  169. Response response;
  170. var fullUrl = getFullUrl(msg.extraFile, msg.sessionId, msg.channelType);
  171. print('下载文件$fullUrl');
  172. var path = await FileCacheMgr().genFilePath(msg.extraFile);
  173. if (msg.msgType == ChatType.ShortVoiceChatType.value) {
  174. path = path + '.wav';
  175. } else if (msg.msgType == ChatType.FileChatType.value) {
  176. var fileMsg = FileChat.fromBuffer(msg.msgContent);
  177. var fileName = fileMsg.name;
  178. path = await FileCacheMgr().genFilePath(fileName);
  179. }
  180. print('文件保存路径$path');
  181. try {
  182. var streamController = getStream(msg.extraFile);
  183. response = await dio.download(fullUrl, path,
  184. onReceiveProgress: (int progress, int total) {
  185. streamController.sink.add(progress / total);
  186. streamLastPercentMap[msg.extraFile] = progress / total;
  187. }, cancelToken: cancelToken);
  188. streamController.close();
  189. streamMap.remove(msg.extraFile);
  190. if (response.statusCode == 200) {
  191. msg.state = MsgState.DownloadSuccess;
  192. msg.localFile = path;
  193. SqlUtil().updateLocalFile(msg.extraFile, path,
  194. isGroup: msg.channelType == ChatChannelType.Group.value);
  195. return path;
  196. } else {
  197. msg.state = MsgState.DownloadFailed;
  198. showToast(I18n.of(Constants.getCurrentContext()).server_error_tips);
  199. return null;
  200. }
  201. } catch (e) {
  202. msg.state = MsgState.DownloadFailed;
  203. showToast(I18n.of(Constants.getCurrentContext()).server_error_tips);
  204. print(response.toString());
  205. }
  206. }
  207. //机器人翻译,type 1文字,2语音
  208. Future<String> commitTranslateSource(
  209. int type, int sourceLang, int toLang, dynamic content) async {
  210. Response response;
  211. var myId = UserData().basicInfo.userId;
  212. Map<String, dynamic> data = {
  213. 'userid': myId,
  214. 'cType': type,
  215. 'ulanguage': sourceLang,
  216. 'tlanguage': toLang
  217. };
  218. data['sign'] = getSign(data);
  219. data['content'] = content;
  220. if (type == 2) {
  221. data['content'] = await MultipartFile.fromFile(content);
  222. }
  223. FormData formData = new FormData.fromMap(data);
  224. try {
  225. print('开始请求翻译$data ${uploadUrl + 'translate/transqt'}');
  226. response = await dio.post(rootUrl + '/translate/transqt', data: formData);
  227. Map resData = response.data;
  228. print('翻译结果$resData');
  229. if (resData['code'] == 0) {
  230. return resData['data'];
  231. }
  232. } catch (e) {
  233. print(e);
  234. }
  235. return null;
  236. }
  237. //获取群列表
  238. Future<List<GroupInfoModel>> getGroupList() async {
  239. Response response;
  240. var myId = UserData().basicInfo.userId;
  241. Map<String, dynamic> data = {
  242. 'userid': myId,
  243. };
  244. data['sign'] = getSign(data);
  245. try {
  246. print('开始请求群列表$data}');
  247. response = await dio.post(rootUrl + '/chat/queryusergroups', data: data);
  248. Map resData = response.data;
  249. if (resData['code'] == 0) {
  250. List resList = resData['data'];
  251. List<GroupInfoModel> groupList = [];
  252. for (var i = 0; i < resList.length; i++) {
  253. var info = GroupInfoServerModel.fromJson(resList[i]);
  254. groupList.add(GroupInfoModel.fromServerGroupInfo(info));
  255. }
  256. return groupList;
  257. }
  258. } catch (e) {
  259. print(e);
  260. }
  261. return null;
  262. }
  263. //获取多个群信息
  264. Future<List<GroupInfoModel>> getMoreGroupInfo(List<int> groupIds) async {
  265. Response response;
  266. var myId = UserData().basicInfo.userId;
  267. Map<String, dynamic> data = {
  268. 'userid': myId,
  269. };
  270. data['sign'] = getSign(data);
  271. data['groupids'] = groupIds;
  272. try {
  273. print('开始请求 getMoreGroupInfo $data');
  274. response = await dio.post(rootUrl + '/chat/querygroups', data: data);
  275. Map resData = response.data;
  276. if (resData['code'] == 0) {
  277. List data = resData['data'];
  278. print('群信息$data');
  279. if (data != null && data.length > 0) {
  280. List<GroupInfoModel> res = [];
  281. for (var each in data) {
  282. var info = GroupInfoServerModel.fromJson(each);
  283. res.add(GroupInfoModel.fromServerGroupInfo(info));
  284. }
  285. return res;
  286. }
  287. }
  288. } catch (e) {
  289. print(e);
  290. }
  291. return null;
  292. }
  293. //获取群信息
  294. Future<GroupInfoModel> getGroupInfo(int groupId) async {
  295. Response response;
  296. var myId = UserData().basicInfo.userId;
  297. Map<String, dynamic> data = {
  298. 'userid': myId,
  299. };
  300. data['sign'] = getSign(data);
  301. data['groupids'] = [groupId];
  302. try {
  303. print('开始请求群信息$data');
  304. response = await dio.post(rootUrl + '/chat/querygroups', data: data);
  305. Map resData = response.data;
  306. if (resData['code'] == 0) {
  307. List data = resData['data'];
  308. print('群信息$data');
  309. if (data != null && data.length > 0) {
  310. var info = GroupInfoServerModel.fromJson(data.first);
  311. return GroupInfoModel.fromServerGroupInfo(info);
  312. }
  313. }
  314. } catch (e) {
  315. print(e);
  316. }
  317. return null;
  318. }
  319. //获取群校验信息
  320. Future<Map> getGroupCheckInfo() async {
  321. Response response;
  322. var myId = UserData().basicInfo.userId;
  323. Map<String, dynamic> data = {
  324. 'userid': myId,
  325. };
  326. data['sign'] = getSign(data);
  327. try {
  328. print('请求获取群校验信息$data');
  329. response = await dio.post(rootUrl + '/chat/checkgroup', data: data);
  330. Map resData = response.data;
  331. if (resData['code'] == 0) {
  332. print('群校验信息${resData['data']}');
  333. return resData['data'];
  334. }
  335. } catch (e) {
  336. print(e);
  337. }
  338. return null;
  339. }
  340. }