飞道的博客

使用 .NET MAUI 开发 ChatGPT 客户端

508人阅读  评论(0)

最近 chatgpt 很火,由于网页版本限制了 ip,还得必须开代理,用起来比较麻烦,所以我尝试用 maui 开发一个聊天小应用,结合 chatgpt 的开放 api 来实现(很多客户端使用网页版本接口用 cookie 的方式,有很多限制(如下图)总归不是很正规)。

效果如下

mac 端由于需要升级 macos13 才能开发调试,这部分我还没有完成,不过 maui 的控件是跨平台的,放在后续我升级系统再说。

开发实战

我是设想开发一个类似 jetbrains 的 ToolBox 应用一样,启动程序在桌面右下角出现托盘图标,点击图标弹出应用(风格在 windows mac 平台保持一致)

需要实现的功能一览

  • 托盘图标(右键点击有 menu)
  • webview(js 和 csharp 互相调用)
  • 聊天 SPA 页面(react 开发,build 后让 webview 展示)

新建一个 maui 工程(vs2022)

坑一:默认编译出来的 exe 是直接双击打不开的

工程文件加上这个配置


  
  1. <WindowsPackageType>None </WindowsPackageType>
  2. <WindowsAppSDKSelfContained Condition="'$(IsUnpackaged)' == 'true'">true </WindowsAppSDKSelfContained>
  3. <SelfContained Condition="'$(IsUnpackaged)' == 'true'">true </SelfContained>

以上修改后,编译出来的 exe 双击就可以打开了

托盘图标(右键点击有 menu)

启动时设置窗口不能改变大小,隐藏 titlebar, 让 Webview 控件占满整个窗口

这里要根据平台不同实现不同了,windows 平台采用 winAPI 调用,具体看工程代码吧!

WebView

在 MainPage.xaml 添加控件

对应的静态 html 等文件放在工程的 Resource\Raw 文件夹下 (整个文件夹里面默认是作为内嵌资源打包的,工程文件里面的如下配置起的作用)


  
  1. <!-- Raw Assets (also remove the "Resources\Raw" prefix) -->
  2. <MauiAsset Include="Resources\Raw\**" LogicalName="%(RecursiveDir)%(Filename)%(Extension)" />

【重点】js 和 csharp 互相调用

这部分我找了很多资料,最终参考了这个 demo,然后改进了下。

主要原理是:

  • js 调用 csharp 方法前先把数据存储在 localstorage 里
  • 然后 windows.location 切换特定的 url 发起调用,返回一个 promise,等待 csharp 的事件
  • csharp 端监听 webview 的 Navigating 事件,异步进行下面处理
  • 根据 url 解析出来 localstorage 的 key
  • 然后 csharp 端调用 excutescript 根据 key 拿到 localstorage 的 value
  • 进行逻辑处理后返回通过事件分发到 js 端

js 的调用封装如下:


  
  1. // 调用csharp的方法封装
  2. export default class CsharpMethod {
  3. constructor( command, data) {
  4. this. RequestPrefix = "request_csharp_";
  5. this. ResponsePrefix = "response_csharp_";
  6. // 唯一
  7. this. dataId = this. RequestPrefix + new Date(). getTime();
  8. // 调用csharp的命令
  9. this. command = command;
  10. // 参数
  11. this. data = { command: command, data: !data ? '' : JSON. stringify(data), key: this. dataId }
  12. }
  13. // 调用csharp 返回promise
  14. call( ) {
  15. // 把data存储到localstorage中 目的是让csharp端获取参数
  16. localStorage. setItem( this. dataId, this. utf8_to_b64( JSON. stringify( this. data)));
  17. let eventKey = this. dataId. replace( this. RequestPrefix, this. ResponsePrefix);
  18. let that = this;
  19. const promise = new Promise( function ( resolve, reject) {
  20. const eventHandler = function ( e) {
  21. window. removeEventListener(eventKey, eventHandler);
  22. let resp = e. newValue;
  23. if (resp) {
  24. // 从base64转换
  25. let realData = that. b64_to_utf8(resp);
  26. if (realData. startsWith( 'err:')) {
  27. reject(realData. substr( 4));
  28. } else {
  29. resolve(realData);
  30. }
  31. } else {
  32. reject( "unknown error :" + eventKey);
  33. }
  34. };
  35. // 注册监听回调(csharp端处理完发起的)
  36. window. addEventListener(eventKey, eventHandler);
  37. });
  38. // 改变location 发送给csharp端
  39. window. location = "/api/" + this. dataId;
  40. return promise;
  41. }
  42. // 转成base64 解决中文乱码
  43. utf8_to_b64( str) {
  44. return window. btoa( unescape( encodeURIComponent(str)));
  45. }
  46. // 从base64转过来 解决中文乱码
  47. b64_to_utf8( str) {
  48. return decodeURIComponent( escape( window. atob(str)));
  49. }
  50. }

前端的使用方式


  
  1. import CsharpMethod from '../../services/api'
  2. // 发起调用csharp的chat事件函数
  3. const method = new CsharpMethod( "chat", { msg: message});
  4. method. call() // call返回promise
  5. . then( data =>{
  6. // 拿到csharp端的返回后展示
  7. onMessageHandler({
  8. message: data,
  9. username: 'Robot',
  10. type: 'chat_message'
  11. });
  12. }). catch( err => {
  13. alert(err);
  14. });

csharp 端的处理:

这么封装后,js 和 csharp 的互相调用就很方便了。

chatgpt 的开放 api 调用

注册好 chatgpt 后可以申请一个 APIKEY。

API 封装:


  
  1. public static async Task<CompletionsResponse> GetResponseDataAsync(string prompt)
  2. {
  3. // Set up the API URL and API key
  4. string apiUrl = "https://api.openai.com/v1/completions";
  5. // Get the request body JSON
  6. decimal temperature = decimal.Parse(Setting.Temperature, CultureInfo.InvariantCulture);
  7. int maxTokens = int.Parse(Setting.MaxTokens, CultureInfo.InvariantCulture);
  8. string requestBodyJson = GetRequestBodyJson(prompt, temperature, maxTokens);
  9. // Send the API request and get the response data
  10. return await SendApiRequestAsync(apiUrl, Setting.ApiKey, requestBodyJson);
  11. }
  12. private static string GetRequestBodyJson(string prompt, decimal temperature, int maxTokens)
  13. {
  14. // Set up the request body
  15. var requestBody = new CompletionsRequestBody
  16. {
  17. Model = "text-davinci-003",
  18. Prompt = prompt,
  19. Temperature = temperature,
  20. MaxTokens = maxTokens,
  21. TopP = 1.0m,
  22. FrequencyPenalty = 0.0m,
  23. PresencePenalty = 0.0m,
  24. N = 1,
  25. Stop = "[END]",
  26. };
  27. // Create a new JsonSerializerOptions object with the IgnoreNullValues and IgnoreReadOnlyProperties properties set to true
  28. var serializerOptions = new JsonSerializerOptions
  29. {
  30. IgnoreNullValues = true,
  31. IgnoreReadOnlyProperties = true,
  32. };
  33. // Serialize the request body to JSON using the JsonSerializer.Serialize method overload that takes a JsonSerializerOptions parameter
  34. return JsonSerializer.Serialize(requestBody, serializerOptions);
  35. }
  36. private static async Task<CompletionsResponse> SendApiRequestAsync(string apiUrl, string apiKey, string requestBodyJson)
  37. {
  38. // Create a new HttpClient for making the API request
  39. using HttpClient client = new HttpClient();
  40. // Set the API key in the request headers
  41. client.DefaultRequestHeaders.Add( "Authorization", "Bearer " + apiKey);
  42. // Create a new StringContent object with the JSON payload and the correct content type
  43. StringContent content = new StringContent(requestBodyJson, Encoding.UTF8, "application/json");
  44. // Send the API request and get the response
  45. HttpResponseMessage response = await client.PostAsync(apiUrl, content);
  46. // Deserialize the response
  47. var responseBody = await response.Content.ReadAsStringAsync();
  48. // Return the response data
  49. return JsonSerializer.Deserialize<CompletionsResponse>(responseBody);
  50. }

调用方式

 var reply = await ChatService.GetResponseDataAsync('xxxxxxxxxx');

完整代码参考~

在学习 maui 的过程中,遇到问题我在 Microsoft Learn 提问,回答的效率很快,推荐大家试试看!

点我了解更多 MAUI 相关资料~


转载:https://blog.csdn.net/helendemeng/article/details/128386667
查看评论
* 以上用户言论只代表其个人观点,不代表本网站的观点或立场