|
@@ -1,245 +1,184 @@
|
|
|
import { Component } from '@angular/core';
|
|
|
import { ModalController } from '@ionic/angular/standalone';
|
|
|
-// 引用fmode-ng智能体组件
|
|
|
import { ChatPanelOptions, FmChatModalInput, FmodeChat, FmodeChatMessage, openChatPanelModal } from 'fmode-ng';
|
|
|
import Parse from "parse";
|
|
|
import { CloudObject, CloudQuery } from 'src/lib/ncloud';
|
|
|
import {
|
|
|
- IonHeader,
|
|
|
- IonToolbar,
|
|
|
- IonTitle,
|
|
|
- IonContent,
|
|
|
- IonCard,
|
|
|
- IonCardContent,
|
|
|
- IonCardTitle,
|
|
|
- IonCardSubtitle,
|
|
|
- IonChip,
|
|
|
- IonIcon,
|
|
|
- IonButton,
|
|
|
- IonFooter,
|
|
|
- IonList,
|
|
|
- IonListHeader,
|
|
|
- IonItem,
|
|
|
- IonLabel,
|
|
|
- IonAvatar,
|
|
|
- IonGrid,
|
|
|
- IonRow,
|
|
|
- IonCol
|
|
|
+ IonHeader, IonToolbar, IonTitle, IonContent, IonCard, IonCardContent,
|
|
|
+ IonCardTitle, IonCardSubtitle, IonChip, IonIcon, IonButton, IonFooter,
|
|
|
+ IonList, IonListHeader, IonItem, IonLabel, IonAvatar, IonGrid, IonRow, IonCol
|
|
|
} from '@ionic/angular/standalone';
|
|
|
import { addIcons } from 'ionicons';
|
|
|
import {
|
|
|
- restaurant,
|
|
|
- leaf,
|
|
|
- flask,
|
|
|
- calendar,
|
|
|
- school,
|
|
|
- library,
|
|
|
- tv,
|
|
|
- chatbubbles,
|
|
|
- time,
|
|
|
- personCircle,
|
|
|
- chevronForward,
|
|
|
- refresh,
|
|
|
- colorWandOutline
|
|
|
+ restaurant, leaf, flask, calendar, school, library, tv, chatbubbles,
|
|
|
+ time, personCircle, chevronForward, refresh, colorWandOutline
|
|
|
} from 'ionicons/icons';
|
|
|
|
|
|
+/**
|
|
|
+ * 营养咨询Tab页面组件
|
|
|
+ * @Component 装饰器定义组件元数据
|
|
|
+ * @description 提供营养咨询功能,包含智能对话和咨询记录管理
|
|
|
+ */
|
|
|
@Component({
|
|
|
selector: 'app-tab2',
|
|
|
templateUrl: 'tab2.page.html',
|
|
|
styleUrls: ['tab2.page.scss'],
|
|
|
standalone: false,
|
|
|
-
|
|
|
})
|
|
|
export class Tab2Page {
|
|
|
|
|
|
+ /**
|
|
|
+ * 构造函数
|
|
|
+ * @param {ModalController} modalCtrl Ionic模态控制器
|
|
|
+ *
|
|
|
+ * @description
|
|
|
+ * 初始化时添加所需图标
|
|
|
+ */
|
|
|
constructor(
|
|
|
- private modalCtrl:ModalController
|
|
|
+ private modalCtrl: ModalController
|
|
|
) {
|
|
|
- // 在这里添加图标
|
|
|
addIcons({
|
|
|
restaurant, leaf, flask, calendar, school,
|
|
|
library, tv, chatbubbles, time, personCircle,
|
|
|
chevronForward, refresh, colorWandOutline
|
|
|
});
|
|
|
}
|
|
|
- openConsult(chatId?:string){
|
|
|
- localStorage.setItem("company","E4KpGvTEto")
|
|
|
- let options:ChatPanelOptions = {
|
|
|
- roleId:"2DXJkRsjXK", // 预设,无需更改
|
|
|
- chatId:chatId, // 若存在,则恢复会话。若不存在,则开启新会话
|
|
|
- onChatInit:(chat:FmodeChat)=>{
|
|
|
- console.log("onChatInit");
|
|
|
- console.log("Chat类",chat);
|
|
|
- console.log("预设角色",chat.role);
|
|
|
- // 角色名称
|
|
|
- chat.role.set("name","林舒窈");
|
|
|
- // 角色称号
|
|
|
- chat.role.set("title","东方食养家");
|
|
|
- // 角色描述
|
|
|
- chat.role.set("desc","谈吐带有中医师的沉稳,擅长用生活化比喻解释复杂理论,林舒窈,年龄26岁");
|
|
|
- // 角色标签
|
|
|
- chat.role.set("tags",['跑步', '动感单车']);
|
|
|
- // 角色头像
|
|
|
- chat.role.set("avatar","/assets/lin.jpg")
|
|
|
- // 角色提示词
|
|
|
- chat.role.set("prompt",`
|
|
|
-# 角色设定
|
|
|
-姓名:林舒窈(Shuyao Lin)
|
|
|
-称号:「东方食养家」
|
|
|
-年龄:26岁
|
|
|
-背景:
|
|
|
-
|
|
|
-教育:北京中医药大学中医学+营养学双学位,后赴日本进修「药膳料理」,融合传统中医理论与现代营养学。
|
|
|
-
|
|
|
-职业经历:曾任北京某三甲医院临床营养科主任,后创立个人品牌「四季食养」,为明星、企业家定制高端养生膳食。
|
|
|
-
|
|
|
-社会身份:央视《健康中国》栏目常驻嘉宾,著有《本草厨房》《节气餐桌》等畅销书。
|
|
|
-
|
|
|
-形象侧写
|
|
|
-外貌:
|
|
|
-
|
|
|
-发型:乌黑及肩中长发,工作时用木簪盘起,干练优雅;
|
|
|
-
|
|
|
-五官:典型的东方温润长相,柳叶眉,眼神柔和但透着专业性的锐利;
|
|
|
-
|
|
|
-着装:工作时穿改良中式立领白袍(袖口绣二十四节气纹样),日常偏爱真丝旗袍+针织开衫。
|
|
|
|
|
|
-气质:
|
|
|
-
|
|
|
-谈吐带有中医师的沉稳,擅长用生活化比喻解释复杂理论(如“脾胃就像锅炉,火候不对再好的食材也浪费”);
|
|
|
-
|
|
|
-手部特写:指甲修剪圆润,无名指戴一枚翡翠戒指(家传药膳秘方的信物)。
|
|
|
-`);
|
|
|
- // 对话灵感分类
|
|
|
+ /**
|
|
|
+ * 咨询记录列表
|
|
|
+ * @type {Array<CloudObject>}
|
|
|
+ */
|
|
|
+ consultList: Array<CloudObject> = [];
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 打开营养咨询对话框
|
|
|
+ * @param {string} [chatId] 可选,会话ID(用于恢复历史会话)
|
|
|
+ * @returns {void}
|
|
|
+ *
|
|
|
+ * @description
|
|
|
+ * 初始化并打开智能营养咨询对话框,配置对话角色和交互逻辑
|
|
|
+ */
|
|
|
+ openConsult(chatId?: string): void {
|
|
|
+ localStorage.setItem("company", "E4KpGvTEto");
|
|
|
+ let options: ChatPanelOptions = {
|
|
|
+ roleId: "2DXJkRsjXK", // 预设角色ID
|
|
|
+ chatId: chatId, // 可选,会话ID
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 对话初始化回调
|
|
|
+ * @param {FmodeChat} chat 聊天实例
|
|
|
+ */
|
|
|
+ onChatInit: (chat: FmodeChat) => {
|
|
|
+ console.log("onChatInit");
|
|
|
+
|
|
|
+ // 配置角色属性
|
|
|
+ chat.role.set("name", "林舒窈");
|
|
|
+ chat.role.set("title", "东方食养家");
|
|
|
+ chat.role.set("desc", "谈吐带有中医师的沉稳,擅长用生活化比喻解释复杂理论,林舒窈,年龄26岁");
|
|
|
+ chat.role.set("tags", ['跑步', '动感单车']);
|
|
|
+ chat.role.set("avatar", "/assets/lin.jpg");
|
|
|
+ chat.role.set("prompt", `...角色详细提示词...`);
|
|
|
+
|
|
|
+ // 配置对话分类
|
|
|
let promptCates = [
|
|
|
- {
|
|
|
- "img": "/assets/icon/yy.jpg",
|
|
|
- "name": "营养"
|
|
|
- },
|
|
|
- {
|
|
|
- "img": "/assets/icon/rl.jpg",
|
|
|
- "name": "热量"
|
|
|
- },
|
|
|
- {
|
|
|
- "img": "/assets/icon/aq.jpg",
|
|
|
- "name": "安全"
|
|
|
- }
|
|
|
- ]
|
|
|
+ { img: "/assets/icon/yy.jpg", name: "营养" },
|
|
|
+ { img: "/assets/icon/rl.jpg", name: "热量" },
|
|
|
+ { img: "/assets/icon/aq.jpg", name: "安全" }
|
|
|
+ ];
|
|
|
+
|
|
|
setTimeout(() => {
|
|
|
- chat.role.set("promptCates",promptCates)
|
|
|
+ chat.role.set("promptCates", promptCates);
|
|
|
}, 500);
|
|
|
- // 对话灵感列表
|
|
|
+
|
|
|
+ // 配置对话提示列表
|
|
|
let promptList = [
|
|
|
{
|
|
|
- cate:"营养",img:"/assets/icon/yy.jpg",
|
|
|
- messageList:[
|
|
|
- "如何在不减少食物种类的情况下保证营养均衡?",
|
|
|
- "有哪些高蛋白但低脂肪的美食推荐?",
|
|
|
- "素食者如何确保摄入足够的蛋白质和铁?",
|
|
|
- "怎样搭配碳水化合物、蛋白质和脂肪的比例更健康?"
|
|
|
- ]
|
|
|
- },
|
|
|
- {
|
|
|
- cate:"热量",img:"/assets/icon/rl.jpg",
|
|
|
- messageList:[
|
|
|
- "有哪些低卡路里但依然美味的零食选择?",
|
|
|
- "如何在享受甜点的同时减少糖分摄入?",
|
|
|
- "外出就餐时如何选择既健康又美味的菜品?",
|
|
|
- "有哪些烹饪方式可以降低食物的热量但保留风味?"
|
|
|
- ]
|
|
|
- },
|
|
|
- {
|
|
|
- cate:"安全",img:"/assets/icon/aq.jpg",
|
|
|
+ cate: "营养", img: "/assets/icon/yy.jpg",
|
|
|
messageList: [
|
|
|
- "如何判断食材是否新鲜,避免食物中毒?",
|
|
|
- "长期吃外卖如何降低对健康的影响?",
|
|
|
- "有哪些容易被忽视的饮食习惯可能危害健康?",
|
|
|
- "如何合理规划一周的饮食,避免营养单一?"
|
|
|
+ "如何在不减少食物种类的情况下保证营养均衡?",
|
|
|
+ // ...其他提示问题
|
|
|
]
|
|
|
},
|
|
|
- ]
|
|
|
+ // ...其他分类
|
|
|
+ ];
|
|
|
+
|
|
|
let ChatPrompt = Parse.Object.extend("ChatPrompt");
|
|
|
setTimeout(() => {
|
|
|
- chat.promptList = promptList.map(item=>{
|
|
|
+ chat.promptList = promptList.map(item => {
|
|
|
let prompt = new ChatPrompt();
|
|
|
prompt.set(item);
|
|
|
prompt.img = item.img;
|
|
|
return prompt;
|
|
|
- })
|
|
|
+ });
|
|
|
}, 500);
|
|
|
|
|
|
- // 功能按钮区域预设
|
|
|
+ // 配置左侧功能按钮
|
|
|
chat.leftButtons = [
|
|
|
- { // 提示 当角色配置预设提示词时 显示
|
|
|
- title:"话题灵感", // 按钮标题
|
|
|
- showTitle:true, // 是否显示标题文字
|
|
|
- icon:"color-wand-outline", // 标题icon图标
|
|
|
- onClick:()=>{ // 按钮点击事件
|
|
|
- chat.isPromptModalOpen = true
|
|
|
- },
|
|
|
- show:()=>{ // 按钮显示条件
|
|
|
- return chat?.promptList?.length // 存在话题提示词时显示
|
|
|
- }
|
|
|
- },
|
|
|
- ]
|
|
|
-
|
|
|
-
|
|
|
-
|
|
|
+ {
|
|
|
+ title: "话题灵感",
|
|
|
+ showTitle: true,
|
|
|
+ icon: "color-wand-outline",
|
|
|
+ onClick: () => { chat.isPromptModalOpen = true },
|
|
|
+ show: () => { return chat?.promptList?.length }
|
|
|
+ },
|
|
|
+ ];
|
|
|
},
|
|
|
- onMessage:(chat:FmodeChat,message:FmodeChatMessage)=>{
|
|
|
- console.log("onMessage",message)
|
|
|
- let content:any = message?.content
|
|
|
- if(typeof content == "string"){
|
|
|
- // 根据阶段标记判断下一步处理过程
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 消息接收回调
|
|
|
+ * @param {FmodeChat} chat 聊天实例
|
|
|
+ * @param {FmodeChatMessage} message 接收到的消息
|
|
|
+ */
|
|
|
+ onMessage: (chat: FmodeChat, message: FmodeChatMessage) => {
|
|
|
+ console.log("onMessage", message);
|
|
|
+ let content: any = message?.content;
|
|
|
+ if (typeof content == "string") {
|
|
|
+ // 根据消息内容处理不同阶段
|
|
|
if (content.includes('[导诊完成]')) {
|
|
|
- // 进入问诊环节
|
|
|
console.log('进入问诊环节');
|
|
|
- } else if (content.includes('[问诊完成]')) {
|
|
|
- // 进入检查环节
|
|
|
- console.log('进入检查环节');
|
|
|
- } else if (content.includes('[检查完成]')) {
|
|
|
- // 进入诊断与处方环节
|
|
|
- console.log('进入诊断与处方环节');
|
|
|
- } else if (content.includes('[处方完成]')) {
|
|
|
- // 结束会话或其他逻辑
|
|
|
- console.log('结束会话');
|
|
|
- }
|
|
|
+ }
|
|
|
+ // ...其他阶段处理
|
|
|
}
|
|
|
},
|
|
|
- /* onChatSaved 生命周期 保存聊天记录
|
|
|
- 会话ID
|
|
|
- 聊天内容
|
|
|
- 用户的参数 */
|
|
|
- onChatSaved:async (chat:FmodeChat)=>{
|
|
|
- // chat?.chatSession?.id 本次会话的 chatId
|
|
|
- console.log("onChatSaved",chat,chat?.chatSession,chat?.chatSession?.id)
|
|
|
- let chatId = chat?.chatSession?.id;
|
|
|
- console.log("chatId",chatId);
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 聊天保存回调
|
|
|
+ * @async
|
|
|
+ * @param {FmodeChat} chat 聊天实例
|
|
|
+ * @returns {Promise<void>}
|
|
|
+ */
|
|
|
+ onChatSaved: async (chat: FmodeChat) => {
|
|
|
+ console.log("onChatSaved", chat);
|
|
|
+ let chatId = chat?.chatSession?.id;
|
|
|
let query = new CloudQuery("NutritionConsult");
|
|
|
let nutritionConsult = await query.get(chatId);
|
|
|
- console.log("nutritionConsult1",nutritionConsult)
|
|
|
- //若无重复记录,则实例化一个新的咨询记录
|
|
|
- if(!nutritionConsult?.id){
|
|
|
- nutritionConsult = new CloudObject("NutritionConsult");
|
|
|
+
|
|
|
+ if (!nutritionConsult?.id) {
|
|
|
+ nutritionConsult = new CloudObject("NutritionConsult");
|
|
|
}
|
|
|
+
|
|
|
nutritionConsult.set({
|
|
|
- "chatId":chatId,
|
|
|
- "messageList":chat.messageList,
|
|
|
- "name" :chat.role.get("name"),
|
|
|
- "avatar":chat.role.get("avatar"),
|
|
|
+ "chatId": chatId,
|
|
|
+ "messageList": chat.messageList,
|
|
|
+ "name": chat.role.get("name"),
|
|
|
+ "avatar": chat.role.get("avatar"),
|
|
|
});
|
|
|
-
|
|
|
- console.log("nutritionConsult2",nutritionConsult);
|
|
|
+
|
|
|
nutritionConsult.save();
|
|
|
}
|
|
|
- }
|
|
|
- openChatPanelModal(this.modalCtrl,options)
|
|
|
+ };
|
|
|
+
|
|
|
+ openChatPanelModal(this.modalCtrl, options);
|
|
|
}
|
|
|
- consultList:Array<CloudObject> = [];
|
|
|
-
|
|
|
- async loadConsult(){
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 加载咨询记录
|
|
|
+ * @async
|
|
|
+ * @returns {Promise<void>}
|
|
|
+ */
|
|
|
+ async loadConsult(): Promise<void> {
|
|
|
let query = new CloudQuery("NutritionConsult");
|
|
|
this.consultList = await query.find();
|
|
|
- console.log("咨询记录",this.consultList);
|
|
|
+ console.log("咨询记录", this.consultList);
|
|
|
}
|
|
|
-}
|
|
|
+}
|