customization.page.ts 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591
  1. import { Component, Input, OnInit } from '@angular/core';
  2. import { CommonModule } from '@angular/common';
  3. import { FormsModule } from '@angular/forms';
  4. import { NavigationExtras, Router } from '@angular/router';
  5. import { IonCardSubtitle, IonProgressBar, IonTextarea, ModalController } from '@ionic/angular/standalone';
  6. import { IonContent, IonHeader, IonTitle, IonToolbar, IonButton, IonLabel, IonItem, IonList, IonBackButton, IonButtons, IonIcon, IonItemDivider, IonAvatar, IonThumbnail, IonItemOptions, IonItemOption, IonItemSliding, IonInput, IonCheckbox, IonRadio, IonToggle, IonRadioGroup, IonSearchbar, IonSegment, IonSegmentButton, IonDatetime, IonFooter, IonCardContent, IonCardTitle, IonCardHeader, IonCard, IonCol, IonRow, IonGrid, IonChip, IonImg } from '@ionic/angular/standalone';
  7. import { TestChatCompletion } from './test-chat-completion';
  8. import { EditTagComponent } from '../edit-tag/edit-tag.component';
  9. import { FmodeChatCompletion, MarkdownPreviewModule } from 'fmode-ng';
  10. import { DalleOptions, ImagineWork } from 'fmode-ng';
  11. import { ChatPanelOptions, FmChatModalInput, FmodeChat, FmodeChatMessage, openChatPanelModal } from 'fmode-ng';
  12. import { Icon } from 'ionicons/dist/types/components/icon/icon';
  13. import { camera, trendingUpOutline, sparklesOutline, cloudyOutline, diceOutline } from 'ionicons/icons';
  14. import { addIcons } from 'ionicons';
  15. import { CloudObject, CloudQuery, CloudUser } from 'src/lib/ncloud';
  16. import { SwiperComponent } from '../swiper/swiper.component';
  17. import { extactAndParseJsonFromString } from 'src/agent/agent.json';
  18. import { RouterModule } from '@angular/router';
  19. import { NavController } from '@ionic/angular';
  20. import { AlertController } from '@ionic/angular';
  21. import { openUserLoginModal } from 'src/lib/user/modal-user-login/modal-user-login.component';
  22. import { openCommentPostModal } from 'src/lib/user/modal-comment-post/modal-comment-post.component';
  23. addIcons({ camera, trendingUpOutline, sparklesOutline, cloudyOutline, diceOutline })
  24. @Component({
  25. selector: 'app-customization',
  26. templateUrl: './customization.page.html',
  27. styleUrls: ['./customization.page.scss'],
  28. standalone: true,
  29. imports: [IonProgressBar,IonContent, IonHeader, IonTitle, IonToolbar, CommonModule, FormsModule, IonButton, IonInput, EditTagComponent, IonTextarea, IonItem, IonList, MarkdownPreviewModule, IonIcon, IonContent, IonHeader, IonTitle, IonToolbar, CommonModule, FormsModule, IonButton, IonLabel, IonLabel, IonList, IonItem, IonBackButton, IonButtons, IonIcon, IonItemDivider, IonAvatar, IonThumbnail, IonItemOptions, IonItemSliding, IonItemOption, IonItemOptions, IonInput, IonCheckbox, IonRadio, IonToggle, IonRadioGroup, IonSearchbar, IonSegment, IonSegmentButton, IonDatetime, IonFooter, IonCardContent, IonCardTitle, IonCardHeader, IonCard, IonCol, IonRow, IonGrid, IonChip, IonImg, IonCardSubtitle, SwiperComponent, RouterModule,]
  30. })
  31. export class CustomizationPage implements OnInit {
  32. public buffer = 0.06;
  33. public progress = 0;
  34. // 修改需求分析
  35. async modifyRequire() {
  36. this.requireComplete=false
  37. let a = await openCommentPostModal(this.modalCtrl, { "title": "需求修改", "title2": "修改" });
  38. if (a == null) {
  39. a = "";
  40. }
  41. let prompt = `你是一名专业服装设计师,这是你的上次回答:
  42. ${this.gptre}
  43. 基于你上次回答,我想做一些修改,修改如下:
  44. ${a},
  45. 请你将上次回答基于我提出的修改修改后再发一遍
  46. `;
  47. let completion = new FmodeChatCompletion([
  48. { role: "system", content: "" },
  49. { role: "user", content: prompt }
  50. ]);
  51. completion.sendCompletion().subscribe(
  52. (message: any) => {
  53. this.gptre=message.content
  54. console.log("Received message:", message);
  55. if(this.progress<0.97){
  56. if(this.progress<0.5){
  57. this.progress+=0.01
  58. }
  59. if(this.progress>=0.5){
  60. this.progress+=0.001
  61. }
  62. }
  63. if (message?.complete) {
  64. this.requireComplete=true
  65. this.progress=0
  66. console.log("完成");
  67. }
  68. }
  69. )
  70. }
  71. async createdetail() {
  72. this.type="detail"
  73. let promotTemplate = `您是一名专业的美术画家,请您根据服装描述内容,将1.其描述的服装细节描述出来,服装描述如下:${this.gptre}`
  74. let completion = new FmodeChatCompletion([
  75. { role: "system", content: "" },
  76. { role: "user", content: promotTemplate }
  77. ])
  78. completion.sendCompletion().subscribe((message: any) => {
  79. // 打印消息体
  80. console.log(message.content)
  81. // 赋值消息内容给组件内属性
  82. this.picdetail = message.content
  83. if(this.progress<0.97){
  84. if(this.progress<0.5){
  85. this.progress+=0.01
  86. }
  87. if(this.progress>=0.5){
  88. this.progress+=0.001
  89. }
  90. }
  91. if (message?.complete) {
  92. this.isComplete = true;
  93. // this.getJson()
  94. this.progress=0
  95. }
  96. })
  97. }
  98. async modifyDetail() {
  99. this.isComplete=false
  100. let a = await openCommentPostModal(this.modalCtrl, { "title": "细节修改", "title2": "修改" });
  101. if (a == null) {
  102. a = "";
  103. }
  104. let prompt = `你是一名专业服装设计师,这是你的上次回答:
  105. ${this.picdetail}
  106. 基于你上次回答,我想做一些修改,修改如下:
  107. ${a},
  108. 请你将上次回答基于我提出的修改修改后再发一遍
  109. `;
  110. let completion = new FmodeChatCompletion([
  111. { role: "system", content: "" },
  112. { role: "user", content: prompt }
  113. ]);
  114. completion.sendCompletion().subscribe(
  115. (message: any) => {
  116. this.picdetail=message.content
  117. console.log("Received message:", message);
  118. if(this.progress<0.97){
  119. if(this.progress<0.5){
  120. this.progress+=0.01
  121. }
  122. if(this.progress>=0.5){
  123. this.progress+=0.001
  124. }
  125. }
  126. if (message?.complete) {
  127. this.isComplete=true
  128. console.log("完成");
  129. this.progress=0
  130. }
  131. }
  132. )
  133. }
  134. getJson() {
  135. this.type="image"
  136. let promt = `请你以以下json格式整合文段内容:
  137. {
  138. "goal":"客户定制服装目的",
  139. "style":"风格",
  140. "color":"颜色",
  141. "feature":"客户特殊的身体特征胸围大点什么的,如果客户没有特殊的身体特征这里填空字符串",
  142. "ACC":"配饰,耳环,领带什么的",
  143. "rim":"辅料,纽扣,拉链什么的",
  144. "period":"定制周期",
  145. "clothing":"以衣服类型+推荐尺码这样的格式拼接,比如西装外套+XXL",
  146. "trousers":"以裤子类型+推荐尺码这样的格式拼接,如果客户没有指定裤子类型,由你来选一种合适的裤子类型",
  147. "shoe":"以鞋子类型+推荐尺码这样的格式拼接,如果客户没有指定鞋子类型,由你来选一种合适的鞋子类型",
  148. "after":"售后信息",
  149. "texture":"材质"
  150. "remark":"备注,用户给出了不属于上述字段的需求就全部拼接在这里显示"
  151. }
  152. 内容如下:${this.picdetail} ${this.gptre}
  153. `
  154. let completion = new FmodeChatCompletion([
  155. { role: "system", content: "" },
  156. { role: "user", content: promt }
  157. ])
  158. completion.sendCompletion().subscribe((message: any) => {
  159. if(this.progress<0.97){
  160. if(this.progress<0.8){
  161. this.progress+=0.005
  162. }
  163. if(this.progress>=0.8){
  164. this.progress+=0.001
  165. }
  166. }
  167. // 打印消息体
  168. console.log(message.content)
  169. // 赋值消息内容给组件内属性
  170. this.JSONdes = message.content
  171. if (message?.complete) {
  172. console.log("json:", this.JSONdes);
  173. this.imagineWork = new ImagineWork();
  174. let options: DalleOptions = { prompt: this.picdetail }
  175. this.imagineWork.draw(options).subscribe(async work => {
  176. console.log("imagineWork", work?.toJSON())
  177. console.log("images", work?.get("images"))
  178. if (work?.get("images")?.length) {
  179. this.images = work?.get("images");
  180. this.progress=0
  181. let a = extactAndParseJsonFromString(this.JSONdes)
  182. // a.comment=await this.presentAlert();
  183. // console.log("a.comment",a.comment);
  184. a.detail = this.picdetail;
  185. a.image =this.images[0]
  186. this.JSONobject = a
  187. this.JSONKeys = Object.keys(this.JSONobject)
  188. this.JSONcomplete = true
  189. console.log(a);
  190. //加入数据库
  191. let newDsign = {
  192. "goal": a['goal'],
  193. "style": a['style'],
  194. "color": a['color'],
  195. "feature": a['feature'],
  196. "ACC": a['ACC'],
  197. "rim": a['rim'],
  198. "period": a['period'],
  199. "clothing": a['clothing'],
  200. "trousers": a['trousers'],
  201. "shoe": a['shoe'],
  202. "after": a['after'],
  203. "texture": a['texture'],
  204. "remark": a['remark'],
  205. "image": a['image'],
  206. "user": new CloudUser().toPointer(),
  207. "detail": a['detail'],
  208. "comment": ""
  209. }
  210. let newObject = new CloudObject("Designs")
  211. newObject.set(newDsign)
  212. newObject.id = (await newObject.save()).id
  213. this.newObject = newObject;
  214. console.log("添加成功");
  215. }
  216. })
  217. }
  218. })
  219. return true;
  220. }
  221. //打开聊天窗口
  222. async openInquiry(doctor: CloudObject) {
  223. let currentUser = new CloudUser();
  224. let userPrompt = ``
  225. if (!currentUser?.id) {
  226. console.log("用户未登录,请登录后重试");
  227. let user = await openUserLoginModal(this.modalCtrl);
  228. if (!user?.id) {
  229. return
  230. }
  231. currentUser = user;
  232. }
  233. if (currentUser?.get("realname")) {
  234. userPrompt += `当前来访的患者,姓名:${currentUser?.get("realname")}`
  235. }
  236. if (currentUser?.get("gender")) {
  237. userPrompt += `,性别:${currentUser?.get("gender")}`
  238. }
  239. if (currentUser?.get("age")) {
  240. userPrompt += `,年龄:${currentUser?.get("age")}`
  241. }
  242. localStorage.setItem("company", "Svehl6FceL")
  243. let options: ChatPanelOptions = {
  244. roleId: "sVT0ezdJml",
  245. chatId: this.chatID,
  246. onClose: (chat: FmodeChat): boolean | Promise<boolean> => {
  247. //开始分析用户和ai聊天记录
  248. console.log("关闭");
  249. if (!this.complete) { return true; }
  250. console.log(this.isComplete);
  251. console.log("开始分析");
  252. this.complete = false;
  253. console.log(chat.messageList);
  254. let contentList = chat.messageList.map(item => ({
  255. role: item.role,
  256. content: item.content
  257. }));
  258. let stringArray = contentList.map(item => `${item.role}:\n ${item.content}`);
  259. console.log(stringArray);
  260. let contentStr = stringArray.slice(2).join("\n")
  261. console.log(contentStr);
  262. let promt = `您作为一名专业的服装推荐师,您的任务是为用户进行特定的服装设计,设计好后面料颜色等都要选定以便后续制造,请帮我分析以下用户user与一位服装设计师assistant的对话,详细分析出:1.用户对于定制服装的需求,2.联想并补全服装的细节,对话如下:${contentStr}
  263. `
  264. let completion = new FmodeChatCompletion([
  265. { role: "system", content: "" },
  266. { role: "user", content: promt }
  267. ])
  268. completion.sendCompletion().subscribe((message: any) => {
  269. if(this.progress<0.97){
  270. if(this.progress<0.5){
  271. this.progress+=0.01
  272. }
  273. if(this.progress>=0.5){
  274. this.progress+=0.001
  275. }
  276. }
  277. console.log(message.content)
  278. console.log("complete:::::", this.isComplete);
  279. this.gptre = message.content
  280. if (message?.complete) {
  281. this.progress=0
  282. this.requireComplete=true;
  283. // this.createImage()
  284. }
  285. })
  286. return true;
  287. },
  288. onChatInit: (chat: FmodeChat) => {
  289. console.log("onChatInit");
  290. console.log("预设角色", chat.role);
  291. chat.role.set("name", doctor?.get("name"));
  292. chat.role.set("title", doctor?.get("title"));
  293. chat.role.set("desc", `一名${doctor?.get("desc")}的服装设计专家,${doctor?.get("name")},年龄${doctor?.get("age")}岁`);
  294. chat.role.set("tags", ["服装", "设计"]);
  295. chat.role.set("avatar", doctor?.get("avater") || "../../assets/img/doctor5.png")
  296. chat.role.set("prompt", `
  297. # 角色设定
  298. 您是一名亲切和蔼的服装设计专家,${doctor?.get("name")},年龄${doctor?.get("age")}岁,需要完成一次完整的服装定制服务。
  299. # 对话环节
  300. 0.引导(根据用户基本情况,引导其描述需要定制的服装)
  301. 1.预设的问询方式:询问顾客定制服装的目的(如婚礼、晚会、日常穿着等)。
  302. - 打招呼,以用户自述为主
  303. - 当信息充足时候,根据用户定制服装目的做出一些推荐服装来引导用户,并进入下一个环节
  304. 2.拓展的问询细节
  305. 例如:询问顾客的风格偏好(如休闲、正式、个性化等)。是否要指定特定的材质?倾向于哪种版型?英式、意式或其他?是否有特定的身体特征需要特别注意,例如肩宽、胸围、腰围等?是否有任何特殊的图案、标志或文字想要添加到服装上?是否需要这件服装与特定的配饰或鞋子或裤子搭配?是否需要特定的辅料,如纽扣、拉链、衬里等?是否需要特定的颜色或配色方案?对服装有没有什么特殊的需求?期望的定制周期是多久?您对售后服务有什么期望?
  306. - 以上这些问题要要逐个问,不能一次性给出让用户作答,每次问完都要根据之前回答和这次提问内容为用户推荐特定的服装来引导用户(比如前面用户回答是为了婚礼,那你应该先说一下婚礼的服装需要满足的特点,因为接下来你要问用户需要什么风格,你就可以基于用户婚礼这个回答先推荐西装,燕尾服等服装,然后再问用户需要什么风格的服装,推荐5个选择),当问询细节补充完成后进入下一个环节
  307. 3.初步的定制结果,并且同时列出定制项目
  308. 初步结果:初步得出现在定制的结果要得出客户定制服装目的,风格,颜色,客户特殊的身体特征,配饰,辅料,定制周期,衣服类型,裤子类型,鞋子类型,售后信息,材质,备注
  309. 定制项目:展现给用户看当前类似与现在定制结果的服装的市场趋势
  310. - 进入下一阶段
  311. 4.在确定好服装后,询问客户身高体重,为他提供合适的尺码推荐
  312. - 等待用户提交身高体重,进入下一阶段
  313. 4.给出初步的定制方案
  314. - 完成定制时,以json格式给出定制方案,json格式如下:
  315. {
  316. "goal":"客户定制服装目的",
  317. "style":"风格",
  318. "color":"颜色",
  319. "feature":"客户特殊的身体特征胸围大点什么的,如果客户没有特殊的身体特征这里填空字符串",
  320. "ACC":"配饰,耳环,领带什么的",
  321. "rim":"辅料,纽扣,拉链什么的",
  322. "period":"定制周期",
  323. "clothing":"以衣服类型+推荐尺码这样的格式拼接,比如西装外套+XXL",
  324. "trousers":"以裤子类型+推荐尺码这样的格式拼接,如果客户没有指定裤子类型,由你来选一种合适的裤子类型",
  325. "shoe":"以鞋子类型+推荐尺码这样的格式拼接,如果客户没有指定鞋子类型,由你来选一种合适的鞋子类型",
  326. "after":"售后信息",
  327. "texture":"材质"
  328. "remark":"备注,用户给出了不属于上述字段的需求就全部拼接在这里显示"
  329. }
  330. ,并在消息结尾附带: [完成]
  331. # 开始话语
  332. 当您准备好了,可以以一个设计师的身份,向来访的用户打招呼。
  333. ${userPrompt}
  334. `);
  335. },
  336. onMessage: (chat: FmodeChat, message: FmodeChatMessage) => {
  337. console.log("onMessage", message)
  338. let content: any = message?.content
  339. this.messageList.push(content);
  340. console.log("this.messageList", this.messageList);
  341. if (typeof content == "string") {
  342. if (content?.indexOf("[完成]") > -1) {
  343. console.log("门诊已完成")
  344. this.complete = true
  345. }
  346. }
  347. },
  348. onChatSaved: (chat: FmodeChat) => {
  349. // chat?.chatSession?.id 本次会话的 chatId
  350. console.log("onChatSaved", chat, chat?.chatSession, chat?.chatSession?.id)
  351. this.chatID = chat?.chatSession?.id
  352. },
  353. }
  354. openChatPanelModal(this.modalCtrl, options)
  355. console.log(this.messageList);
  356. }
  357. @Input()
  358. type: "require" | "detail" | "image" = "require"
  359. typeChange(ev: any) {
  360. this.type = ev?.detail?.value || ev?.value || 'require'
  361. }
  362. public isComplete: boolean = false;
  363. public requireComplete: boolean = true;
  364. editTags: Array<string> = []
  365. picdetail: string = ""
  366. public JSONKeys: string[] = []
  367. public JSONKeys2: string[] = []
  368. async toComment() {
  369. let a = await openCommentPostModal(this.modalCtrl, { "title": "评论", "title2": "评论" })
  370. if (a == null) {
  371. a = ""
  372. }
  373. this.newObject.set({ "comment": a })
  374. this.newObject.save()
  375. console.log("添加评论成功");
  376. }
  377. toReport() {
  378. // this.nav.navigateRoot(['report'], {
  379. // queryParams: {
  380. // 'object': JSON.stringify(this.JSONobject)
  381. // }
  382. // });
  383. const state: any = { myObject: this.JSONobject }; // 你要传递的对象数据
  384. const navigationExtras: NavigationExtras = {
  385. state: state
  386. };
  387. this.router.navigate(['/report'], navigationExtras);
  388. // this.router.navigate(['/report',this.JSONobject2])
  389. }
  390. ngOnInit() {
  391. this.loadDoctorList()
  392. }
  393. test() {
  394. console.log("开始");
  395. let newDsign = {
  396. "goal": "婚礼",
  397. "style": "修身燕尾服",
  398. "color": "黑色",
  399. "feature": "",
  400. "ACC": "领带",
  401. "rim": "纽扣",
  402. "period": "7天",
  403. "clothing": "燕尾服+XXL",
  404. "trousers": "裤子+XXL",
  405. "shoe": "鞋子+42码",
  406. "after": "清洗",
  407. "texture": "丝绸",
  408. "remark": "",
  409. "detail": ""
  410. }
  411. let newObject = new CloudObject("Designs")
  412. newObject.set(newDsign)
  413. console.log(newObject.save());
  414. }
  415. constructor(
  416. private modalCtrl: ModalController,
  417. private router: Router,
  418. public nav: NavController,
  419. private alertController: AlertController
  420. ) {
  421. // 示例任务,自己生成图片后请存储新的ID
  422. console.log('ImagineWork instance:', this.imagineWork);
  423. // this.imagineWork = new ImagineWork("Wx6o1Js1gA");
  424. // this.imagineWork.fetchTask().then(work => {
  425. // this.images = this.imagineWork?.images || [];
  426. // })
  427. }
  428. imagineWork: ImagineWork | undefined
  429. images: Array<string> = []
  430. public gptre = `在这段对话中,我们可以分析出用户的需求和服装的细节设计。以下是详细分析:
  431. ### 1. 用户对于定制服装的需求
  432. - **场合**:用户明确表示需要定制服装的场合是婚礼,这意味着服装需要正式且优雅。
  433. - **风格选择**:用户希望设计师帮忙选择风格,并没有明确的偏好,这表明用户可能对婚礼服装的具体风格不太了解或者希望设计师提供专业建议。
  434. - **颜色和材质**:用户在颜色和材质方面也没有特别要求,完全依赖设计师的选择,这显示出用户对设计师的信任。
  435. - **尺码和身体特征**:用户提供了身高和体重,并提到“胸围大点”,这意味着在服装的剪裁上需要特别注意胸部的合身度。
  436. - **售后服务**:用户希望有售后服务,特别是调整和清洗服务,显示出用户对服装的长期使用和维护的关注。
  437. ### 2. 联想并补全服装的细节
  438. - **服装类型**:选择了经典的燕尾服,适合正式的婚礼场合。
  439. - **颜色**:选择了经典的黑色,符合婚礼的正式氛围。
  440. - **材质**:选择了丝绸,因其光泽感和柔软度,适合高档场合。
  441. - **配饰**:选择了黑色领结和白色口袋巾,经典而优雅,能够提升整体形象。
  442. - **辅料**:选择了黑色纽扣和丝绸衬里,确保整体风格统一且高档。
  443. - **定制周期**:设定为3周,符合一般定制的时间要求。
  444. - **尺码**:根据用户的身高和体重,推荐XXL尺码,特别注意胸围的合身。
  445. - **鞋子类型**:建议42-43码的皮鞋,符合用户的身高体重。
  446. - **售后服务**:包括免费调整和清洗服务,确保用户在婚礼后也能保持服装的最佳状态。
  447. - **备注**:用户希望随时联系,表明用户对沟通的重视。
  448. ### 总结
  449. 通过对用户需求的分析和对服装细节的补全,我们可以为用户提供一套完整的婚礼燕尾服定制方案。这不仅满足了用户对正式场合服装的需求,也考虑到了用户的个人特征和售后服务的期望。这样的定制方案将有助于用户在婚礼上展现最佳形象。`
  450. messageList: string[] = []
  451. title: string = "123"
  452. chatID = ""
  453. complete: boolean = false
  454. JSONcomplete: boolean = false
  455. JSONdes = ""
  456. JSONobject: { [key: string]: string } = {}
  457. /** 示例:问诊ChatPanel面板 */
  458. newObject: CloudObject = new CloudObject("Designs")
  459. shareData: any = {}
  460. async loadDoctorList() {
  461. let query = new CloudQuery("Designer");
  462. query.include("depart")
  463. this.doctorList = await query.find()
  464. console.log(this.doctorList);
  465. }
  466. doctorList: Array<CloudObject> = []
  467. }