Browse Source

feat: test-chat-panel

cainiao-hue 3 tháng trước cách đây
mục cha
commit
800759a433

+ 16 - 0
soul-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'
+  },
 ];

+ 0 - 4
soul-app/src/app/tab2/README.md

@@ -8,10 +8,6 @@ Tab2 页面结构
 匹配结果展示区域
 示例内容:
 说明文本: “根据您的需求,智能匹配合适的心理咨询师。”
-选择项:
-选项一: “焦虑”
-选项二: “抑郁”
-选项三: “人际关系”
 按钮: “开始匹配”
 匹配结果: “为您推荐的心理咨询师是:张医生,擅长焦虑和抑郁。”
 2. 陪聊服务区

+ 5 - 6
soul-app/src/app/tab2/tab2.page.html

@@ -1,6 +1,6 @@
 <ion-header>
   <ion-toolbar>
-    <ion-title>心理健康服务</ion-title>
+    <ion-title>心理陪聊服务</ion-title>
   </ion-toolbar>
 </ion-header>
 
@@ -13,8 +13,7 @@
       </ion-card-header>
       <ion-card-content>
         <p>根据您与陪聊机器人的问答情况,分析你的心理问题类型,再为您智能匹配合适的心理陪聊师。</p>
-        <!--
-        <ion-item>
+        <!--<ion-item>
           <ion-label>选择心理问题类型</ion-label>
           <ion-select [(ngModel)]="selectedIssue">
             <ion-select-option value="anxiety">焦虑</ion-select-option>
@@ -25,7 +24,7 @@
       -->
         <ion-button expand="full" (click)="matchCounselor()">开始匹配</ion-button>
         <div *ngIf="matchedCounselor" class="match-result">
-          <p>为您推荐的心理咨询师是:{{ matchedCounselor?.name }},擅长{{ matchedCounselor?.specialty }}。</p>
+          <p>为您推荐的心理咨询师是:{{ matchedCounselor.name }},擅长{{ matchedCounselor.specialty }}。</p>
         </div>
       </ion-card-content>
     </ion-card>
@@ -39,7 +38,7 @@
       </ion-card-header>
       <ion-card-content>
         <p>在这里,我们的陪聊服务宗旨是为您提供情感支持和倾诉的机会,无论是生活上的开心,还是工作上的糟糕,你都可以跟我分享,这里是属于你一个人的空间,你可以大胆放心的使用。</p>
-        <ion-button expand="full" (click)="startChat()">开始陪聊</ion-button>
+        <ion-button expand="full" (click)="goChat()">开始陪聊</ion-button>
         <!--<ol>
           <li>选择陪聊时间</li>
           <li>与陪聊师匹配</li>
@@ -69,4 +68,4 @@
       </ion-card-content>
     </ion-card>
   </section>
-</ion-content>
+</ion-content>

+ 18 - 18
soul-app/src/app/tab2/tab2.page.ts

@@ -4,6 +4,7 @@ import { ExploreContainerComponent } from '../explore-container/explore-containe
 import { IonButton, IonCard, IonCardContent, IonCardHeader, IonCardTitle, IonItem, IonLabel,
    IonList,IonSelect, IonSelectOption } from '@ionic/angular/standalone';
 import { CommonModule } from '@angular/common';
+import { Router } from '@angular/router';
 
 @Component({
   selector: 'app-tab2',
@@ -21,10 +22,12 @@ import { CommonModule } from '@angular/common';
 })
 export class Tab2Page {
 
-  constructor() {
-    // 其他构造函数代码
+  constructor(
+    private router:Router,
+    ) {
+
   }
-  selectedIssue:string='';
+  //selectedIssue:string='';
   matchedCounselor: { name: string; specialty: string } | null = null;
 
   questions = [
@@ -47,20 +50,17 @@ export class Tab2Page {
 
   matchCounselor() {
     // 这里可以根据selectedIssue进行匹配逻辑
-    if (this.selectedIssue === 'anxiety') {
-      this.matchedCounselor = { name: '张医生', specialty: '焦虑' };
-    } else if (this.selectedIssue === 'depression') {
-      this.matchedCounselor = { name: '李医生', specialty: '抑郁' };
-    } else if (this.selectedIssue === 'relationship') {
-      this.matchedCounselor = { name: '王医生', specialty: '人际关系' };
-    } else {
-      this.matchedCounselor = null;
-    }
+    //if (this.selectedIssue === 'anxiety') {
+      //this.matchedCounselor = { name: '张医生', specialty: '焦虑' };
+    //} else if (this.selectedIssue === 'depression') {
+      //this.matchedCounselor = { name: '李医生', specialty: '抑郁' };
+    //} else if (this.selectedIssue === 'relationship') {
+      //this.matchedCounselor = { name: '王医生', specialty: '人际关系' };
+    //} else {
+      //this.matchedCounselor = null;
+    //}
   }
-
-  startChat() {
-    // 开始陪聊的逻辑
-    console.log('开始陪聊服务');
+  goChat(){
+      this.router.navigateByUrl("/chat/session/role/2DXJkRsjXK")
   }
-
-}
+}

+ 9 - 0
soul-app/src/app/test-chat-panel/test-chat-panel.component.html

@@ -0,0 +1,9 @@
+<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
soul-app/src/app/test-chat-panel/test-chat-panel.component.scss

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

+ 22 - 0
soul-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();
+  });
+});

+ 221 - 0
soul-app/src/app/test-chat-panel/test-chat-panel.component.ts

@@ -0,0 +1,221 @@
+import { CommonModule } from '@angular/common';
+import { ChangeDetectorRef, Component, OnInit, ViewChild } from '@angular/core';
+import { ActivatedRoute } from '@angular/router';
+import { ModalController } from '@ionic/angular/standalone';
+import { ChatPanelComponent } from 'fmode-ng'
+import Parse from "parse";
+import { combineLatest } from 'rxjs';
+
+// 添加Icons
+import { addIcons } from 'ionicons';
+import * as icons from 'ionicons/icons';
+addIcons(icons);
+
+@Component({
+  selector: 'app-test-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","一名亲切可爱的心理咨询师,小聊,年龄28岁");
+            this.chatComp.fmodeChat.role.set("tags",["焦虑","压力"]);
+            this.chatComp.fmodeChat.role.set("avatar","/assets/img/2.png")
+            this.chatComp.fmodeChat.role.set("prompt",`
+# 角色设定
+您是一名亲切可爱的心理咨询师,小聊,年龄28岁,需要完成陪来访者聊聊天,随意轻松一些。
+
+# 对话环节
+0.破冰,互相了解,引导用户介绍自己
+1.拓展话题,根据用户的介绍,拓展一些和其心理状态相关的话题
+- 引导,可深入的点,以用户自述为主
+- 当信息充足时候,确认用户心理状态,并进入下一个环节
+2.引导收尾,委婉引导用户结束本次对话
+- 用户同意结束后,结束本次对话,如果依依不舍,可以再陪聊一会儿`);
+
+// # 开始话语
+// 当您准备好了,可以以一个医生的身份,向来访的用户打招呼。
+//             `);
+
+            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
+        })
+      }
+   }
+}