From 51e7501121f3919aeb3db9f9dc1a1281a89b8c65 Mon Sep 17 00:00:00 2001 From: qianlifeng Date: Thu, 21 Dec 2023 23:53:37 +0800 Subject: [PATCH] Optimize chatgpt plugin and result refresh implementation --- .../launcher/wox_launcher_controller.dart | 58 +++++++------ .../wox/lib/utils/wox_websocket_msg_util.dart | 31 ++++++- Wox/plugin/system/chatgpt.go | 83 +++++++++++++++---- Wox/plugin/system/url.go | 22 +++++ 4 files changed, 147 insertions(+), 47 deletions(-) diff --git a/Wox.UI.Flutter/wox/lib/modules/launcher/wox_launcher_controller.dart b/Wox.UI.Flutter/wox/lib/modules/launcher/wox_launcher_controller.dart index 36e2c1e1d..04ecd1966 100644 --- a/Wox.UI.Flutter/wox/lib/modules/launcher/wox_launcher_controller.dart +++ b/Wox.UI.Flutter/wox/lib/modules/launcher/wox_launcher_controller.dart @@ -1,5 +1,4 @@ import 'dart:async'; -import 'dart:convert'; import 'dart:io'; import 'dart:ui'; @@ -193,13 +192,14 @@ class WoxLauncherController extends GetxController implements WoxLauncherInterfa if (query.queryType == WoxQueryTypeEnum.WOX_QUERY_TYPE_INPUT.code) { // save the cursor position - final cursorPosition = queryBoxTextFieldController.selection.baseOffset; + var cursorPosition = queryBoxTextFieldController.selection.baseOffset; queryBoxTextFieldController.text = query.queryText; if (moveCursorToEnd) { moveQueryBoxCursorToEnd(); } else { // try to restore the cursor position after set text, which will reset the cursor position + cursorPosition = cursorPosition > queryBoxTextFieldController.text.length ? queryBoxTextFieldController.text.length : cursorPosition; queryBoxTextFieldController.selection = TextSelection(baseOffset: cursorPosition, extentOffset: cursorPosition); } } else { @@ -273,13 +273,7 @@ class WoxLauncherController extends GetxController implements WoxLauncherInterfa filterResultActions.refresh(); } - void handleWebSocketMessage(event) { - var msg = WoxWebsocketMsg.fromJson(jsonDecode(event)); - if (msg.success == false) { - Logger.instance.error("Received error message: ${msg.toJson()}"); - return; - } - + void handleWebSocketMessage(WoxWebsocketMsg msg) { if (msg.method != "Query") { Logger.instance.info("Received message: ${msg.method}"); } @@ -304,9 +298,6 @@ class WoxLauncherController extends GetxController implements WoxLauncherInterfa final theme = WoxTheme.fromJson(msg.data); WoxThemeUtil.instance.changeTheme(theme); woxTheme.value = theme; - } else if (msg.method == "Refresh") { - final result = WoxRefreshableResult.fromJson(msg.data); - onRefreshResult(result); } } @@ -478,11 +469,23 @@ class WoxLauncherController extends GetxController implements WoxLauncherInterfa } startRefreshSchedule() { - Timer.periodic(const Duration(milliseconds: 100), (timer) { + var isRequesting = {}; + Timer.periodic(const Duration(milliseconds: 100), (timer) async { + var isVisible = await windowManager.isVisible(); + if (!isVisible) { + return; + } + refreshCounter = refreshCounter + 100; for (var result in queryResults) { if (result.refreshInterval > 0 && refreshCounter % result.refreshInterval == 0) { - var msg = WoxWebsocketMsg( + if (isRequesting.containsKey(result.id)) { + continue; + } else { + isRequesting[result.id] = true; + } + + final msg = WoxWebsocketMsg( id: const UuidV4().generate(), type: WoxMsgTypeEnum.WOX_MSG_TYPE_REQUEST.code, method: WoxMsgMethodEnum.WOX_MSG_METHOD_REFRESH.code, @@ -498,27 +501,22 @@ class WoxLauncherController extends GetxController implements WoxLauncherInterfa ).toJson(), }, ); - WoxWebsocketMsgUtil.instance.sendMessage(msg); + WoxWebsocketMsgUtil.instance.sendMessage(msg).then((resp) { + final refreshResult = WoxRefreshableResult.fromJson(resp); + result.title.value = refreshResult.title; + result.subTitle.value = refreshResult.subTitle; + result.icon.value = refreshResult.icon; + result.preview = refreshResult.preview; + currentPreview.value = refreshResult.preview; + result.contextData = refreshResult.contextData; + result.refreshInterval = refreshResult.refreshInterval; + isRequesting.remove(result.id); + }); } } }); } - void onRefreshResult(WoxRefreshableResult result) { - for (var i = 0; i < queryResults.length; i++) { - if (queryResults[i].id == result.resultId) { - queryResults[i].title.value = result.title; - queryResults[i].subTitle.value = result.subTitle; - queryResults[i].icon.value = result.icon; - queryResults[i].preview = result.preview; - currentPreview.value = result.preview; - queryResults[i].contextData = result.contextData; - queryResults[i].refreshInterval = result.refreshInterval; - break; - } - } - } - @override void dispose() { queryBoxFocusNode.dispose(); diff --git a/Wox.UI.Flutter/wox/lib/utils/wox_websocket_msg_util.dart b/Wox.UI.Flutter/wox/lib/utils/wox_websocket_msg_util.dart index 7b74cdd8e..d190e227b 100644 --- a/Wox.UI.Flutter/wox/lib/utils/wox_websocket_msg_util.dart +++ b/Wox.UI.Flutter/wox/lib/utils/wox_websocket_msg_util.dart @@ -1,7 +1,9 @@ +import 'dart:async'; import 'dart:convert'; import 'package:web_socket_channel/web_socket_channel.dart'; import 'package:wox/entity/wox_websocket_msg.dart'; +import 'package:wox/enums/wox_msg_method_enum.dart'; import 'package:wox/utils/log.dart'; class WoxWebsocketMsgUtil { @@ -19,6 +21,8 @@ class WoxWebsocketMsgUtil { int connectionAttempts = 1; + final Map _completers = {}; + void _connect() { _channel?.sink.close(); _channel = null; @@ -26,7 +30,19 @@ class WoxWebsocketMsgUtil { _channel = WebSocketChannel.connect(uri); _channel!.stream.listen( (event) { - onMessageReceived(event); + var msg = WoxWebsocketMsg.fromJson(jsonDecode(event)); + if (msg.success == false) { + Logger.instance.error("Received error message: ${msg.toJson()}"); + return; + } + + if (_completers.containsKey(msg.id)) { + _completers[msg.id]!.complete(msg); + _completers.remove(msg.id); + return; + } + + onMessageReceived(msg); }, onDone: () { _reconnect(); @@ -50,7 +66,18 @@ class WoxWebsocketMsgUtil { } // send message to websocket server - void sendMessage(WoxWebsocketMsg msg) { + Future sendMessage(WoxWebsocketMsg msg) async { + // if query message, send it directly, no need to wait for response + // because query result may return multiple times + if (msg.method == WoxMsgMethodEnum.WOX_MSG_METHOD_QUERY.code) { + _channel?.sink.add(jsonEncode(msg)); + return; + } + + Completer completer = Completer(); + _completers[msg.id] = completer; _channel?.sink.add(jsonEncode(msg)); + var responseMsg = await completer.future as WoxWebsocketMsg; + return responseMsg.data; } } diff --git a/Wox/plugin/system/chatgpt.go b/Wox/plugin/system/chatgpt.go index dccdf14e5..7a992ce83 100644 --- a/Wox/plugin/system/chatgpt.go +++ b/Wox/plugin/system/chatgpt.go @@ -19,6 +19,10 @@ import ( var chatgptIcon = plugin.NewWoxImageBase64(``) var chatgptLoadingIcon = plugin.NewWoxImageBase64(``) +var chatgptDeleteIcon = plugin.NewWoxImageSvg(``) +var chatgptNewChatIcon = plugin.NewWoxImageBase64(``) +var chatgptActiveChatIcon = plugin.NewWoxImageBase64(``) +var chatgptRenameIcon = plugin.NewWoxImageBase64(``) func init() { plugin.AllSystemPlugin = append(plugin.AllSystemPlugin, &ChatgptPlugin{}) @@ -29,9 +33,10 @@ type ChatgptPlugin struct { client *openai.Client nonActiveChats []*Chat - activeChat *Chat - activeChatResult *plugin.QueryResult - activeChatAnswer string + activeChat *Chat + activeChatResult *plugin.QueryResult + activeChatAnswer string + isSummaryActiveChatTitle bool } type Chat struct { @@ -140,10 +145,11 @@ func (c *ChatgptPlugin) queryConversation(ctx context.Context, query plugin.Quer c.activeChatResult = &plugin.QueryResult{ Title: c.activeChat.Title, SubTitle: "Current active chat", - Icon: chatgptIcon, + Icon: chatgptActiveChatIcon, } } + c.activeChatResult.Title = c.activeChat.Title chatHistory := c.activeChat.Format() if chatHistory == "" && query.Search == "" { chatHistory = "Please ask anything to continue..." @@ -212,6 +218,18 @@ func (c *ChatgptPlugin) queryConversation(ctx context.Context, query plugin.Quer Text: c.activeChatAnswer, Timestamp: util.GetSystemTimestamp(), }) + + if c.activeChat.Title == "New chat" && len(c.activeChat.Conversations) >= 4 && !c.isSummaryActiveChatTitle { + c.api.Log(ctx, "start to summary chat title") + c.isSummaryActiveChatTitle = true + if title, titleErr := c.summaryChatTitle(ctx, c.activeChat.Conversations); titleErr == nil { + c.activeChat.Title = title + c.activeChatResult.Title = title + current.Title = title + } + c.isSummaryActiveChatTitle = false + } + c.activeChatAnswer = "" c.saveActiveChat(ctx) return current @@ -231,6 +249,7 @@ func (c *ChatgptPlugin) queryConversation(ctx context.Context, query plugin.Quer { Name: "Delete chat", PreventHideAfterAction: true, + Icon: chatgptDeleteIcon, Action: func(actionContext plugin.ActionContext) { c.deleteChat(ctx, c.activeChat.Id, true) c.api.ChangeQuery(ctx, share.ChangedQuery{ @@ -239,34 +258,38 @@ func (c *ChatgptPlugin) queryConversation(ctx context.Context, query plugin.Quer }) }, }, + { + Name: "Rename chat", + PreventHideAfterAction: true, + Icon: chatgptRenameIcon, + Action: func(actionContext plugin.ActionContext) { + //TODO: + c.api.ChangeQuery(ctx, share.ChangedQuery{ + QueryType: plugin.QueryTypeInput, + QueryText: query.TriggerKeyword + " ", + }) + }, + }, } results = append(results, *c.activeChatResult) } - newChatPreviewData := "Please input conversation title to continue..." - if query.Search != "" { - newChatPreviewData = fmt.Sprintf("Please input conversation title to continue\n\nTitle: %s", query.Search) - } results = append(results, plugin.QueryResult{ Title: "Start a new chat", Preview: plugin.WoxPreview{ PreviewType: plugin.WoxPreviewTypeMarkdown, - PreviewData: newChatPreviewData, + PreviewData: "Start a new chat", }, - Icon: chatgptIcon, + Icon: chatgptNewChatIcon, Actions: []plugin.QueryResultAction{ { Name: "Start", PreventHideAfterAction: true, Action: func(actionContext plugin.ActionContext) { - if query.Search == "" { - return - } - newChat := &Chat{ Id: uuid.NewString(), - Title: query.Search, + Title: "New chat", CreatedTimestamp: util.GetSystemTimestamp(), } c.nonActiveChats = append(c.nonActiveChats, newChat) @@ -308,6 +331,7 @@ func (c *ChatgptPlugin) queryConversation(ctx context.Context, query plugin.Quer { Name: "Delete chat", PreventHideAfterAction: true, + Icon: chatgptDeleteIcon, Action: func(actionContext plugin.ActionContext) { c.deleteChat(ctx, chat.Id, true) c.api.ChangeQuery(ctx, share.ChangedQuery{ @@ -569,3 +593,32 @@ func (c *ChatgptPlugin) deleteChat(ctx context.Context, chatId string, isActive c.saveChats(ctx) } + +func (c *ChatgptPlugin) summaryChatTitle(ctx context.Context, conversations []Conversation) (string, error) { + conversations = append(conversations, Conversation{ + Role: openai.ChatMessageRoleUser, + Text: "Please summary above chat in 20 words, don't give any other unnecessary information, just the summarized title itself.", + }) + + var chatMessages []openai.ChatCompletionMessage + for _, conversation := range conversations { + chatMessages = append(chatMessages, openai.ChatCompletionMessage{ + Role: conversation.Role, + Content: conversation.Text, + }) + } + + resp, err := c.client.CreateChatCompletion( + ctx, + openai.ChatCompletionRequest{ + Model: openai.GPT3Dot5Turbo, + Messages: chatMessages, + }, + ) + if err != nil { + c.api.Log(ctx, fmt.Sprintf("Failed to summary chat title: %s", err.Error())) + return "", err + } + + return resp.Choices[0].Message.Content, nil +} diff --git a/Wox/plugin/system/url.go b/Wox/plugin/system/url.go index f92dea3db..1e96077e7 100644 --- a/Wox/plugin/system/url.go +++ b/Wox/plugin/system/url.go @@ -3,6 +3,7 @@ package system import ( "context" "regexp" + "time" "wox/plugin" "wox/util" ) @@ -51,6 +52,27 @@ func (r *UrlPlugin) Init(ctx context.Context, initParams plugin.InitParams) { func (r *UrlPlugin) Query(ctx context.Context, query plugin.Query) (results []plugin.QueryResult) { if len(r.reg.FindStringIndex(query.Search)) > 0 { + results = append(results, plugin.QueryResult{ + Title: query.Search, + SubTitle: "Open the typed URL from Wox", + Score: 100, + Icon: urlIcon, + RefreshInterval: 100, + OnRefresh: func(result plugin.RefreshableResult) plugin.RefreshableResult { + time.Sleep(time.Second) + result.Title = util.GetSystemTimestampStr() + result.SubTitle = util.GetSystemTimestampStr() + return result + }, + Actions: []plugin.QueryResultAction{ + { + Name: "Open in browser", + Action: func(actionContext plugin.ActionContext) { + util.ShellOpen(query.Search) + }, + }, + }, + }) results = append(results, plugin.QueryResult{ Title: query.Search, SubTitle: "Open the typed URL from Wox",