controller.dart 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246
  1. import 'dart:convert';
  2. import 'package:electronic_assistant/base/base_controller.dart';
  3. import 'package:electronic_assistant/data/bean/agenda.dart';
  4. import 'package:electronic_assistant/data/bean/chat_item.dart';
  5. import 'package:electronic_assistant/data/bean/file_chat_item.dart';
  6. import 'package:electronic_assistant/data/bean/progressing_chat_item.dart';
  7. import 'package:electronic_assistant/data/bean/reference_chat_item.dart';
  8. import 'package:electronic_assistant/data/repositories/account_repository.dart';
  9. import 'package:electronic_assistant/data/repositories/chat_repository.dart';
  10. import 'package:electronic_assistant/data/repositories/talk_repository.dart';
  11. import 'package:electronic_assistant/module/chat/start/view.dart';
  12. import 'package:electronic_assistant/resource/colors.gen.dart';
  13. import 'package:flutter/material.dart';
  14. import 'package:get/get.dart';
  15. import 'package:pull_to_refresh/pull_to_refresh.dart';
  16. import 'package:uuid/uuid.dart';
  17. import '../../data/bean/stream_chat_origin_data.dart';
  18. import '../../data/bean/talks.dart';
  19. import '../../data/consts/error_code.dart';
  20. import '../../router/app_pages.dart';
  21. import '../../utils/http_handler.dart';
  22. class ChatController extends BaseController {
  23. final RefreshController refreshController =
  24. RefreshController(initialLoadStatus: LoadStatus.loading);
  25. final ScrollController listScrollController = ScrollController();
  26. final TextEditingController inputController = TextEditingController();
  27. final RxList chatItems = [].obs;
  28. final Rxn<TalkBean> talkInfo = Rxn<TalkBean>();
  29. final Rxn<Agenda> agenda = Rxn<Agenda>();
  30. @override
  31. void onInit() {
  32. super.onInit();
  33. loadMoreHistory();
  34. checkArguments();
  35. }
  36. @override
  37. void onReady() {
  38. super.onReady();
  39. if (accountRepository.userInfo.value?.profession == null ||
  40. accountRepository.userInfo.value?.post == null) {
  41. showStartSheet();
  42. }
  43. }
  44. void checkArguments() {
  45. List<dynamic> arguments = Get.arguments ?? [];
  46. if (arguments.isEmpty) {
  47. return;
  48. }
  49. if (arguments.length > 1 && arguments[1] is Agenda) {
  50. agenda.value = arguments[1];
  51. }
  52. if (arguments[0] is TalkBean) {
  53. talkInfo.value = arguments[0];
  54. sendInitialMessage();
  55. } else if (arguments[0] is String) {
  56. talkRepository
  57. .talkInfo(arguments[0])
  58. .then((response) => talkInfo.value = response.talkInfo)
  59. .then((_) => sendInitialMessage());
  60. }
  61. }
  62. void sendInitialMessage() {
  63. final talkInfo = this.talkInfo.value;
  64. if (talkInfo == null) {
  65. return;
  66. }
  67. chatItems.insert(
  68. 0,
  69. FileChatItem(talkInfo,
  70. id: const Uuid().v4(),
  71. conversationId: "",
  72. role: "user",
  73. content: "",
  74. createTime: DateTime.now().toString()));
  75. if (agenda.value != null) {
  76. _sendMessage("以这份谈话为背景,你为我执行“${agenda.value?.content}”这一事项");
  77. }
  78. }
  79. onSendClick() {
  80. if (inputController.text.isEmpty) {
  81. return;
  82. }
  83. String chatContent = inputController.text;
  84. inputController.clear();
  85. _sendMessage(chatContent);
  86. }
  87. void onAddFileClick() {
  88. Get.toNamed(RoutePath.fileSearch)?.then((talkInfo) {
  89. if (talkInfo is TalkBean) {
  90. this.talkInfo.value = talkInfo;
  91. agenda.value = null;
  92. sendInitialMessage();
  93. }
  94. });
  95. }
  96. Future<void> loadMoreHistory() {
  97. bool isEmpty = chatItems.isEmpty;
  98. return chatRepository
  99. .chatHistory(chatItems.isEmpty ? null : chatItems.last.id)
  100. .then((value) => chatItems.addAll(value))
  101. .whenComplete(() => refreshController.loadComplete())
  102. .whenComplete(() {
  103. if (isEmpty) {
  104. _scrollToBottom();
  105. }
  106. });
  107. }
  108. void showStartSheet() {
  109. WidgetsBinding.instance.addPostFrameCallback((_) {
  110. showModalBottomSheet(
  111. context: Get.context!,
  112. isScrollControlled: true,
  113. barrierColor: ColorName.black55,
  114. backgroundColor: ColorName.transparent,
  115. builder: (BuildContext context) {
  116. return const ChatStartPage();
  117. },
  118. ).then((result) {
  119. if (!accountRepository.isLogin.value) {
  120. Get.back();
  121. }
  122. });
  123. });
  124. }
  125. void _sendMessage(String chatContent) {
  126. chatItems.insert(0, createUserChatItem(chatContent));
  127. ProgressingChatItem progressingChatItem = ProgressingChatItem(
  128. id: const Uuid().v4(),
  129. conversationId: chatItems.last.conversationId,
  130. role: "assistant",
  131. content: "",
  132. createTime: DateTime.now().toString(),
  133. );
  134. chatItems.insert(0, progressingChatItem);
  135. _scrollToBottom();
  136. chatRepository
  137. .streamChat(chatContent,
  138. talkId: talkInfo.value?.id, agendaId: agenda.value?.id)
  139. .then((stream) {
  140. stream.listen((event) {
  141. try {
  142. Map<String, dynamic> json = jsonDecode(event.data);
  143. if (json.isEmpty) {
  144. return;
  145. }
  146. StreamChatOriginData data = StreamChatOriginData.fromJson(json);
  147. if (data.choices == null || data.choices!.isEmpty) {
  148. return;
  149. }
  150. Delta? delta = data.choices![0].delta;
  151. if (delta == null) {
  152. return;
  153. }
  154. progressingChatItem.append(delta.content ?? "");
  155. } catch (ignore) {}
  156. }, onDone: () {
  157. progressingChatItem.content = progressingChatItem.streamContent.value;
  158. progressingChatItem.isFinished.value = true;
  159. }, onError: (error) {
  160. progressingChatItem.isFailed.value = true;
  161. progressingChatItem.error.value = "网络错误,请检查网络连接";
  162. debugPrint("error: $error");
  163. debugPrintStack();
  164. });
  165. }).catchError((error) {
  166. progressingChatItem.isFailed.value = true;
  167. if (error is ServerErrorException) {
  168. progressingChatItem.error.value = error.message ?? "服务出错,请稍后再试";
  169. if (error.code == ErrorCode.errorCodeNoLogin) {
  170. Get.toNamed(RoutePath.login)?.then((loginSuccess) async {
  171. if (loginSuccess != null && loginSuccess) {
  172. _clearChat();
  173. await loadMoreHistory();
  174. _sendMessage(chatContent);
  175. }
  176. });
  177. } else if (error.code == ErrorCode.errorCodeNoProfession) {
  178. showStartSheet();
  179. }
  180. } else {
  181. progressingChatItem.error.value = "网络错误,请检查网络连接";
  182. debugPrint("error: $error");
  183. debugPrintStack();
  184. }
  185. });
  186. }
  187. void _clearChat() async {
  188. return chatItems.clear();
  189. }
  190. void _scrollToBottom() {
  191. Future.delayed(const Duration(milliseconds: 300), () {
  192. listScrollController.animateTo(
  193. 0,
  194. duration: const Duration(milliseconds: 300),
  195. curve: Curves.easeOut,
  196. );
  197. });
  198. }
  199. createUserChatItem(String chatContent) {
  200. final id = const Uuid().v4();
  201. final conversationId =
  202. chatItems.isEmpty ? "" : chatItems.last.conversationId;
  203. const role = "user";
  204. final content = chatContent;
  205. final createTime = DateTime.now().toString();
  206. final talkInfo = this.talkInfo.value;
  207. return talkInfo == null
  208. ? ChatItem(
  209. id: id,
  210. conversationId: conversationId,
  211. role: role,
  212. content: chatContent,
  213. createTime: createTime)
  214. : ReferenceChatItem(
  215. talkInfo: talkInfo,
  216. id: id,
  217. conversationId: conversationId,
  218. role: role,
  219. content: content,
  220. createTime: createTime);
  221. }
  222. onDeleteReference() {
  223. talkInfo.value = null;
  224. agenda.value = null;
  225. }
  226. }