controller.dart 6.7 KB

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