电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

使用 .NET MAUI 開發 ChatGPT 客戶端的流程

瀏覽:118日期:2022-06-09 08:58:56
目錄
  • 開發實戰
    • 托盤圖標(右鍵點擊有 menu)
    • WebView
    • 【重點】js 和 csharp 互相調用
    • chatgpt 的開放 api 調用

最近 chatgpt 很火,由于網頁版本限制了 ip,還得必須開代理,用起來比較麻煩,所以我嘗試用 maui 開發一個聊天小應用,結合 chatgpt 的開放 api 來實現(很多客戶端使用網頁版本接口用 cookie 的方式,有很多限制(如下圖)總歸不是很正規)。

效果如下

mac 端由于需要升級 macos13 才能開發調試,這部分我還沒有完成,不過 maui 的控件是跨平臺的,放在后續我升級系統再說。

開發實戰

我是設想開發一個類似 jetbrains 的 ToolBox 應用一樣,啟動程序在桌面右下角出現托盤圖標,點擊圖標彈出應用(風格在 windows mac 平臺保持一致)

需要實現的功能一覽

  • 托盤圖標(右鍵點擊有 menu)
  • webview(js 和 csharp 互相調用)
  • 聊天 SPA 頁面(react 開發,build 后讓 webview 展示)

新建一個 maui 工程(vs2022)

坑一:默認編譯出來的 exe 是直接雙擊打不開的

工程文件加上這個配置

<WindowsPackageType>None</WindowsPackageType><WindowsAppSDKSelfContained Condition=""$(IsUnpackaged)" == "true"">true</WindowsAppSDKSelfContained><SelfContained Condition=""$(IsUnpackaged)" == "true"">true</SelfContained>

以上修改后,編譯出來的 exe 雙擊就可以打開了

托盤圖標(右鍵點擊有 menu)

啟動時設置窗口不能改變大小,隱藏 titlebar, 讓 Webview 控件占滿整個窗口

這里要根據平臺不同實現不同了,windows 平臺采用 winAPI 調用,具體看工程代碼吧!

WebView

在 MainPage.xaml 添加控件

對應的靜態 html 等文件放在工程的 Resource\Raw 文件夾下 (整個文件夾里面默認是作為內嵌資源打包的,工程文件里面的如下配置起的作用)

<!-- Raw Assets (also remove the "Resources\Raw" prefix) --><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 的調用封裝如下:

// 調用csharp的方法封裝export default class CsharpMethod {  constructor(command, data) {    this.RequestPrefix = "request_csharp_";    this.ResponsePrefix = "response_csharp_";    // 唯一    this.dataId = this.RequestPrefix + new Date().getTime();    // 調用csharp的命令    this.command = command;    // 參數    this.data = { command: command, data: !data ? "" : JSON.stringify(data), key: this.dataId }  }   // 調用csharp 返回promise  call() {    // 把data存儲到localstorage中 目的是讓csharp端獲取參數    localStorage.setItem(this.dataId, this.utf8_to_b64(JSON.stringify(this.data)));    let eventKey = this.dataId.replace(this.RequestPrefix, this.ResponsePrefix);    let that = this;    const promise = new Promise(function (resolve, reject) {      const eventHandler = function (e) {window.removeEventListener(eventKey, eventHandler);let resp = e.newValue;if (resp) {  // 從base64轉換  let realData = that.b64_to_utf8(resp);  if (realData.startsWith("err:")) {    reject(realData.substr(4));  } else {    resolve(realData);  }} else {  reject("unknown error :" + eventKey);}      };      // 注冊監聽回調(csharp端處理完發起的)      window.addEventListener(eventKey, eventHandler);    });    // 改變location 發送給csharp端    window.location = "/api/" + this.dataId;    return promise;  }   // 轉成base64 解決中文亂碼  utf8_to_b64(str) {    return window.btoa(unescape(encodeURIComponent(str)));  }  // 從base64轉過來 解決中文亂碼  b64_to_utf8(str) {    return decodeURIComponent(escape(window.atob(str)));  } }

前端的使用方式

import CsharpMethod from "../../services/api" // 發起調用csharp的chat事件函數const method = new CsharpMethod("chat", {msg: message});method.call() // call返回promise.then(data =>{  // 拿到csharp端的返回后展示  onMessageHandler({    message: data,    username: "Robot",    type: "chat_message"  });}).catch(err =>  {    alert(err);});

csharp 端的處理:

這么封裝后,js 和 csharp 的互相調用就很方便了。

chatgpt 的開放 api 調用

注冊好 chatgpt 后可以申請一個 APIKEY。

API 封裝:

  public static async Task<CompletionsResponse> GetResponseDataAsync(string prompt){    // Set up the API URL and API key    string apiUrl = "https://api.openai.com/v1/completions";     // Get the request body JSON    decimal temperature = decimal.Parse(Setting.Temperature, CultureInfo.InvariantCulture);    int maxTokens = int.Parse(Setting.MaxTokens, CultureInfo.InvariantCulture);    string requestBodyJson = GetRequestBodyJson(prompt, temperature, maxTokens);     // Send the API request and get the response data    return await SendApiRequestAsync(apiUrl, Setting.ApiKey, requestBodyJson);} private static string GetRequestBodyJson(string prompt, decimal temperature, int maxTokens){    // Set up the request body    var requestBody = new CompletionsRequestBody    {Model = "text-davinci-003",Prompt = prompt,Temperature = temperature,MaxTokens = maxTokens,TopP = 1.0m,FrequencyPenalty = 0.0m,PresencePenalty = 0.0m,N = 1,Stop = "[END]",    };     // Create a new JsonSerializerOptions object with the IgnoreNullValues and IgnoreReadOnlyProperties properties set to true    var serializerOptions = new JsonSerializerOptions    {IgnoreNullValues = true,IgnoreReadOnlyProperties = true,    };     // Serialize the request body to JSON using the JsonSerializer.Serialize method overload that takes a JsonSerializerOptions parameter    return JsonSerializer.Serialize(requestBody, serializerOptions);} private static async Task<CompletionsResponse> SendApiRequestAsync(string apiUrl, string apiKey, string requestBodyJson){    // Create a new HttpClient for making the API request    using HttpClient client = new HttpClient();     // Set the API key in the request headers    client.DefaultRequestHeaders.Add("Authorization", "Bearer " + apiKey);     // Create a new StringContent object with the JSON payload and the correct content type    StringContent content = new StringContent(requestBodyJson, Encoding.UTF8, "application/json");     // Send the API request and get the response    HttpResponseMessage response = await client.PostAsync(apiUrl, content);     // Deserialize the response    var responseBody = await response.Content.ReadAsStringAsync();     // Return the response data    return JsonSerializer.Deserialize<CompletionsResponse>(responseBody);}

調用方式

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

在學習 maui 的過程中,遇到問題我在 Microsoft Learn 提問,回答的效率很快,推薦大家試試看!

到此這篇關于使用 .NET MAUI 開發 ChatGPT 客戶端的文章就介紹到這了,更多相關.NET MAUI 開發 ChatGPT 內容請搜索以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持!

標簽: ASP.NET
主站蜘蛛池模板: 半容积式换热器_北京浮动盘管换热器厂家|北京亿丰上达 | 超声骨密度仪-动脉硬化检测仪器-人体成分分析仪厂家/品牌/价格_南京科力悦 | 双齿辊破碎机-大型狼牙破碎机视频-对辊破碎机价格/型号图片-金联机械设备生产厂家 | 铁素体测量仪/检测仪/铁素体含量测试仪-苏州圣光仪器有限公司 | 高博医疗集团上海阿特蒙医院 | 软膜天花_软膜灯箱_首选乐创品牌_一站式天花软膜材料供应商! | 优秀的临床医学知识库,临床知识库,医疗知识库,满足电子病历四级要求,免费试用 | 净水器代理,净水器招商,净水器加盟-FineSky德国法兹全屋净水 | 安全光栅|射频导纳物位开关|音叉料位计|雷达液位计|两级跑偏开关|双向拉绳开关-山东卓信机械有限公司 | 冲锋衣滑雪服厂家-冲锋衣定制工厂-滑雪服加工厂-广东睿牛户外(S-GERT) | 科研ELISA试剂盒,酶联免疫检测试剂盒,昆虫_植物ELISA酶免试剂盒-上海仁捷生物科技有限公司 | 土壤墒情监测站_土壤墒情监测仪_土壤墒情监测系统_管式土壤墒情站-山东风途物联网 | 高通量组织研磨仪-多样品组织研磨仪-全自动组织研磨仪-研磨者科技(广州)有限公司 | 培训中心-翰香原香酥板栗饼加盟店总部-正宗板栗酥饼技术 | 中央空调维修、中央空调保养、螺杆压缩机维修-苏州东菱空调 | 苏州防水公司_厂房屋面外墙防水_地下室卫生间防水堵漏-苏州伊诺尔防水工程有限公司 | 搜木网 - 木业全产业链交易平台,免费搜货、低价买货! | 吉祥新世纪铝塑板_生产铝塑板厂家_铝塑板生产厂家_临沂市兴达铝塑装饰材料有限公司 | 喷播机厂家_二手喷播机租赁_水泥浆洒布机-河南青山绿水机电设备有限公司 | 广州办公室设计,办公室装修,写字楼设计,办公室装修公司_德科 | 神超官网_焊接圆锯片_高速钢锯片_硬质合金锯片_浙江神超锯业制造有限公司 | 鲁网 - 山东省重点新闻网站,山东第一财经门户| 冻干机(冷冻干燥机)_小型|实验型|食品真空冷冻干燥机-松源 | ET3000双钳形接地电阻测试仪_ZSR10A直流_SXJS-IV智能_SX-9000全自动油介质损耗测试仪-上海康登 | 工程管道/塑料管材/pvc排水管/ppr给水管/pe双壁波纹管等品牌管材批发厂家-河南洁尔康建材 | 热镀锌槽钢|角钢|工字钢|圆钢|H型钢|扁钢|花纹板-天津千百顺钢铁贸易有限公司 | 挨踢网-大家的导航!| 防伪溯源|防窜货|微信二维码营销|兆信_行业内领先的防伪防窜货数字化营销解决方案供应商 | 不锈钢螺丝,不锈钢螺栓,不锈钢标准件-江苏百德特种合金有限公司 交变/复合盐雾试验箱-高低温冲击试验箱_安奈设备产品供应杭州/江苏南京/安徽马鞍山合肥等全国各地 | 洁净化验室净化工程_成都实验室装修设计施工_四川华锐净化公司 | 压力变送器-上海武锐自动化设备有限公司 | 智能垃圾箱|垃圾房|垃圾分类亭|垃圾分类箱专业生产厂家定做-宿迁市传宇环保设备有限公司 | UV固化机_UVLED光固化机_UV干燥机生产厂家-上海冠顶公司专业生产UV固化机设备 | 德国UST优斯特氢气检漏仪-德国舒赐乙烷检测仪-北京泽钏 | 真空泵维修保养,普发,阿尔卡特,荏原,卡西亚玛,莱宝,爱德华干式螺杆真空泵维修-东莞比其尔真空机电设备有限公司 | 河南道路标志牌_交通路标牌_交通标志牌厂家-郑州路畅交通 | 细石混凝土泵_厂家_价格-烟台九达机械有限公司 | 博博会2021_中国博物馆及相关产品与技术博览会【博博会】 | 心肺复苏模拟人|医学模型|急救护理模型|医学教学模型上海康人医学仪器设备有限公司 | 天然气分析仪-液化气二甲醚分析仪|传昊仪器 | 龙门加工中心-数控龙门加工中心厂家价格-山东海特数控机床有限公司_龙门加工中心-数控龙门加工中心厂家价格-山东海特数控机床有限公司 |