controller.dart 6.9 KB

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