fmode-ng 0.0.204 → 0.0.205
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/aigc/chat/chat-header-area/comp-header-area.component.mjs +3 -3
- package/esm2022/lib/aigc/chat/chat-modal-input/modal-input.component.mjs +2 -2
- package/esm2022/lib/core/parse/fmode.query.mjs +3 -3
- package/esm2022/lib/social/wechat/wechat-jssdk.service.mjs +12 -11
- package/esm2022/lib/user/account/account.service.mjs +8 -2
- package/esm2022/lib/user/login/auth.service.mjs +139 -53
- package/esm2022/lib/user/modal-user-login/modal-user-login.component.mjs +3 -8
- package/fesm2022/fmode-ng.mjs +488 -401
- package/fesm2022/fmode-ng.mjs.map +1 -1
- package/lib/aigc/story/fm-story-list/story-preview.d.ts +1 -1
- package/lib/person/edit-upload/edit-upload.component.d.ts +1 -1
- package/lib/social/wechat/wechat-jssdk.service.d.ts +2 -3
- package/lib/user/login/auth.service.d.ts +9 -5
- package/package.json +1 -1
|
@@ -5,11 +5,11 @@ import { CompAvatarRoleImageComponent } from '../../avatar';
|
|
|
5
5
|
import { CompAvatarRoleVideoComponent } from '../../avatar';
|
|
6
6
|
import { FmodeChat } from '../../service-fmai/service-chat';
|
|
7
7
|
import { addIcons } from 'ionicons';
|
|
8
|
-
import { chevronBackOutline, ellipsisHorizontalOutline } from 'ionicons/icons';
|
|
8
|
+
import { chevronBackOutline, ellipsisHorizontalOutline, resizeOutline } from 'ionicons/icons';
|
|
9
9
|
import * as i0 from "@angular/core";
|
|
10
10
|
import * as i1 from "@ionic/angular/standalone";
|
|
11
11
|
import * as i2 from "@angular/common";
|
|
12
|
-
addIcons({ chevronBackOutline, ellipsisHorizontalOutline });
|
|
12
|
+
addIcons({ chevronBackOutline, ellipsisHorizontalOutline, resizeOutline });
|
|
13
13
|
export class FmChatHeaderArea {
|
|
14
14
|
constructor(navCtrl) {
|
|
15
15
|
this.navCtrl = navCtrl;
|
|
@@ -46,4 +46,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
46
46
|
}], goBack: [{
|
|
47
47
|
type: Input
|
|
48
48
|
}] } });
|
|
49
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"comp-header-area.component.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/chat/chat-header-area/comp-header-area.component.ts","../../../../../../../projects/fmode-ng/src/lib/aigc/chat/chat-header-area/comp-header-area.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,eAAe,CAAC;AACjD,OAAO,EAAE,UAAU,EAAE,OAAO,EAAE,SAAS,EAAE,OAAO,EAAC,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,UAAU,EAAE,QAAQ,EAAE,aAAa,EAAE,SAAS,EAAE,MAAM,2BAA2B,CAAC;AAC7L,OAAO,EAAE,4BAA4B,EAAE,MAAM,cAAc,CAAC;AAC5D,OAAO,EAAE,4BAA4B,EAAE,MAAM,cAAc,CAAC;AAC5D,OAAO,EAAE,SAAS,EAAE,MAAM,iCAAiC,CAAC;AAE5D,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAC,kBAAkB,EAAC,yBAAyB,EAAE,aAAa,EAAC,MAAM,gBAAgB,CAAC;;;;AAC3F,QAAQ,CAAC,EAAC,kBAAkB,EAAC,yBAAyB,EAAC,aAAa,EAAC,CAAC,CAAA;AAiBtE,MAAM,OAAO,gBAAgB;IAI3B,YACU,OAAqB;QAArB,YAAO,GAAP,OAAO,CAAc;QAH/B,gBAAW,GAAW,KAAK,CAAA;QAe3B,WAAM,GAAG,GAAE,EAAE;YACX,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;QACtB,CAAC,CAAA;IAbC,CAAC;IAEH,KAAK,CAAC,IAAI;QACR,IAAG,IAAI,CAAC,IAAI,EAAE,OAAO,EAAC,CAAC;YACrB,IAAI,WAAW,GAAI,MAAM,IAAI,CAAC,IAAI,EAAE,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;YACtD,IAAG,CAAC,WAAW;gBAAE,OAAM;QACzB,CAAC;QACD,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;+GAdU,gBAAgB;mGAAhB,gBAAgB,2HC1B7B,+wHAiFc,2bD/DV,YAAY,mIAAC,UAAU,mFAAC,UAAU,8EAAC,SAAS,oPAAC,OAAO,2JACpD,QAAQ,sDACR,QAAQ,iFAAC,SAAS,oGAAC,OAAO,yFAAC,OAAO,0NAAC,OAAO,yLAAC,QAAQ,6FAAC,OAAO;gBAC3D,GAAG;gBACH,4BAA4B,gGAC5B,4BAA4B;;4FAGnB,gBAAgB;kBAd5B,SAAS;+BACE,qBAAqB,cAGpB,IAAI,WACP;wBACN,YAAY,EAAC,UAAU,EAAC,UAAU,EAAC,SAAS,EAAC,OAAO;wBACpD,QAAQ,EAAC,SAAS;wBAClB,QAAQ,EAAC,SAAS,EAAC,OAAO,EAAC,OAAO,EAAC,OAAO,EAAC,QAAQ,EAAC,OAAO;wBAC3D,GAAG;wBACH,4BAA4B;wBAC5B,4BAA4B;qBAC7B;kFAGQ,IAAI;sBAAZ,KAAK;gBAgBN,MAAM;sBADL,KAAK","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, Input } from '@angular/core';\r\nimport { IonButtons, IonCard, IonHeader, IonIcon,IonButton, IonItem, IonLabel, IonList, IonNote, IonTitle, IonToolbar, IonModal, NavController, IonAvatar } from '@ionic/angular/standalone';\r\nimport { CompAvatarRoleImageComponent } from '../../avatar';\r\nimport { CompAvatarRoleVideoComponent } from '../../avatar';\r\nimport { FmodeChat } from '../../service-fmai/service-chat';\r\n\r\nimport { addIcons } from 'ionicons';\r\nimport {chevronBackOutline,ellipsisHorizontalOutline, resizeOutline} from 'ionicons/icons';\r\naddIcons({chevronBackOutline,ellipsisHorizontalOutline,resizeOutline})\r\n\r\n\r\n@Component({\r\n  selector: 'fm-chat-header-area',\r\n  templateUrl: './comp-header-area.component.html',\r\n  styleUrls: ['./comp-header-area.component.scss'],\r\n  standalone:true,\r\n  imports:[\r\n    CommonModule,IonToolbar,IonButtons,IonButton,IonIcon,\r\n    IonModal,IonAvatar,\r\n    IonTitle,IonHeader,IonList,IonItem,IonCard,IonLabel,IonNote,\r\n    // \r\n    CompAvatarRoleImageComponent,\r\n    CompAvatarRoleVideoComponent\r\n  ]\r\n})\r\nexport class FmChatHeaderArea {\r\n  @Input() chat:FmodeChat\r\n  isModalOpen:boolean = false\r\n\r\n  constructor(\r\n    private navCtrl:NavController\r\n  ){}\r\n\r\n  async back(){\r\n    if(this.chat?.onClose){\r\n      let closeResult =  await this.chat?.onClose(this.chat)\r\n      if(!closeResult) return\r\n    }\r\n    this.goBack();\r\n  }\r\n\r\n  @Input()\r\n  goBack = ()=>{\r\n    this.navCtrl.back();\r\n  }\r\n} \r\n","<ion-toolbar *ngIf=\"chat?.isAvatarShow==false\">\r\n    <ion-buttons slot=\"start\">\r\n        <ion-button (click)=\"back()\">\r\n            <ion-icon name=\"chevron-back-outline\"></ion-icon>\r\n        </ion-button>\r\n      </ion-buttons>\r\n    <ion-title>\r\n      <div class=\"title-avatar-area\">\r\n        <div class=\"avatar-img\" (click)=\"chat.showAvatar()\">\r\n          <img *ngIf=\"chat?.role?.get('avatar')||chat?.role?.get('thumb')\" [src]=\"chat?.role?.get('avatar')||chat?.role?.get('thumb')\" />\r\n          <ion-icon *ngIf=\"chat?.role?.get('avatarConfig')\" name=\"resize-outline\"></ion-icon>\r\n        </div>\r\n        <span (click)=\"chat.showAvatar()\">\r\n          {{chat?.role?.get(\"name\")}}\r\n        </span>\r\n      </div>\r\n    </ion-title>\r\n\r\n    <ion-buttons slot=\"end\">\r\n        <ion-button (click)=\"isModalOpen=true\"> <ion-icon name=\"ellipsis-horizontal-outline\"></ion-icon> </ion-button>\r\n    </ion-buttons>\r\n</ion-toolbar>\r\n\r\n<div class=\"avatar-area\" *ngIf=\"chat?.isAvatarShow==true\">\r\n  <fm-avatar-role-image *ngIf=\"chat?.avatarMode=='image'\" [fmodeChat]=\"chat\" [role]=\"avatarRole\"></fm-avatar-role-image>\r\n  <fm-avatar-role-video [goBack]=\"goBack\" *ngIf=\"chat?.avatarMode=='video'\" [fmodeChat]=\"chat\" [role]=\"avatarRole\"></fm-avatar-role-video>\r\n</div>\r\n\r\n\r\n<ion-modal [isOpen]=\"isModalOpen\" (willDismiss)=\"isModalOpen=false\">\r\n    <ng-template>\r\n      <ion-header>\r\n        <ion-toolbar>\r\n          <ion-buttons slot=\"start\">\r\n            <ion-button (click)=\"isModalOpen=false\">返回</ion-button>\r\n          </ion-buttons>\r\n          <ion-title>简介</ion-title>\r\n        </ion-toolbar>\r\n      </ion-header>\r\n      <ion-content class=\"ion-padding\">\r\n\r\n        <ion-card style=\"margin: 0px;\">\r\n          <img style=\"width:100%\" [src]=\"chat?.role?.get('thumb')\" alt=\"\">\r\n          <ion-card-header>\r\n            <ion-card-subtitle>{{chat?.role?.get('tags')}}</ion-card-subtitle>\r\n            <ion-card-title>{{chat?.role?.get(\"name\")}}</ion-card-title>\r\n          </ion-card-header>\r\n        \r\n          <ion-card-content>\r\n            {{chat?.role?.get(\"desc\")}}\r\n    \r\n                <ion-list [inset]=\"true\" style=\"margin:0px;\">\r\n                  <!-- <ion-item>\r\n                    <ion-avatar *ngIf=\"chat?.role?.get('thumb')\" aria-hidden=\"true\" slot=\"start\">\r\n                      <img [src]=\"chat?.role?.get('thumb')\" />\r\n                    </ion-avatar>\r\n                    <ion-label>{{chat?.role?.get(\"name\")}}</ion-label>\r\n                  </ion-item> -->\r\n                  <ion-item lines=\"none\" *ngIf=\"chat?.role?.get('age')\">\r\n                    <ion-note slot=\"start\">年龄</ion-note>\r\n                    <ion-label>{{chat?.role?.get(\"age\")}}</ion-label>\r\n                  </ion-item>\r\n                  <ion-item lines=\"none\" *ngIf=\"chat?.role?.get('gender')\">\r\n                    <ion-note slot=\"start\">性别</ion-note>\r\n                    <ion-label>{{chat?.role?.get(\"gender\")}}</ion-label>\r\n                  </ion-item>\r\n                  <ion-item lines=\"none\">\r\n                    <ion-note slot=\"start\">称号</ion-note>\r\n                    <ion-label>{{chat?.role?.get(\"title\")}}</ion-label>\r\n                  </ion-item>\r\n            \r\n                  <ion-item lines=\"none\">\r\n                    <ion-note slot=\"start\">擅长</ion-note>\r\n                    <ion-label>{{chat?.role?.get(\"tags\")?.join(',')}}</ion-label>\r\n                  </ion-item>\r\n    \r\n                </ion-list>\r\n              </ion-card-content>\r\n        </ion-card>\r\n      </ion-content>\r\n    </ng-template>\r\n  </ion-modal>"]}
|
|
@@ -86,7 +86,7 @@ export class FmChatModalInput {
|
|
|
86
86
|
let that = this;
|
|
87
87
|
this.chat.focusUserInput = () => {
|
|
88
88
|
that.chat.isVoiceInputMode = false;
|
|
89
|
-
that.userInputComp
|
|
89
|
+
that.userInputComp?.setFocus();
|
|
90
90
|
};
|
|
91
91
|
}
|
|
92
92
|
// ngAfterViewInit(): void {
|
|
@@ -329,4 +329,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
329
329
|
}], role: [{
|
|
330
330
|
type: Input
|
|
331
331
|
}] } });
|
|
332
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/chat/chat-modal-input/modal-input.component.ts","../../../../../../../projects/fmode-ng/src/lib/aigc/chat/chat-modal-input/modal-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit,AfterViewInit, Output, ViewChild } from '@angular/core';\r\nimport { ActivatedRoute,Router, RouterModule } from '@angular/router';\r\nimport { AlertController, NavController, Platform, ToastController } from '@ionic/angular';\r\nimport { FmodeChat } from '../../service-fmai/service-chat';\r\nimport { ChatService } from '../../service-fmai/service-chat';\r\n\r\n// import { AuthService } from 'fmode-ng';\r\nimport {FmodeParse,FmodeObject,FmodeQuery,FmodeUser} from \"../../../core/parse\";const Parse = FmodeParse.with(\"nova\");\r\nimport { ImagineService } from '../../service-fmai/service-imagine/imagine.service';\r\nimport { IonButton, IonContent, IonIcon, IonInput, IonItem, IonList, IonModal, IonPopover, IonTextarea, IonToolbar, ModalController } from '@ionic/angular/standalone';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\r\n\r\nimport { ModalAudioMessageComponent } from \"./modal-audio-message/modal-audio-message.component\";\r\nimport { FmChatMessageCard } from '../chat-message-card/comp-message-card.component';\r\n\r\nimport { addIcons } from 'ionicons';\r\nimport {imageOutline,chevronBackOutline,ellipsisHorizontalOutline,chevronDownOutline,chatboxEllipsesOutline,micOutline,paperPlaneOutline,shareSocialOutline,settingsOutline, alertOutline, colorWandOutline, peopleOutline} from 'ionicons/icons';\r\nimport { AccountService } from '../../../user/account/account.service';\r\nimport { FmodeChatMessage } from '../../../core/agent/chat/interface';\r\naddIcons({colorWandOutline,peopleOutline,alertOutline,imageOutline,chevronBackOutline,ellipsisHorizontalOutline,chevronDownOutline,chatboxEllipsesOutline,micOutline,paperPlaneOutline,shareSocialOutline,settingsOutline})\r\n\r\n@Component({\r\n  selector: 'fm-chat-modal-input',\r\n  templateUrl: './modal-input.component.html',\r\n  styleUrls: ['./modal-input.component.scss'],\r\n  standalone:true,\r\n  imports:[\r\n    CommonModule,FormsModule,ReactiveFormsModule,\r\n    RouterModule,\r\n    IonToolbar,IonItem,IonButton,IonList,IonModal,\r\n    IonInput,IonIcon,IonTextarea,IonPopover,IonContent,\r\n    // \r\n    FmChatMessageCard,\r\n    ModalAudioMessageComponent\r\n  ]\r\n})\r\nexport class FmChatModalInput implements OnInit{\r\n  @ViewChild(ModalAudioMessageComponent) audioComp:ModalAudioMessageComponent\r\n  @ViewChild(\"userInput\") userInputComp:IonTextarea\r\n\r\n\r\n  @Input() chat:FmodeChat;\r\n  @Input() message:FmodeChatMessage\r\n  @Input() role:FmodeObject\r\n  user:FmodeUser\r\n  errorText:string = ``\r\n\r\n  // 语音输入控制\r\n  isAudioModal:boolean = false;\r\n  closeAudio(){\r\n    this.audioComp?.cancel();\r\n    this.isAudioModal=false;\r\n  }\r\n  audioModalHeightPoint = 0.35;\r\n  async startTalk(){\r\n    // Check if already sending\r\n    if (this.isSending) {\r\n      return false;\r\n    }\r\n\r\n    let height = document.body.clientHeight || 960;\r\n    this.audioModalHeightPoint = Number((165/height).toFixed(2))\r\n\r\n    this.chat.stopPlayingVoice();\r\n\r\n      // 弹出认证组件\r\n    let modal:any \r\n    modal = await this.modalCtrl.create({\r\n      component:ModalAudioMessageComponent,\r\n      componentProps:{\r\n        chat:this.chat,\r\n        modal:modal,\r\n        onBreakPointSet:()=>{\r\n          modal?.setCurrentBreakpoint(this.audioModalHeightPoint)\r\n      }},\r\n      breakpoints:[this.audioModalHeightPoint],\r\n      initialBreakpoint:this.audioModalHeightPoint,\r\n    })\r\n    modal.present()\r\n    // this.isAudioModal=true;\r\n  }\r\n  // account:any;\r\n  authServ:any;\r\n  constructor(\r\n    private toastCtrl:ToastController,\r\n    private alertCtrl:AlertController,\r\n    private modalCtrl:ModalController,\r\n    private router:Router,\r\n    private imagineServ:ImagineService,\r\n    // private authServ:AuthService,\r\n    public chatServ:ChatService,\r\n    private account:AccountService,\r\n  ){\r\n    // this.chatServ.getChatSession().then(()=>{\r\n    //   console.log(this.chatServ.chatList);\r\n    // })\r\n    this.user = Parse.User.current();\r\n\r\n  }\r\n  ngOnInit(){\r\n    this.loadModel();\r\n    // console.log(this.chat);\r\n    let that = this;\r\n    this.chat.focusUserInput = ()=>{\r\n      that.chat.isVoiceInputMode = false;\r\n      that.userInputComp.setFocus();\r\n    }\r\n  }\r\n\r\n  // ngAfterViewInit(): void {\r\n  //   console.log(this.chat)\r\n  // }\r\n\r\n  async loadModel(){\r\n    let model = this.chat?.role?.get(\"model\")\r\n    await this.chat.loadModelList(model)\r\n  }\r\n  \r\n  async setMessageImage(){\r\n    // let url = `https://file-cloud.fmode.cn/Svehl6FceL/20240515/061503671.jpeg`;\r\n    let url = await this.imagineServ.getimg();\r\n    this.chat.userImage = url;\r\n    console.log(this.chat?.userImage);\r\n    // content是数组，添加\r\n    // content非数组，重组\r\n  }\r\n \r\n  onInputFocus(){\r\n    // console.log(\"onInputFocus\",this.chat.scrollToBottom)\r\n    this.chat.isTexting=true;\r\n    this.chat.scrollToBottom&&this.chat.scrollToBottom();\r\n  }\r\n  \r\n  onKeyDown(event:KeyboardEvent){\r\n    // 检查是否按下了Ctrl键和Enter键\r\n    if (event.ctrlKey && event.key === 'Enter') {\r\n      // 执行相应的逻辑\r\n      console.log('Ctrl+Enter 被按下');\r\n      this.sendMessage();\r\n    }\r\n  }\r\n\r\n  // 发送消息 防抖机制\r\n  isSending = false;\r\n  lastMessageTimeout: any;\r\n  lastMessageTimestamp: number = 0;\r\n  replyTimeout = 15000; // 15 seconds timeout\r\n  async sendMessage(){\r\n    // Check if already sending\r\n    if (this.isSending) {\r\n      return false;\r\n    }\r\n     // Check if waiting for previous reply\r\n    const now = Date.now();\r\n    if (this.lastMessageTimestamp > 0 && (now - this.lastMessageTimestamp) < this.replyTimeout) {\r\n      this.errorText = `请等待上一条消息的回复或稍后再试`;\r\n      let toast = await this.toastCtrl.create({\r\n        message: this.errorText,\r\n        position: \"top\",\r\n        icon: 'alert',\r\n        color: \"warning\",\r\n        duration: 1000\r\n      });\r\n      toast.present();\r\n      return false;\r\n    }\r\n\r\n    this.isSending = true;\r\n    this.lastMessageTimestamp = Date.now();\r\n\r\n    // 检测用户登录情况\r\n    // let isLoginLock = await this.authServ.checkLoginLock()\r\n    // if(!isLoginLock) return false\r\n\r\n    // 检测余额及模型付费限制\r\n    let payCheck = await this.checkBalance()\r\n    if(!payCheck) {\r\n      this.isSending = false;\r\n      return false\r\n    }\r\n\r\n    // 检测用户输入内容空值\r\n    if(!this.chat.userInput){\r\n      this.errorText = `内容不能为空`\r\n      let toast = await this.toastCtrl.create({\r\n        message:this.errorText,\r\n        position:\"top\",\r\n        icon:'alert',\r\n        color:\"warning-circle\",\r\n        duration:1000\r\n      })\r\n      toast.present();\r\n      this.isSending = false;\r\n      return\r\n    }\r\n\r\n     // Clear any existing timeout\r\n    if (this.lastMessageTimeout) {\r\n      clearTimeout(this.lastMessageTimeout);\r\n    }\r\n    // Set timeout to reset sending state if no reply\r\n    this.lastMessageTimeout = setTimeout(() => {\r\n      this.isSending = false;\r\n      this.lastMessageTimestamp = 0;\r\n    }, this.replyTimeout);\r\n\r\n    // 正常发送消息\r\n    // this.chat.isTalkMode=true; // 开启说话模式\r\n    this.chat?.sendMessage(this.chat?.userInput,this.chat?.userImage,(msg)=>{},{\r\n      onMessageStart:(message:any)=>{\r\n         // When reply comes, clear timeout and reset state\r\n        clearTimeout(this.lastMessageTimeout);\r\n        this.isSending = false;\r\n        this.lastMessageTimestamp = 0;\r\n      },\r\n      onSSMLComplete:(voice:any)=>{\r\n        console.log(voice)\r\n      }\r\n    });\r\n    this.chat.userInput = ``\r\n    this.chat.userImage = ``\r\n  }\r\n  async checkBalance(){\r\n    let price = 10;\r\n    \r\n    // 校验账户余额\r\n    let billing = await this.account.getBilling();\r\n    if (billing?.credit?.balance >= price) {\r\n      this.chat.isDirect = true; // 余额充足，不限速\r\n    }\r\n\r\n     // 模型是否开启付费限制\r\n     if(!this.chat?.currentModel?.get?.(\"payLimit\")) {\r\n      return true\r\n    }\r\n\r\n    // let price = await this.calcPrice()\r\n    if (billing?.credit?.balance < price) {\r\n      let alert = await this.alertCtrl.create({\r\n        header:\"注意\",\r\n        subHeader:\"您的余额不足，请充值后解锁高级模型\",\r\n        buttons:[\r\n          {\r\n            role:\"cancel\",\r\n            text:\"取消\"\r\n          },\r\n          {\r\n            role:\"destructive\",\r\n            text:\"充值\",\r\n            handler:()=>{\r\n              // 待更换成原地弹窗，原地付费，避免页面跳转\r\n              this.router.navigateByUrl(\"/account/billing\")\r\n            }\r\n          }\r\n        ]\r\n      })\r\n      alert.present()\r\n      return false\r\n    }\r\n    return true\r\n  }\r\n\r\n  //查询问答内容分享数据表：ChatShare\r\n  async getChatShare(){\r\n    this.user = Parse.User.current();\r\n    let query = new Parse.Query('ChatShare');\r\n    query.equalTo('user', Parse.User.current().id);\r\n    query.equalTo('session', this.chat?.sessionId)\r\n    let chatShare = await query.first()\r\n    // console.log(chatShare);\r\n  }\r\n\r\n  async toggleChatShare() {\r\n    //获取ChatShare数据表\r\n    let query = new Parse.Query('ChatShare');\r\n    query.equalTo('user', Parse.User.current().id);\r\n    query.equalTo('role', this.chat?.role.id);\r\n    query.equalTo('session', this.chat?.sessionId);\r\n    // query.equalTo('messageList', this.chat?.messageList);\r\n    query.select('objectId')\r\n    let share = await query.first();\r\n    // console.log(share);\r\n\r\n    //如果id不存在，在数据表内创建新的记录\r\n    if(!share?.id){\r\n      let obj = Parse.Object.extend('ChatShare')\r\n      share = new obj()\r\n      share.set('user', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"_User\",\r\n        \"objectId\":  Parse.User.current()?.id\r\n      })\r\n      share.set('session', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"ChatSession\",\r\n        \"objectId\": this.chat?.sessionId\r\n      })\r\n      share.set('role', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"AvatarRole\",\r\n        \"objectId\": this.chat?.role.id\r\n      })\r\n      share.set('company', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"Company\",\r\n        \"objectId\": \"E4KpGvTEto\"\r\n      })\r\n      share.set('messageList', this.chat?.messageList)\r\n    } else {\r\n      //id存在，点击分享按钮执行更新messageList数据的操作\r\n      share.set('messageList', this.chat?.messageList)\r\n    }\r\n    await share.save();\r\n    this.getChatShare();\r\n}\r\n\r\nasync chatShareSuccessMessage() {\r\n    let toast = await this.toastCtrl.create({\r\n      duration:1000,\r\n      message:\"分享成功\",\r\n      color:'primary',\r\n      icon:\"information-circle\",\r\n      position:\"top\"\r\n    })\r\n    toast.present();\r\n    return\r\n}\r\n\r\n  //分享弹框组件\r\n  isShare = false;\r\n\r\n  showShare(): void {\r\n    this.isShare = true;\r\n  }\r\n\r\n  handleOkShare(): void {\r\n    // console.log(this.chat);\r\n    // console.log(this.chat?.sessionId);\r\n    // console.log(this.chat?.messageList);\r\n    // console.log(this.chat?.role.id);\r\n    this.toggleChatShare();\r\n    this.chatShareSuccessMessage();\r\n    this.isShare = false;\r\n  }\r\n\r\n  handleCancelShare(): void {\r\n    this.isShare = false;\r\n  }\r\n}\r\n","<ion-toolbar>\r\n    <ion-item class=\"button-item\" lines=\"none\">\r\n        <!-- 设置 -->\r\n        <!-- <ion-button fill=\"outline\" slot=\"start\">\r\n            <ion-icon name=\"settings-outline\"></ion-icon> \r\n        </ion-button> -->\r\n        <ng-container *ngFor=\"let button of chat?.leftButtons\">\r\n          <ion-button style=\"--padding-start:10px;--padding-end:10px;\"\r\n           shape=\"round\" *ngIf=\"button?.show&&button?.show()\" fill=\"outline\" [title]=\"button?.title\" slot=\"start\" (click)=\"button.onClick()\">\r\n            <ion-icon [name]=\"button?.icon\" [slot]=\"button?.showTitle?'start':'icon-only'\"></ion-icon>\r\n            {{button?.showTitle&&button?.title}}\r\n          </ion-button>\r\n        </ng-container>\r\n\r\n        <ng-container *ngFor=\"let button of chat?.role?.get('buttons')\">\r\n            <ion-button shape=\"round\" (click)=\"chatServ.doButtonAction(button)\" fill=\"outline\"  slot=\"start\">\r\n                {{button?.name}}\r\n            </ion-button>\r\n        </ng-container>\r\n\r\n        <!--分享按钮-->\r\n        @if(!chat?.hideShare){\r\n          <ion-button shape=\"round\" *ngIf=\"chat?.messageList?.length>1\" (click)=\"showShare()\" fill=\"outline\" title=\"分享\" slot=\"end\">\r\n            <ion-icon name=\"share-social-outline\"></ion-icon>\r\n          </ion-button>\r\n        }\r\n          <ion-modal [isOpen]=\"isShare\">\r\n            <ng-template>\r\n              <ion-header>\r\n                <ion-toolbar>\r\n                  <ion-buttons slot=\"start\">\r\n                    <ion-button (click)=\"handleCancelShare()\">取消</ion-button>\r\n                  </ion-buttons>\r\n                  <ion-title>对话分享</ion-title>\r\n                  <ion-buttons slot=\"end\">\r\n                    <ion-button (click)=\"handleOkShare()\">分享</ion-button>\r\n                  </ion-buttons>\r\n                </ion-toolbar>\r\n              </ion-header>\r\n              <ion-content class=\"ion-padding\">\r\n                <ng-container *ngFor=\"let message of chat?.messageList;let index=index;\">\r\n                    <!-- 内容格式化区域 -->\r\n                    <fm-chat-message-card [chat]=\"chat\" *ngIf=\"!message?.hidden\" [index]=\"index\" [message]=\"message\" [role]=\"chat?.role\"></fm-chat-message-card>\r\n                </ng-container>\r\n\r\n                <div *ngIf=\"false\" class=\"popup-content\">\r\n                  <div *ngFor=\"let message of chat?.messageList\">\r\n                    <!-- 头像 -->\r\n                    <div class=\"item-row user\" *ngIf=\"message?.role!='system'\">\r\n                      <div>\r\n                        <img class=\"avatar\" *ngIf=\"message?.role!='user'\" [src]=\"(chat?.role?.get('avatar') || chat?.role?.get('thumb') || 'https://file-cloud.fmode.cn/E4KpGvTEto/20230930/l413e6090731854.png')+'?'+'x-image-process=image/resize,m_fixed,w_100'+'&imageView2/1/w/32/h/32'\" >\r\n                      </div>\r\n                      <div class=\"user-question\">\r\n                        <app-comp-user-avatar [user]=\"user\" *ngIf=\"message?.role=='user'\"></app-comp-user-avatar>\r\n                      </div>\r\n                    </div>\r\n                    <!-- 内容 -->\r\n                    <div class=\"message-wrapper\">\r\n                      <div class=\"message-content-user\">\r\n                        <div class=\"user-message\" *ngIf=\"message?.role === 'user'\">\r\n                          <div class=\"item-content\">\r\n                            <!-- <fm-markdown-preview *ngIf=\"!message?.complete\" class=\"content-style\" [content]=\"message?.content\" [render]=\"false\"></fm-markdown-preview> -->\r\n                            <fm-markdown-preview *ngIf=\"message?.complete\" class=\"content-style\" [content]=\"message?.content\"></fm-markdown-preview>\r\n                          </div>\r\n                        </div>\r\n                      </div>\r\n                      <div class=\"message-content-role\">\r\n                        <div class=\"role-message\" *ngIf=\"message?.role !== 'user' && message?.role !== 'system'\">\r\n                          <div class=\"item-content\">\r\n                            <!-- <fm-markdown-preview *ngIf=\"!message?.complete\" class=\"content-style\" [content]=\"message?.content\" [render]=\"false\"></fm-markdown-preview> -->\r\n                            <fm-markdown-preview *ngIf=\"message?.complete\" class=\"content-style\" [content]=\"message?.content\"></fm-markdown-preview>\r\n                          </div>\r\n                        </div>\r\n                      </div>\r\n                      <div class=\"message-content-system\">\r\n                        <div class=\"system-message\" *ngIf=\"message?.role === 'system'\">\r\n                          <div class=\"item-content\">\r\n                            <!-- <fm-markdown-preview *ngIf=\"!message?.complete\" class=\"content-style\" [content]=\"message?.content\" [render]=\"false\"></fm-markdown-preview> -->\r\n                            <fm-markdown-preview *ngIf=\"message?.complete\" class=\"content-style\" [content]=\"message?.content\"></fm-markdown-preview>\r\n                          </div>\r\n                        </div>\r\n                      </div>\r\n                    </div>\r\n                    <div class=\"chat-time\" *ngIf=\"message?.createdAt\" [ngClass]=\"{'role-time': message?.role !== 'user'}\">\r\n                      <span>{{message?.createdAt | date:\"dd/MM/yyyy, HH/mm/ss a\"}}</span>\r\n                    </div>\r\n                  </div>\r\n                </div>   \r\n              </ion-content>\r\n            </ng-template>\r\n          </ion-modal>\r\n\r\n        <!-- 图片 -->\r\n        <ion-button shape=\"round\" *ngIf=\"chat?.currentModel?.get?.('config')?.imageEnabled\" fill=\"outline\"  slot=\"end\" (click)=\"setMessageImage()\">\r\n          <ion-icon name=\"image-outline\"></ion-icon>\r\n        </ion-button>\r\n        <!-- 模型 -->\r\n        @if(!chat?.hideModalSelect){\r\n          <ion-button shape=\"round\" fill=\"outline\" slot=\"end\"  id=\"model-button\">\r\n              <ion-icon name=\"chevron-down-outline\"></ion-icon>\r\n              {{chat?.currentModel?.get&&chat?.currentModel?.get?.(\"name\")||\"Fmode-C1.0-128k\"}}\r\n          </ion-button>\r\n          <ion-popover trigger=\"model-button\" [dismissOnSelect]=\"true\">\r\n              <ng-template>\r\n                <ion-content>\r\n                  <ion-list>\r\n                      <ng-container *ngFor=\"let model of chat.modelList\">\r\n                          <ion-item (click)=\"chat.currentModel = model\" [button]=\"true\" [detail]=\"false\">\r\n                            {{model?.get(\"name\")}}\r\n                            <ion-note slot=\"end\">{{model?.get(\"credit\")}}/k</ion-note>\r\n                          </ion-item>\r\n                      </ng-container>\r\n                  </ion-list>\r\n                </ion-content>\r\n              </ng-template>\r\n            </ion-popover>\r\n          }\r\n\r\n    </ion-item>\r\n\r\n    <ion-item class=\"input-item\" lines=\"none\">\r\n        <!-- 语音消息输入 -->\r\n        <ng-container *ngIf=\"chat?.isVoiceInputMode\">\r\n          <!-- 切换文本输入 -->\r\n          <ion-button class=\"btn-input-change\" color=\"primary\" (click)=\"chat.isVoiceInputMode=false\" shape=\"round\" size=\"large\">\r\n            <ion-icon name=\"chatbox-ellipses-outline\" slot=\"icon-only\"></ion-icon>\r\n          </ion-button>\r\n          \r\n          <div class=\"btn-voice-start\" (click)=\"startTalk()\" [class.disabled]=\"isSending\">\r\n            <span>\r\n              点击讲话\r\n            </span> \r\n          </div>\r\n        </ng-container>\r\n\r\n        <!-- 文本消息输入 -->\r\n        <ng-container *ngIf=\"!chat?.isVoiceInputMode\">\r\n          <!-- 切换语音输入 -->\r\n          <ion-button [style.display]=\"chat.isTexting?'none':'flex'\" class=\"btn-input-change\" color=\"primary\" *ngIf=\"chat?.role?.get('voiceConfig')\" (click)=\"chat.isVoiceInputMode=true\" shape=\"round\" size=\"large\">\r\n            <ion-icon name=\"mic-outline\" slot=\"icon-only\"></ion-icon>\r\n          </ion-button>\r\n\r\n          <!-- 文本输入区域 -->\r\n          <ion-textarea\r\n          #userInput\r\n          *ngIf=\"chat\" (keydown)=\"onKeyDown($event)\"\r\n          [errorText]=\"errorText\"\r\n          [(ngModel)]=\"chat.userInput\"\r\n          (ionFocus)=\"onInputFocus()\"\r\n          (ionBlur)=\"chat.isTexting=false\"\r\n          [autoGrow]=\"true\" shape=\"round\" fill=\"outline\"\r\n          label=\"Ctrl + Enter 发送消息\" placeholder=\"请输入您的提示词\"\r\n          labelPlacement=\"floating\"></ion-textarea>\r\n          \r\n          <!-- 文本发送按钮 -->\r\n          <ion-button [disabled]=\"isSending\"\r\n          color=\"primary\" shape=\"round\" size=\"large\" (click)=\"sendMessage()\">\r\n            <ion-icon name=\"paper-plane-outline\" slot=\"icon-only\"></ion-icon>\r\n          </ion-button>\r\n        </ng-container>\r\n    </ion-item>\r\n</ion-toolbar>\r\n\r\n\r\n<!-- 语音消息输入：弹出区域 -->\r\n<!-- <ion-modal #audioModal [isOpen]=\"isAudioModal\" (willDismiss)=\"closeAudio()\" [initialBreakpoint]=\"audioModalHeightPoint\" [breakpoints]=\"[0, audioModalHeightPoint]\">\r\n  <ng-template>\r\n    <fm-modal-audio-message #audioComp *ngIf=\"isAudioModal\" [chat]=\"chat\" [modal]=\"audioModal\"></fm-modal-audio-message>\r\n  </ng-template>\r\n</ion-modal> -->"]}
|
|
332
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/chat/chat-modal-input/modal-input.component.ts","../../../../../../../projects/fmode-ng/src/lib/aigc/chat/chat-modal-input/modal-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnInit,AfterViewInit, Output, ViewChild } from '@angular/core';\r\nimport { ActivatedRoute,Router, RouterModule } from '@angular/router';\r\nimport { AlertController, NavController, Platform, ToastController } from '@ionic/angular';\r\nimport { FmodeChat } from '../../service-fmai/service-chat';\r\nimport { ChatService } from '../../service-fmai/service-chat';\r\n\r\n// import { AuthService } from 'fmode-ng';\r\nimport {FmodeParse,FmodeObject,FmodeQuery,FmodeUser} from \"../../../core/parse\";const Parse = FmodeParse.with(\"nova\");\r\nimport { ImagineService } from '../../service-fmai/service-imagine/imagine.service';\r\nimport { IonButton, IonContent, IonIcon, IonInput, IonItem, IonList, IonModal, IonPopover, IonTextarea, IonToolbar, ModalController } from '@ionic/angular/standalone';\r\nimport { CommonModule } from '@angular/common';\r\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\r\n\r\nimport { ModalAudioMessageComponent } from \"./modal-audio-message/modal-audio-message.component\";\r\nimport { FmChatMessageCard } from '../chat-message-card/comp-message-card.component';\r\n\r\nimport { addIcons } from 'ionicons';\r\nimport {imageOutline,chevronBackOutline,ellipsisHorizontalOutline,chevronDownOutline,chatboxEllipsesOutline,micOutline,paperPlaneOutline,shareSocialOutline,settingsOutline, alertOutline, colorWandOutline, peopleOutline} from 'ionicons/icons';\r\nimport { AccountService } from '../../../user/account/account.service';\r\nimport { FmodeChatMessage } from '../../../core/agent/chat/interface';\r\naddIcons({colorWandOutline,peopleOutline,alertOutline,imageOutline,chevronBackOutline,ellipsisHorizontalOutline,chevronDownOutline,chatboxEllipsesOutline,micOutline,paperPlaneOutline,shareSocialOutline,settingsOutline})\r\n\r\n@Component({\r\n  selector: 'fm-chat-modal-input',\r\n  templateUrl: './modal-input.component.html',\r\n  styleUrls: ['./modal-input.component.scss'],\r\n  standalone:true,\r\n  imports:[\r\n    CommonModule,FormsModule,ReactiveFormsModule,\r\n    RouterModule,\r\n    IonToolbar,IonItem,IonButton,IonList,IonModal,\r\n    IonInput,IonIcon,IonTextarea,IonPopover,IonContent,\r\n    // \r\n    FmChatMessageCard,\r\n    ModalAudioMessageComponent\r\n  ]\r\n})\r\nexport class FmChatModalInput implements OnInit{\r\n  @ViewChild(ModalAudioMessageComponent) audioComp:ModalAudioMessageComponent\r\n  @ViewChild(\"userInput\") userInputComp:IonTextarea\r\n\r\n\r\n  @Input() chat:FmodeChat;\r\n  @Input() message:FmodeChatMessage\r\n  @Input() role:FmodeObject\r\n  user:FmodeUser\r\n  errorText:string = ``\r\n\r\n  // 语音输入控制\r\n  isAudioModal:boolean = false;\r\n  closeAudio(){\r\n    this.audioComp?.cancel();\r\n    this.isAudioModal=false;\r\n  }\r\n  audioModalHeightPoint = 0.35;\r\n  async startTalk(){\r\n    // Check if already sending\r\n    if (this.isSending) {\r\n      return false;\r\n    }\r\n\r\n    let height = document.body.clientHeight || 960;\r\n    this.audioModalHeightPoint = Number((165/height).toFixed(2))\r\n\r\n    this.chat.stopPlayingVoice();\r\n\r\n      // 弹出认证组件\r\n    let modal:any \r\n    modal = await this.modalCtrl.create({\r\n      component:ModalAudioMessageComponent,\r\n      componentProps:{\r\n        chat:this.chat,\r\n        modal:modal,\r\n        onBreakPointSet:()=>{\r\n          modal?.setCurrentBreakpoint(this.audioModalHeightPoint)\r\n      }},\r\n      breakpoints:[this.audioModalHeightPoint],\r\n      initialBreakpoint:this.audioModalHeightPoint,\r\n    })\r\n    modal.present()\r\n    // this.isAudioModal=true;\r\n  }\r\n  // account:any;\r\n  authServ:any;\r\n  constructor(\r\n    private toastCtrl:ToastController,\r\n    private alertCtrl:AlertController,\r\n    private modalCtrl:ModalController,\r\n    private router:Router,\r\n    private imagineServ:ImagineService,\r\n    // private authServ:AuthService,\r\n    public chatServ:ChatService,\r\n    private account:AccountService,\r\n  ){\r\n    // this.chatServ.getChatSession().then(()=>{\r\n    //   console.log(this.chatServ.chatList);\r\n    // })\r\n    this.user = Parse.User.current();\r\n\r\n  }\r\n  ngOnInit(){\r\n    this.loadModel();\r\n    // console.log(this.chat);\r\n    let that = this;\r\n    this.chat.focusUserInput = ()=>{\r\n      that.chat.isVoiceInputMode = false;\r\n      that.userInputComp?.setFocus();\r\n    }\r\n  }\r\n\r\n  // ngAfterViewInit(): void {\r\n  //   console.log(this.chat)\r\n  // }\r\n\r\n  async loadModel(){\r\n    let model = this.chat?.role?.get(\"model\")\r\n    await this.chat.loadModelList(model)\r\n  }\r\n  \r\n  async setMessageImage(){\r\n    // let url = `https://file-cloud.fmode.cn/Svehl6FceL/20240515/061503671.jpeg`;\r\n    let url = await this.imagineServ.getimg();\r\n    this.chat.userImage = url;\r\n    console.log(this.chat?.userImage);\r\n    // content是数组，添加\r\n    // content非数组，重组\r\n  }\r\n \r\n  onInputFocus(){\r\n    // console.log(\"onInputFocus\",this.chat.scrollToBottom)\r\n    this.chat.isTexting=true;\r\n    this.chat.scrollToBottom&&this.chat.scrollToBottom();\r\n  }\r\n  \r\n  onKeyDown(event:KeyboardEvent){\r\n    // 检查是否按下了Ctrl键和Enter键\r\n    if (event.ctrlKey && event.key === 'Enter') {\r\n      // 执行相应的逻辑\r\n      console.log('Ctrl+Enter 被按下');\r\n      this.sendMessage();\r\n    }\r\n  }\r\n\r\n  // 发送消息 防抖机制\r\n  isSending = false;\r\n  lastMessageTimeout: any;\r\n  lastMessageTimestamp: number = 0;\r\n  replyTimeout = 15000; // 15 seconds timeout\r\n  async sendMessage(){\r\n    // Check if already sending\r\n    if (this.isSending) {\r\n      return false;\r\n    }\r\n     // Check if waiting for previous reply\r\n    const now = Date.now();\r\n    if (this.lastMessageTimestamp > 0 && (now - this.lastMessageTimestamp) < this.replyTimeout) {\r\n      this.errorText = `请等待上一条消息的回复或稍后再试`;\r\n      let toast = await this.toastCtrl.create({\r\n        message: this.errorText,\r\n        position: \"top\",\r\n        icon: 'alert',\r\n        color: \"warning\",\r\n        duration: 1000\r\n      });\r\n      toast.present();\r\n      return false;\r\n    }\r\n\r\n    this.isSending = true;\r\n    this.lastMessageTimestamp = Date.now();\r\n\r\n    // 检测用户登录情况\r\n    // let isLoginLock = await this.authServ.checkLoginLock()\r\n    // if(!isLoginLock) return false\r\n\r\n    // 检测余额及模型付费限制\r\n    let payCheck = await this.checkBalance()\r\n    if(!payCheck) {\r\n      this.isSending = false;\r\n      return false\r\n    }\r\n\r\n    // 检测用户输入内容空值\r\n    if(!this.chat.userInput){\r\n      this.errorText = `内容不能为空`\r\n      let toast = await this.toastCtrl.create({\r\n        message:this.errorText,\r\n        position:\"top\",\r\n        icon:'alert',\r\n        color:\"warning-circle\",\r\n        duration:1000\r\n      })\r\n      toast.present();\r\n      this.isSending = false;\r\n      return\r\n    }\r\n\r\n     // Clear any existing timeout\r\n    if (this.lastMessageTimeout) {\r\n      clearTimeout(this.lastMessageTimeout);\r\n    }\r\n    // Set timeout to reset sending state if no reply\r\n    this.lastMessageTimeout = setTimeout(() => {\r\n      this.isSending = false;\r\n      this.lastMessageTimestamp = 0;\r\n    }, this.replyTimeout);\r\n\r\n    // 正常发送消息\r\n    // this.chat.isTalkMode=true; // 开启说话模式\r\n    this.chat?.sendMessage(this.chat?.userInput,this.chat?.userImage,(msg)=>{},{\r\n      onMessageStart:(message:any)=>{\r\n         // When reply comes, clear timeout and reset state\r\n        clearTimeout(this.lastMessageTimeout);\r\n        this.isSending = false;\r\n        this.lastMessageTimestamp = 0;\r\n      },\r\n      onSSMLComplete:(voice:any)=>{\r\n        console.log(voice)\r\n      }\r\n    });\r\n    this.chat.userInput = ``\r\n    this.chat.userImage = ``\r\n  }\r\n  async checkBalance(){\r\n    let price = 10;\r\n    \r\n    // 校验账户余额\r\n    let billing = await this.account.getBilling();\r\n    if (billing?.credit?.balance >= price) {\r\n      this.chat.isDirect = true; // 余额充足，不限速\r\n    }\r\n\r\n     // 模型是否开启付费限制\r\n     if(!this.chat?.currentModel?.get?.(\"payLimit\")) {\r\n      return true\r\n    }\r\n\r\n    // let price = await this.calcPrice()\r\n    if (billing?.credit?.balance < price) {\r\n      let alert = await this.alertCtrl.create({\r\n        header:\"注意\",\r\n        subHeader:\"您的余额不足，请充值后解锁高级模型\",\r\n        buttons:[\r\n          {\r\n            role:\"cancel\",\r\n            text:\"取消\"\r\n          },\r\n          {\r\n            role:\"destructive\",\r\n            text:\"充值\",\r\n            handler:()=>{\r\n              // 待更换成原地弹窗，原地付费，避免页面跳转\r\n              this.router.navigateByUrl(\"/account/billing\")\r\n            }\r\n          }\r\n        ]\r\n      })\r\n      alert.present()\r\n      return false\r\n    }\r\n    return true\r\n  }\r\n\r\n  //查询问答内容分享数据表：ChatShare\r\n  async getChatShare(){\r\n    this.user = Parse.User.current();\r\n    let query = new Parse.Query('ChatShare');\r\n    query.equalTo('user', Parse.User.current().id);\r\n    query.equalTo('session', this.chat?.sessionId)\r\n    let chatShare = await query.first()\r\n    // console.log(chatShare);\r\n  }\r\n\r\n  async toggleChatShare() {\r\n    //获取ChatShare数据表\r\n    let query = new Parse.Query('ChatShare');\r\n    query.equalTo('user', Parse.User.current().id);\r\n    query.equalTo('role', this.chat?.role.id);\r\n    query.equalTo('session', this.chat?.sessionId);\r\n    // query.equalTo('messageList', this.chat?.messageList);\r\n    query.select('objectId')\r\n    let share = await query.first();\r\n    // console.log(share);\r\n\r\n    //如果id不存在，在数据表内创建新的记录\r\n    if(!share?.id){\r\n      let obj = Parse.Object.extend('ChatShare')\r\n      share = new obj()\r\n      share.set('user', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"_User\",\r\n        \"objectId\":  Parse.User.current()?.id\r\n      })\r\n      share.set('session', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"ChatSession\",\r\n        \"objectId\": this.chat?.sessionId\r\n      })\r\n      share.set('role', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"AvatarRole\",\r\n        \"objectId\": this.chat?.role.id\r\n      })\r\n      share.set('company', {\r\n        \"__type\": \"Pointer\",\r\n        \"className\": \"Company\",\r\n        \"objectId\": \"E4KpGvTEto\"\r\n      })\r\n      share.set('messageList', this.chat?.messageList)\r\n    } else {\r\n      //id存在，点击分享按钮执行更新messageList数据的操作\r\n      share.set('messageList', this.chat?.messageList)\r\n    }\r\n    await share.save();\r\n    this.getChatShare();\r\n}\r\n\r\nasync chatShareSuccessMessage() {\r\n    let toast = await this.toastCtrl.create({\r\n      duration:1000,\r\n      message:\"分享成功\",\r\n      color:'primary',\r\n      icon:\"information-circle\",\r\n      position:\"top\"\r\n    })\r\n    toast.present();\r\n    return\r\n}\r\n\r\n  //分享弹框组件\r\n  isShare = false;\r\n\r\n  showShare(): void {\r\n    this.isShare = true;\r\n  }\r\n\r\n  handleOkShare(): void {\r\n    // console.log(this.chat);\r\n    // console.log(this.chat?.sessionId);\r\n    // console.log(this.chat?.messageList);\r\n    // console.log(this.chat?.role.id);\r\n    this.toggleChatShare();\r\n    this.chatShareSuccessMessage();\r\n    this.isShare = false;\r\n  }\r\n\r\n  handleCancelShare(): void {\r\n    this.isShare = false;\r\n  }\r\n}\r\n","<ion-toolbar>\r\n    <ion-item class=\"button-item\" lines=\"none\">\r\n        <!-- 设置 -->\r\n        <!-- <ion-button fill=\"outline\" slot=\"start\">\r\n            <ion-icon name=\"settings-outline\"></ion-icon> \r\n        </ion-button> -->\r\n        <ng-container *ngFor=\"let button of chat?.leftButtons\">\r\n          <ion-button style=\"--padding-start:10px;--padding-end:10px;\"\r\n           shape=\"round\" *ngIf=\"button?.show&&button?.show()\" fill=\"outline\" [title]=\"button?.title\" slot=\"start\" (click)=\"button.onClick()\">\r\n            <ion-icon [name]=\"button?.icon\" [slot]=\"button?.showTitle?'start':'icon-only'\"></ion-icon>\r\n            {{button?.showTitle&&button?.title}}\r\n          </ion-button>\r\n        </ng-container>\r\n\r\n        <ng-container *ngFor=\"let button of chat?.role?.get('buttons')\">\r\n            <ion-button shape=\"round\" (click)=\"chatServ.doButtonAction(button)\" fill=\"outline\"  slot=\"start\">\r\n                {{button?.name}}\r\n            </ion-button>\r\n        </ng-container>\r\n\r\n        <!--分享按钮-->\r\n        @if(!chat?.hideShare){\r\n          <ion-button shape=\"round\" *ngIf=\"chat?.messageList?.length>1\" (click)=\"showShare()\" fill=\"outline\" title=\"分享\" slot=\"end\">\r\n            <ion-icon name=\"share-social-outline\"></ion-icon>\r\n          </ion-button>\r\n        }\r\n          <ion-modal [isOpen]=\"isShare\">\r\n            <ng-template>\r\n              <ion-header>\r\n                <ion-toolbar>\r\n                  <ion-buttons slot=\"start\">\r\n                    <ion-button (click)=\"handleCancelShare()\">取消</ion-button>\r\n                  </ion-buttons>\r\n                  <ion-title>对话分享</ion-title>\r\n                  <ion-buttons slot=\"end\">\r\n                    <ion-button (click)=\"handleOkShare()\">分享</ion-button>\r\n                  </ion-buttons>\r\n                </ion-toolbar>\r\n              </ion-header>\r\n              <ion-content class=\"ion-padding\">\r\n                <ng-container *ngFor=\"let message of chat?.messageList;let index=index;\">\r\n                    <!-- 内容格式化区域 -->\r\n                    <fm-chat-message-card [chat]=\"chat\" *ngIf=\"!message?.hidden\" [index]=\"index\" [message]=\"message\" [role]=\"chat?.role\"></fm-chat-message-card>\r\n                </ng-container>\r\n\r\n                <div *ngIf=\"false\" class=\"popup-content\">\r\n                  <div *ngFor=\"let message of chat?.messageList\">\r\n                    <!-- 头像 -->\r\n                    <div class=\"item-row user\" *ngIf=\"message?.role!='system'\">\r\n                      <div>\r\n                        <img class=\"avatar\" *ngIf=\"message?.role!='user'\" [src]=\"(chat?.role?.get('avatar') || chat?.role?.get('thumb') || 'https://file-cloud.fmode.cn/E4KpGvTEto/20230930/l413e6090731854.png')+'?'+'x-image-process=image/resize,m_fixed,w_100'+'&imageView2/1/w/32/h/32'\" >\r\n                      </div>\r\n                      <div class=\"user-question\">\r\n                        <app-comp-user-avatar [user]=\"user\" *ngIf=\"message?.role=='user'\"></app-comp-user-avatar>\r\n                      </div>\r\n                    </div>\r\n                    <!-- 内容 -->\r\n                    <div class=\"message-wrapper\">\r\n                      <div class=\"message-content-user\">\r\n                        <div class=\"user-message\" *ngIf=\"message?.role === 'user'\">\r\n                          <div class=\"item-content\">\r\n                            <!-- <fm-markdown-preview *ngIf=\"!message?.complete\" class=\"content-style\" [content]=\"message?.content\" [render]=\"false\"></fm-markdown-preview> -->\r\n                            <fm-markdown-preview *ngIf=\"message?.complete\" class=\"content-style\" [content]=\"message?.content\"></fm-markdown-preview>\r\n                          </div>\r\n                        </div>\r\n                      </div>\r\n                      <div class=\"message-content-role\">\r\n                        <div class=\"role-message\" *ngIf=\"message?.role !== 'user' && message?.role !== 'system'\">\r\n                          <div class=\"item-content\">\r\n                            <!-- <fm-markdown-preview *ngIf=\"!message?.complete\" class=\"content-style\" [content]=\"message?.content\" [render]=\"false\"></fm-markdown-preview> -->\r\n                            <fm-markdown-preview *ngIf=\"message?.complete\" class=\"content-style\" [content]=\"message?.content\"></fm-markdown-preview>\r\n                          </div>\r\n                        </div>\r\n                      </div>\r\n                      <div class=\"message-content-system\">\r\n                        <div class=\"system-message\" *ngIf=\"message?.role === 'system'\">\r\n                          <div class=\"item-content\">\r\n                            <!-- <fm-markdown-preview *ngIf=\"!message?.complete\" class=\"content-style\" [content]=\"message?.content\" [render]=\"false\"></fm-markdown-preview> -->\r\n                            <fm-markdown-preview *ngIf=\"message?.complete\" class=\"content-style\" [content]=\"message?.content\"></fm-markdown-preview>\r\n                          </div>\r\n                        </div>\r\n                      </div>\r\n                    </div>\r\n                    <div class=\"chat-time\" *ngIf=\"message?.createdAt\" [ngClass]=\"{'role-time': message?.role !== 'user'}\">\r\n                      <span>{{message?.createdAt | date:\"dd/MM/yyyy, HH/mm/ss a\"}}</span>\r\n                    </div>\r\n                  </div>\r\n                </div>   \r\n              </ion-content>\r\n            </ng-template>\r\n          </ion-modal>\r\n\r\n        <!-- 图片 -->\r\n        <ion-button shape=\"round\" *ngIf=\"chat?.currentModel?.get?.('config')?.imageEnabled\" fill=\"outline\"  slot=\"end\" (click)=\"setMessageImage()\">\r\n          <ion-icon name=\"image-outline\"></ion-icon>\r\n        </ion-button>\r\n        <!-- 模型 -->\r\n        @if(!chat?.hideModalSelect){\r\n          <ion-button shape=\"round\" fill=\"outline\" slot=\"end\"  id=\"model-button\">\r\n              <ion-icon name=\"chevron-down-outline\"></ion-icon>\r\n              {{chat?.currentModel?.get&&chat?.currentModel?.get?.(\"name\")||\"Fmode-C1.0-128k\"}}\r\n          </ion-button>\r\n          <ion-popover trigger=\"model-button\" [dismissOnSelect]=\"true\">\r\n              <ng-template>\r\n                <ion-content>\r\n                  <ion-list>\r\n                      <ng-container *ngFor=\"let model of chat.modelList\">\r\n                          <ion-item (click)=\"chat.currentModel = model\" [button]=\"true\" [detail]=\"false\">\r\n                            {{model?.get(\"name\")}}\r\n                            <ion-note slot=\"end\">{{model?.get(\"credit\")}}/k</ion-note>\r\n                          </ion-item>\r\n                      </ng-container>\r\n                  </ion-list>\r\n                </ion-content>\r\n              </ng-template>\r\n            </ion-popover>\r\n          }\r\n\r\n    </ion-item>\r\n\r\n    <ion-item class=\"input-item\" lines=\"none\">\r\n        <!-- 语音消息输入 -->\r\n        <ng-container *ngIf=\"chat?.isVoiceInputMode\">\r\n          <!-- 切换文本输入 -->\r\n          <ion-button class=\"btn-input-change\" color=\"primary\" (click)=\"chat.isVoiceInputMode=false\" shape=\"round\" size=\"large\">\r\n            <ion-icon name=\"chatbox-ellipses-outline\" slot=\"icon-only\"></ion-icon>\r\n          </ion-button>\r\n          \r\n          <div class=\"btn-voice-start\" (click)=\"startTalk()\" [class.disabled]=\"isSending\">\r\n            <span>\r\n              点击讲话\r\n            </span> \r\n          </div>\r\n        </ng-container>\r\n\r\n        <!-- 文本消息输入 -->\r\n        <ng-container *ngIf=\"!chat?.isVoiceInputMode\">\r\n          <!-- 切换语音输入 -->\r\n          <ion-button [style.display]=\"chat.isTexting?'none':'flex'\" class=\"btn-input-change\" color=\"primary\" *ngIf=\"chat?.role?.get('voiceConfig')\" (click)=\"chat.isVoiceInputMode=true\" shape=\"round\" size=\"large\">\r\n            <ion-icon name=\"mic-outline\" slot=\"icon-only\"></ion-icon>\r\n          </ion-button>\r\n\r\n          <!-- 文本输入区域 -->\r\n          <ion-textarea\r\n          #userInput\r\n          *ngIf=\"chat\" (keydown)=\"onKeyDown($event)\"\r\n          [errorText]=\"errorText\"\r\n          [(ngModel)]=\"chat.userInput\"\r\n          (ionFocus)=\"onInputFocus()\"\r\n          (ionBlur)=\"chat.isTexting=false\"\r\n          [autoGrow]=\"true\" shape=\"round\" fill=\"outline\"\r\n          label=\"Ctrl + Enter 发送消息\" placeholder=\"请输入您的提示词\"\r\n          labelPlacement=\"floating\"></ion-textarea>\r\n          \r\n          <!-- 文本发送按钮 -->\r\n          <ion-button [disabled]=\"isSending\"\r\n          color=\"primary\" shape=\"round\" size=\"large\" (click)=\"sendMessage()\">\r\n            <ion-icon name=\"paper-plane-outline\" slot=\"icon-only\"></ion-icon>\r\n          </ion-button>\r\n        </ng-container>\r\n    </ion-item>\r\n</ion-toolbar>\r\n\r\n\r\n<!-- 语音消息输入：弹出区域 -->\r\n<!-- <ion-modal #audioModal [isOpen]=\"isAudioModal\" (willDismiss)=\"closeAudio()\" [initialBreakpoint]=\"audioModalHeightPoint\" [breakpoints]=\"[0, audioModalHeightPoint]\">\r\n  <ng-template>\r\n    <fm-modal-audio-message #audioComp *ngIf=\"isAudioModal\" [chat]=\"chat\" [modal]=\"audioModal\"></fm-modal-audio-message>\r\n  </ng-template>\r\n</ion-modal> -->"]}
|