csdn1233 3 months ago
parent
commit
9997dfae4d

+ 16 - 0
AIart-app/src/app/app.routes.ts

@@ -5,4 +5,20 @@ export const routes: Routes = [
     path: '',
     loadChildren: () => import('./tabs/tabs.routes').then((m) => m.routes),
   },
+  // 聊天模块
+  {
+    path: "chat/session/role/:roleId",
+    loadComponent: () => import('./test-chat-panel/test-chat-panel.component').then(m => m.TestChatPanelComponent),
+    runGuardsAndResolvers: "always",
+  },
+  {
+    path: "chat/session/chat/:chatId",
+    loadComponent: () => import('./test-chat-panel/test-chat-panel.component').then(m => m.TestChatPanelComponent),
+    runGuardsAndResolvers: "always",
+  },
+  {
+    path: 'chat/pro/chat/:chatId',
+    redirectTo: '/chat/session/chat/:chatId',
+    pathMatch: 'full'
+  },
 ];

+ 9 - 7
AIart-app/src/app/tab4/tab4.page.html

@@ -1,17 +1,19 @@
 <ion-header [translucent]="true">
   <ion-toolbar>
     <ion-title>
-      tab 4
+      Chat模块组件演示
     </ion-title>
   </ion-toolbar>
 </ion-header>
 
+<!-- {{title}} -->
 <ion-content [fullscreen]="true">
-  <ion-header collapse="condense">
-    <ion-toolbar>
-      <ion-title size="large">tab 4</ion-title>
-    </ion-toolbar>
-  </ion-header>
+  <h1>页面:配置路由和参数的聊天页</h1>
+  <ion-button (click)="goChat()">开始页面聊天</ion-button>
+  <h1>组件:直接弹出的聊天组件</h1>
+  <ion-button (click)="openChat()">开始新聊天</ion-button>
+  <ion-button (click)="restoreChat('yHEHqMQDNv')">恢复会话</ion-button>
+  <h1>示例:门诊问诊的智能体示例(ChatPanel组件)</h1>
+  <ion-button (click)="openInquiry()">进入门诊</ion-button>
 
-  <app-explore-container name="tab 4 page"></app-explore-container>
 </ion-content>

+ 116 - 3
AIart-app/src/app/tab4/tab4.page.ts

@@ -1,14 +1,127 @@
 import { Component } from '@angular/core';
-import { IonHeader, IonToolbar, IonTitle, IonContent } from '@ionic/angular/standalone';
+import { IonHeader, IonToolbar, IonTitle, IonContent, ModalController, IonButton } from '@ionic/angular/standalone';
 import { ExploreContainerComponent } from '../explore-container/explore-container.component';
+import { Router } from '@angular/router';
+import { ChatPanelOptions, FmChatModalInput, FmodeChat, FmodeChatMessage, openChatPanelModal } from 'fmode-ng';
 
 @Component({
   selector: 'app-tab4',
   templateUrl: 'tab4.page.html',
   styleUrls: ['tab4.page.scss'],
   standalone: true,
-  imports: [IonHeader, IonToolbar, IonTitle, IonContent, ExploreContainerComponent],
+  imports: [IonHeader, IonToolbar, IonTitle, IonContent, ExploreContainerComponent, IonButton, FmChatModalInput,],
 })
 export class tab4Page {
-  constructor() { }
+
+  constructor(
+    private modalCtrl: ModalController,
+    private router: Router,
+  ) {
+
+  }
+  title: string = "123"
+  /** 示例:问诊ChatPanel面板 */
+  openInquiry() {
+    localStorage.setItem("company", "E4KpGvTEto")
+    let options: ChatPanelOptions = {
+      roleId: "2DXJkRsjXK",
+      onChatInit: (chat: FmodeChat) => {
+        console.log("onChatInit");
+        console.log("预设角色", chat.role);
+        chat.role.set("name", "晓晓");
+        chat.role.set("title", "全科医生");
+        chat.role.set("desc", "一名亲切和蔼的门诊全科主任医生,晓晓,年龄36岁");
+        chat.role.set("tags", ["全科", "门诊"]);
+        chat.role.set("avatar", "https://nova-cloud.obs.cn-south-1.myhuaweicloud.com/storage/aigc/imagine/Q4Zif7fTbK-0.png")
+        chat.role.set("prompt", `
+# 角色设定
+您是一名亲切和蔼的专业的全科医生,晓晓,年龄36岁,需要完成一次完整的门诊服务。
+
+# 对话环节
+0.导诊(根据用户基本情况,引导挂号合适的科室)
+1.预设的问询方式(感冒问呼吸、肚子疼叩诊)
+- 打招呼,以用户自述为主
+- 当信息充足时候,确认用户症状对应的科室,并进入下一个环节
+2.拓展的问询细节
+例如:用户反映呼吸不畅,拓展出:是否咳嗽;是否感觉痛或者痒等其他需要的问题。
+- 当问询细节补充完成后进入下一个环节
+3.初步的诊断结果,并且同时列出检查检验项目
+初步诊断:确定需要有哪些进一步检查
+检查检验:获取医学客观数据
+- 等待用户提交客观数据,进入下一阶段
+4.给出诊断方案并给出处方
+- 完成处方时,请在消息结尾附带: [完成]
+
+# 开始话语
+当您准备好了,可以以一个医生的身份,向来访的用户打招呼。`);
+      },
+      onMessage: (chat: FmodeChat, message: FmodeChatMessage) => {
+        console.log("onMessage", message)
+        let content: any = message?.content
+        if (typeof content == "string") {
+          if (content?.indexOf("[完成]") > -1) {
+            console.log("门诊已完成")
+          }
+        }
+      },
+      onChatSaved: (chat: FmodeChat) => {
+        // chat?.chatSession?.id 本次会话的 chatId
+        console.log("onChatSaved", chat, chat?.chatSession, chat?.chatSession?.id)
+      }
+    }
+    openChatPanelModal(this.modalCtrl, options)
+  }
+
+  /**
+   * 开始聊天
+   */
+  openChat() {
+    let options: ChatPanelOptions = {
+      roleId: "2DXJkRsjXK",
+      onChatSaved: (chat: FmodeChat) => {
+        // chat?.chatSession?.id 本次会话的 chatId
+        console.log("onChatSaved", chat, chat?.chatSession, chat?.chatSession?.id)
+      },
+    }
+    openChatPanelModal(this.modalCtrl, options)
+  }
+  /**
+   * 恢复聊天
+   * @chatId 从onChatSaved生命周期中,获取chat?.chatSession?.id
+   */
+  restoreChat(chatId: string) {
+    let options: ChatPanelOptions = {
+      roleId: "2DXJkRsjXK",
+      chatId: chatId
+    }
+    openChatPanelModal(this.modalCtrl, options)
+  }
+
+  goChat() {
+    this.router.navigateByUrl("/chat/session/role/2DXJkRsjXK")
+  }
+
+
+  // audioModalHeightPoint:number = 0.35;
+  // async startTalk(){
+  //   // 根据手机兼容性,适配组件弹出高度
+  //   let height = document.body.clientHeight || 960;
+  //   this.audioModalHeightPoint = Number((165/height).toFixed(2));
+
+  //   // 弹出组件
+  //   let modal:any
+  //   let chat:any
+  //   modal = await this.modalCtrl.create({
+  //     component:ModalAudioMessageComponent,
+  //     componentProps:{
+  //       chat:chat,
+  //       modal:modal,
+  //       onBreakPointSet:()=>{
+  //         modal?.setCurrentBreakpoint(this.audioModalHeightPoint)
+  //       }
+  //     }
+  //   })
+  //   modal.present();
+  // }
+
 }

+ 3 - 0
AIart-app/src/app/test-chat-panel/test-chat-panel.component.html

@@ -0,0 +1,3 @@
+<app-chat-panel *ngIf="leftButtons?.length&&modelList?.length" #chatComp [roleId]="roleId" [chatId]="chatId"
+  [leftButtons]="leftButtons" [modelList]="modelList" [isDirect]="isDirect" [hideModalSelect]="hideModalSelect"
+  [hideInputPreview]="hideInputPreview"></app-chat-panel>

+ 3 - 0
AIart-app/src/app/test-chat-panel/test-chat-panel.component.scss

@@ -0,0 +1,3 @@
+app-chat-panel {
+    height: 100vh;
+}

+ 22 - 0
AIart-app/src/app/test-chat-panel/test-chat-panel.component.spec.ts

@@ -0,0 +1,22 @@
+import { ComponentFixture, TestBed, waitForAsync } from '@angular/core/testing';
+
+import { TestChatPanelComponent } from './test-chat-panel.component';
+
+describe('TestChatPanelComponent', () => {
+  let component: TestChatPanelComponent;
+  let fixture: ComponentFixture<TestChatPanelComponent>;
+
+  beforeEach(waitForAsync(() => {
+    TestBed.configureTestingModule({
+      imports: [TestChatPanelComponent],
+    }).compileComponents();
+
+    fixture = TestBed.createComponent(TestChatPanelComponent);
+    component = fixture.componentInstance;
+    fixture.detectChanges();
+  }));
+
+  it('should create', () => {
+    expect(component).toBeTruthy();
+  });
+});

+ 238 - 0
AIart-app/src/app/test-chat-panel/test-chat-panel.component.ts

@@ -0,0 +1,238 @@
+import { CommonModule } from '@angular/common';
+import { ChangeDetectorRef, Component, OnInit, ViewChild } from '@angular/core';
+import { ActivatedRoute } from '@angular/router';
+import { ModalController } from '@ionic/angular';
+import { ChatPanelComponent } from 'fmode-ng';
+import { combineLatest } from 'rxjs';
+import Parse from "parse"
+
+@Component({
+  selector: 'app-chat-panel',
+  templateUrl: './test-chat-panel.component.html',
+  styleUrls: ['./test-chat-panel.component.scss'],
+  standalone: true,
+  imports: [
+    CommonModule,
+    ChatPanelComponent,
+  ]
+})
+export class TestChatPanelComponent implements OnInit {
+  @ViewChild(ChatPanelComponent) chatComp: ChatPanelComponent | undefined
+  leftButtons: any[] = []
+  modelList: any[] = []
+  isDirect: boolean = true;
+  hideShare: boolean = true;
+  hideModalSelect: boolean = true;
+  hideInputPreview: boolean = true;
+  chatId: string = ""
+  roleId: string = ""
+  pid: string = ""
+  constructor(
+    private route: ActivatedRoute,
+    private cdRef: ChangeDetectorRef,
+    private modalCtrl: ModalController
+  ) {
+    combineLatest([this.route.params, this.route.queryParams]).subscribe(async (data: any) => {
+      let params = data[0] || {}
+
+      this.chatId = params['chatId'] || this.chatId || null;
+      this.roleId = params['roleId'] || this.roleId || null;
+      this.pid = params['pid'] || this.pid || null;
+      console.log("this.pid", this.pid)
+      // 异步加载的后续数据 操作按钮
+      let bint = setInterval(() => {
+        if (this.roleId) {
+          clearInterval(bint);
+          return
+        }
+        this.initPanelConfig();
+      }, 2000);
+    })
+  }
+
+
+  ngOnInit() {
+    this.initPanelConfig();
+    // 异步加载的后续数据 提示词
+    let pint = setInterval(() => {
+      if (this.chatComp?.fmodeChat?.promptList?.length) {
+        clearInterval(pint);
+        return
+      }
+      this.getChatPrompt();
+    }, 2000);
+
+    // 异步加载的后续数据 采访人物 ChatSession.person
+    let personInt = setInterval(() => {
+      if (this.chatComp?.fmodeChat?.chatSession?.get("person")) {
+        clearInterval(personInt)
+      }
+      if (!this.chatComp?.fmodeChat?.chatSession?.get("person")) {
+        if (this.pid) {
+          this.chatComp?.fmodeChat?.chatSession?.set("person", { type: "Pointer", className: "Person", objectId: this.pid })
+        }
+      }
+    }, 2000);
+  }
+
+  // 初始化聊天面板的设置
+  initPanelConfig() {
+    this.roleId = this.chatComp?.fmodeChat?.chatSession?.get("role")?.id || this.roleId;
+
+    // 按钮自定义
+    this.leftButtons = [
+      // 提示 当角色配置预设提示词时 显示
+      {
+        title: "话题灵感",
+        showTitle: true,
+        icon: "color-wand-outline",
+        onClick: () => {
+          if (this.chatComp) {
+            this.chatComp.fmodeChat.isPromptModalOpen = true
+          }
+        },
+        show: () => {
+          return this.chatComp?.fmodeChat?.promptList?.length
+        }
+      }
+    ]
+
+    this.leftButtons.push({ // 总结 结束并归档本次对话
+      title: "AI总结对话",
+      showTitle: true,
+      icon: "archive-outline",
+      onClick: () => {
+        if (this.chatComp) {
+          // this.chatComp.fmodeChat.isPromptModalOpen = true
+          if (this.chatComp.fmodeChat) {
+            console.log(JSON.stringify(this.chatComp.fmodeChat.messageList))
+            // alert("处理对话记录")
+          }
+        }
+      },
+      show: () => {
+        return !this.chatComp?.fmodeChat?.chatSession?.get("story")?.id
+      }
+    })
+
+    this.leftButtons.push({ // 总结 结束并归档本次对话
+      title: "聊天心理分析",
+      showTitle: true,
+      icon: "archive-outline",
+      onClick: () => {
+        if (this.chatComp) {
+          // this.chatComp.fmodeChat.isPromptModalOpen = true
+          if (this.chatComp.fmodeChat) {
+            let messageList = JSON.parse(JSON.stringify(this.chatComp.fmodeChat.messageList))
+            messageList = messageList.filter((item: any) => item.role != "system" && item?.hidden != true)
+            let qaContent = messageList.map((item: any) => {
+              let roleName = "当前用户"
+              if (item.role != "user") {
+                if (this.chatComp && this.chatComp.fmodeChat.role) {
+                  roleName = this.chatComp.fmodeChat.role.get("name");
+                } else {
+                  roleName = "AI助理"
+                }
+              }
+              return `${roleName}:${item.content}`
+            }
+            ).join("\n")
+            console.log(qaContent)
+            // alert("处理对话记录")
+          }
+        }
+      },
+      show: () => {
+        return !this.chatComp?.fmodeChat?.chatSession?.get("story")?.id
+      }
+    })
+
+
+
+    setTimeout(() => {
+      if (this.chatComp && this.chatComp.fmodeChat) {
+        // 自定义左下角操作按钮
+        console.log("左下角操作按钮", this.chatComp.fmodeChat.leftButtons);
+        this.chatComp.fmodeChat.leftButtons = this.leftButtons;
+
+        // 自定义角色名称
+        console.log("自定义角色", this.chatComp.fmodeChat.role);
+        this.chatComp.fmodeChat.role.set("name", "晓晓");
+        this.chatComp.fmodeChat.role.set("title", "心理咨询师");
+        this.chatComp.fmodeChat.role.set("desc", "一名亲切和蔼的心理咨询师,晓晓,年龄36岁");
+        this.chatComp.fmodeChat.role.set("tags", ["焦虑", "抑郁"]);
+        this.chatComp.fmodeChat.role.set("avatar", "https://nova-cloud.obs.cn-south-1.myhuaweicloud.com/storage/aigc/imagine/Q4Zif7fTbK-0.png")
+        this.chatComp.fmodeChat.role.set("prompt", `
+# 角色设定
+您是一名亲切和蔼的心理咨询师,晓晓,年龄36岁,需要完成陪来访者聊聊天,随意轻松一些。
+
+# 对话环节
+0.破冰,互相了解,引导用户介绍自己
+1.拓展话题,根据用户的介绍,拓展一些和其心理状态相关的话题
+- 引导,可深入的点,以用户自述为主
+- 当信息充足时候,确认用户心理状态,并进入下一个环节
+2.引导收尾,委婉引导用户结束本次对话
+- 用户同意结束后,结束本次对话,如果依依不舍,可以再陪聊一会儿`);
+        //             this.chatComp.fmodeChat.role.set("name","晓晓");
+        //             this.chatComp.fmodeChat.role.set("title","主任医师");
+        //             this.chatComp.fmodeChat.role.set("desc","一名专业的全科医生,晓晓,年龄36岁");
+        //             this.chatComp.fmodeChat.role.set("tags",["呼吸道","感染科"]);
+        //             this.chatComp.fmodeChat.role.set("avatar","https://nova-cloud.obs.cn-south-1.myhuaweicloud.com/storage/aigc/imagine/Q4Zif7fTbK-0.png")
+        //             this.chatComp.fmodeChat.role.set("prompt",`
+        // # 角色设定
+        // 您是一名专业的全科医生,晓晓,年龄36岁,需要完成一次完整的门诊服务。
+
+        // # 对话环节
+        // 0.导诊(根据用户基本情况,引导挂号合适的科室)
+        // 1.预设的问询方式(感冒问呼吸、肚子疼叩诊)
+        // - 打招呼,以用户自述为主
+        // - 当信息充足时候,确认用户症状对应的科室,并进入下一个环节
+        // 2.拓展的问询细节
+        // 例如:用户反映呼吸不畅,拓展出:是否咳嗽;是否感觉痛或者痒等其他需要的问题。
+        // - 当问询细节补充完成后进入下一个环节
+        // 3.初步的诊断结果,并且同时列出检查检验项目
+        // 初步诊断:确定需要有哪些进一步检查
+        // 检查检验:获取医学客观数据
+        // - 等待用户提交客观数据,进入下一阶段
+        // 4.给出诊断方案并给出处方
+
+        // # 开始话语
+        // 当您准备好了,可以以一个医生的身份,向来访的用户打招呼。
+        //             `);
+
+        this.cdRef.detectChanges();
+      }
+    }, 1000)
+
+
+    // 模型自定义
+    let ChatModel = Parse.Object.extend("ChatModel");
+    let model1 = new ChatModel();
+    model1.set({
+      name: "语伴4.5-128k",
+      code: "fmode-4.5-128k",
+      model: "gpt-4o-mini",
+      credit: 0.096,
+    })
+    this.modelList = [model1]
+
+
+    console.log("initPanelConfig", this.leftButtons, this.modelList)
+  }
+
+  async getChatPrompt() {
+    let query = new Parse.Query('ChatPrompt')
+    query.notEqualTo('isDeleted', true)
+    //  query.equalTo('company', localStorage.getItem("company"))
+    query.equalTo('role', this.chatComp?.fmodeChat?.role)
+    query.include('role')
+    let promptData = await query.find()
+    if (this.chatComp && this.chatComp.fmodeChat) {
+      this.chatComp.fmodeChat.promptList = promptData
+      this.chatComp.fmodeChat.promptList.forEach((item: any) => {
+        let cate = item.get('role').get('promptCates').filter((cate: any) => cate.name == item.get('cate'))
+        item.img = cate[0].img
+      })
+    }
+  }
+}