pcm-agents 0.6.97 → 0.6.98

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.
Files changed (145) hide show
  1. package/dist/cjs/components-position-selector.cjs.entry.js +1 -1
  2. package/dist/cjs/{conversation-utils-DEyWLr1X.js → conversation-utils-C2L56p0X.js} +4 -4
  3. package/dist/cjs/{conversation-utils-DEyWLr1X.js.map → conversation-utils-C2L56p0X.js.map} +1 -1
  4. package/dist/cjs/{index-C_wovzZM.js → index-6ScSQMo5.js} +3 -3
  5. package/dist/cjs/{index-C_wovzZM.js.map → index-6ScSQMo5.js.map} +1 -1
  6. package/dist/cjs/index.cjs.js +1 -1
  7. package/dist/cjs/oem-htws-modal.cjs.entry.js +1 -1
  8. package/dist/cjs/oem-jd-modal.cjs.entry.js +2 -2
  9. package/dist/cjs/oem-jlpp-modal.cjs.entry.js +1 -1
  10. package/dist/cjs/oem-jlsx-modal.cjs.entry.js +2 -2
  11. package/dist/cjs/oem-jlzz-modal.cjs.entry.js +1 -1
  12. package/dist/cjs/oem-mnct-modal.cjs.entry.js +1 -1
  13. package/dist/cjs/oem-mnms-modal.cjs.entry.js +2 -2
  14. package/dist/cjs/oem-mnms-modal.cjs.entry.js.map +1 -1
  15. package/dist/cjs/oem-mnms-modal.entry.cjs.js.map +1 -1
  16. package/dist/cjs/oem-qgqjl-modal.cjs.entry.js +3 -3
  17. package/dist/cjs/oem-zygh-modal_31.cjs.entry.js +5 -5
  18. package/dist/cjs/oem-zygh-modal_31.cjs.entry.js.map +1 -1
  19. package/dist/cjs/pcm-hr-chat-modal.cjs.entry.js +2 -2
  20. package/dist/cjs/{sentry-reporter-D-zjvMQZ.js → sentry-reporter-Wf6gLq1s.js} +3 -3
  21. package/dist/cjs/{sentry-reporter-D-zjvMQZ.js.map → sentry-reporter-Wf6gLq1s.js.map} +1 -1
  22. package/dist/collection/components/oem-mnms-modal/oem-mnms-modal.js +1 -1
  23. package/dist/collection/components/oem-mnms-modal/oem-mnms-modal.js.map +1 -1
  24. package/dist/collection/components/pcm-mnms-modal/pcm-mnms-modal.js +1 -1
  25. package/dist/collection/components/pcm-mnms-modal/pcm-mnms-modal.js.map +1 -1
  26. package/dist/collection/components/pcm-mnms-zp-modal/pcm-mnms-zp-modal.js +1 -1
  27. package/dist/collection/components/pcm-mnms-zp-modal/pcm-mnms-zp-modal.js.map +1 -1
  28. package/dist/components/components-position-selector.js +1 -1
  29. package/dist/components/index.js +1 -1
  30. package/dist/components/oem-htws-modal.js +6 -6
  31. package/dist/components/oem-jd-modal.js +4 -4
  32. package/dist/components/oem-jlpp-modal.js +8 -8
  33. package/dist/components/oem-jlsx-modal.js +6 -6
  34. package/dist/components/oem-jlzz-modal.js +6 -6
  35. package/dist/components/oem-mnct-modal.js +8 -8
  36. package/dist/components/oem-mnms-modal.js +10 -10
  37. package/dist/components/oem-mnms-modal.js.map +1 -1
  38. package/dist/components/oem-qgqjl-modal.js +10 -10
  39. package/dist/components/oem-zygh-modal.js +6 -6
  40. package/dist/components/{p-D139Ae-m.js → p--b8iXa1k.js} +3 -3
  41. package/dist/components/{p-D139Ae-m.js.map → p--b8iXa1k.js.map} +1 -1
  42. package/dist/components/{p-6bkwAKoN.js → p-72SR4lcU.js} +3 -3
  43. package/dist/components/{p-6bkwAKoN.js.map → p-72SR4lcU.js.map} +1 -1
  44. package/dist/components/{p-CF5t0Na2.js → p-BHgYS8ia.js} +3 -3
  45. package/dist/components/{p-CF5t0Na2.js.map → p-BHgYS8ia.js.map} +1 -1
  46. package/dist/components/{p-BuuSFiV6.js → p-BWgGPwna.js} +3 -3
  47. package/dist/components/{p-BuuSFiV6.js.map → p-BWgGPwna.js.map} +1 -1
  48. package/dist/components/{p-C_6dc91o.js → p-CNgI7G9g.js} +3 -3
  49. package/dist/components/{p-C_6dc91o.js.map → p-CNgI7G9g.js.map} +1 -1
  50. package/dist/components/{p-rOAW6Z1w.js → p-CsCmaYhd.js} +5 -5
  51. package/dist/components/{p-rOAW6Z1w.js.map → p-CsCmaYhd.js.map} +1 -1
  52. package/dist/components/{p-ny7Ww0_g.js → p-D1xultOn.js} +3 -3
  53. package/dist/components/{p-ny7Ww0_g.js.map → p-D1xultOn.js.map} +1 -1
  54. package/dist/components/{p-CgSFF1H-.js → p-DM8HJ3S-.js} +3 -3
  55. package/dist/components/{p-CgSFF1H-.js.map → p-DM8HJ3S-.js.map} +1 -1
  56. package/dist/components/{p-CPVPHk5Q.js → p-EbgYrut6.js} +3 -3
  57. package/dist/components/{p-CPVPHk5Q.js.map → p-EbgYrut6.js.map} +1 -1
  58. package/dist/components/{p-DB1Xb_SN.js → p-qDtv62HM.js} +3 -3
  59. package/dist/components/{p-DB1Xb_SN.js.map → p-qDtv62HM.js.map} +1 -1
  60. package/dist/components/{p-Bqe8RnBv.js → p-rE1cm-2A.js} +3 -3
  61. package/dist/components/{p-Bqe8RnBv.js.map → p-rE1cm-2A.js.map} +1 -1
  62. package/dist/components/{p-CeCXTxye.js → p-xec1MDLu.js} +4 -4
  63. package/dist/components/{p-CeCXTxye.js.map → p-xec1MDLu.js.map} +1 -1
  64. package/dist/components/pcm-1zhanshi-mnms-modal.js +5 -5
  65. package/dist/components/pcm-app-chat-modal.js +1 -1
  66. package/dist/components/pcm-ats-mnms-modal.js +5 -5
  67. package/dist/components/pcm-card.js +1 -1
  68. package/dist/components/pcm-chat-message.js +1 -1
  69. package/dist/components/pcm-digital-human.js +1 -1
  70. package/dist/components/pcm-export-records-modal.js +1 -1
  71. package/dist/components/pcm-hr-chat-modal.js +2 -2
  72. package/dist/components/pcm-htws-modal.js +8 -8
  73. package/dist/components/pcm-hyzj-modal.js +5 -5
  74. package/dist/components/pcm-jd-modal.js +5 -5
  75. package/dist/components/pcm-jlpp-modal.js +8 -8
  76. package/dist/components/pcm-jlsx-modal.js +5 -5
  77. package/dist/components/pcm-jlzz-modal.js +6 -6
  78. package/dist/components/pcm-mnct-modal.js +7 -7
  79. package/dist/components/pcm-mnms-modal.js +9 -9
  80. package/dist/components/pcm-mnms-modal.js.map +1 -1
  81. package/dist/components/pcm-mnms-zp-modal.js +9 -9
  82. package/dist/components/pcm-mnms-zp-modal.js.map +1 -1
  83. package/dist/components/pcm-mobile-input-btn.js +1 -1
  84. package/dist/components/pcm-mobile-upload-btn.js +1 -1
  85. package/dist/components/pcm-msbg-modal.js +5 -5
  86. package/dist/components/pcm-qgqjl-modal.js +9 -9
  87. package/dist/components/pcm-thousand-resume-wrapper.js +1 -1
  88. package/dist/components/pcm-upload.js +1 -1
  89. package/dist/components/pcm-virtual-chat-modal.js +1 -1
  90. package/dist/components/pcm-zsk-chat-modal.js +2 -2
  91. package/dist/components/pcm-zygh-modal.js +7 -7
  92. package/dist/components/pcm-zygh-v2-modal.js +7 -7
  93. package/dist/esm/components-position-selector.entry.js +1 -1
  94. package/dist/esm/{conversation-utils-D45d-WT8.js → conversation-utils-BO3752fo.js} +4 -4
  95. package/dist/esm/{conversation-utils-D45d-WT8.js.map → conversation-utils-BO3752fo.js.map} +1 -1
  96. package/dist/esm/{index-B2sU0ath.js → index-ChiwOP63.js} +3 -3
  97. package/dist/esm/{index-B2sU0ath.js.map → index-ChiwOP63.js.map} +1 -1
  98. package/dist/esm/index.js +1 -1
  99. package/dist/esm/oem-htws-modal.entry.js +1 -1
  100. package/dist/esm/oem-jd-modal.entry.js +2 -2
  101. package/dist/esm/oem-jlpp-modal.entry.js +1 -1
  102. package/dist/esm/oem-jlsx-modal.entry.js +2 -2
  103. package/dist/esm/oem-jlzz-modal.entry.js +1 -1
  104. package/dist/esm/oem-mnct-modal.entry.js +1 -1
  105. package/dist/esm/oem-mnms-modal.entry.js +2 -2
  106. package/dist/esm/oem-mnms-modal.entry.js.map +1 -1
  107. package/dist/esm/oem-qgqjl-modal.entry.js +3 -3
  108. package/dist/esm/oem-zygh-modal_31.entry.js +5 -5
  109. package/dist/esm/oem-zygh-modal_31.entry.js.map +1 -1
  110. package/dist/esm/pcm-hr-chat-modal.entry.js +2 -2
  111. package/dist/esm/{sentry-reporter-BWPbyZsF.js → sentry-reporter-CbxGWjKB.js} +3 -3
  112. package/dist/esm/{sentry-reporter-BWPbyZsF.js.map → sentry-reporter-CbxGWjKB.js.map} +1 -1
  113. package/dist/pcm-agents/index.esm.js +1 -1
  114. package/dist/pcm-agents/oem-mnms-modal.entry.esm.js.map +1 -1
  115. package/dist/pcm-agents/{p-3069b708.entry.js → p-083ed4c8.entry.js} +2 -2
  116. package/dist/pcm-agents/{p-efe2d769.entry.js → p-0c24aaa6.entry.js} +3 -3
  117. package/dist/pcm-agents/{p-efe2d769.entry.js.map → p-0c24aaa6.entry.js.map} +1 -1
  118. package/dist/pcm-agents/{p-c194b3b5.entry.js → p-0f7ecd6e.entry.js} +2 -2
  119. package/dist/pcm-agents/{p-b7a2f1fb.entry.js → p-510a4e7d.entry.js} +2 -2
  120. package/dist/pcm-agents/{p-5af061ed.entry.js → p-83eb56c2.entry.js} +2 -2
  121. package/dist/pcm-agents/{p-32b990fa.entry.js → p-8f048708.entry.js} +2 -2
  122. package/dist/pcm-agents/{p-5def8872.entry.js → p-9192a541.entry.js} +2 -2
  123. package/dist/pcm-agents/{p-KC5DRaFJ.js → p-BcP-XCgZ.js} +2 -2
  124. package/dist/pcm-agents/{p-KC5DRaFJ.js.map → p-BcP-XCgZ.js.map} +1 -1
  125. package/dist/pcm-agents/p-ZaVCcNkc.js +2 -0
  126. package/dist/pcm-agents/{p-Duxw4aNn.js.map → p-ZaVCcNkc.js.map} +1 -1
  127. package/dist/pcm-agents/{p-50e9b7c8.entry.js → p-cd4b6377.entry.js} +2 -2
  128. package/dist/pcm-agents/{p-f511c080.entry.js → p-d0a26bca.entry.js} +2 -2
  129. package/dist/pcm-agents/{p-f511c080.entry.js.map → p-d0a26bca.entry.js.map} +1 -1
  130. package/dist/pcm-agents/{p-245d3dba.entry.js → p-ec87f4ff.entry.js} +2 -2
  131. package/dist/pcm-agents/{p-9e352fde.entry.js → p-fbeb3b8d.entry.js} +2 -2
  132. package/dist/pcm-agents/{p-BOH1gYZj.js → p-pt92A5Nu.js} +2 -2
  133. package/dist/pcm-agents/{p-BOH1gYZj.js.map → p-pt92A5Nu.js.map} +1 -1
  134. package/dist/pcm-agents/pcm-agents.esm.js +1 -1
  135. package/package.json +1 -1
  136. package/dist/pcm-agents/p-Duxw4aNn.js +0 -2
  137. /package/dist/pcm-agents/{p-3069b708.entry.js.map → p-083ed4c8.entry.js.map} +0 -0
  138. /package/dist/pcm-agents/{p-c194b3b5.entry.js.map → p-0f7ecd6e.entry.js.map} +0 -0
  139. /package/dist/pcm-agents/{p-b7a2f1fb.entry.js.map → p-510a4e7d.entry.js.map} +0 -0
  140. /package/dist/pcm-agents/{p-5af061ed.entry.js.map → p-83eb56c2.entry.js.map} +0 -0
  141. /package/dist/pcm-agents/{p-32b990fa.entry.js.map → p-8f048708.entry.js.map} +0 -0
  142. /package/dist/pcm-agents/{p-5def8872.entry.js.map → p-9192a541.entry.js.map} +0 -0
  143. /package/dist/pcm-agents/{p-50e9b7c8.entry.js.map → p-cd4b6377.entry.js.map} +0 -0
  144. /package/dist/pcm-agents/{p-245d3dba.entry.js.map → p-ec87f4ff.entry.js.map} +0 -0
  145. /package/dist/pcm-agents/{p-9e352fde.entry.js.map → p-fbeb3b8d.entry.js.map} +0 -0
@@ -1 +1 @@
1
- {"version":3,"names":["oemMnmsModalCss","globalCss","hostCss","OemMnmsModal","modalTitle","token","isOpen","modalClosed","icon","zIndex","isShowHeader","isNeedClose","conversationId","defaultQuery","maxRecordingTime","waitingTime","fullscreen","customInputs","showWorkspaceHistory","oemId","digitalId","openingIndex","enableVirtualHuman","showEndInterviewButton","playbackRate","uploadSuccess","streamComplete","conversationStart","interviewComplete","tokenInvalid","someErrorEvent","sendSmsCodeRequest","interviewMode","recordingError","showCopyButton","showFeedbackButtons","selectedFile","isUploading","uploadedFileInfo","showChatModal","phoneNumber","verificationCode","isSendingCode","countdown","isVerifyingCode","showPositionSelector","jobDescription","isSubmitting","tokenInvalidListener","removeErrorListener","virtualChatModalRef","handleTokenChange","newToken","authStore","getToken","setToken","handleIsOpenChange","newValue","this","setComponentActive","resetState","verifyApiKey","componentWillLoad","configStore","setItem","emit","ErrorEventBus","addErrorListener","errorDetail","document","addEventListener","disconnectedCallback","removeEventListener","countdownTimer","clearInterval","handleClose","handleJobDescriptionChange","event","textarea","target","value","handlePhoneNumberChange","input","handleVerificationCodeChange","startCountdown","setInterval","handleSendCode","trim","Message","error","phoneRegex","test","mobile","codeType","handleOpenPositionSelector","info","handleClosePositionSelector","handlePositionSelected","position","detail","formatted_text","success","handleStartInterview","async","job_info","result","verifySmsCode","pcmUploadRef","getIsUploading","render","modalStyle","String","containerClass","overlayClass","isLoading","disableJdInput","Boolean","disableResumeUpload","file_url","resume_content","h","class","style","src","alt","onClick","htmlFor","type","name","onOk","e","id","placeholder","rows","onInput","disabled","ref","el","maxFileSize","multiple","mobileUploadAble","acceptFileSuffixList","uploadParams","tags","defaultFileList","cos_key","file_name","file_size","ext","undefined","onUploadChange","maxLength","length","href","rel","botId","question_number","ai_interview_type","phone","onPositionSelected"],"sources":["src/components/oem-mnms-modal/oem-mnms-modal.css?tag=oem-mnms-modal&encapsulation=shadow","src/global/global.css?tag=oem-mnms-modal&encapsulation=shadow","src/global/host.css?tag=oem-mnms-modal&encapsulation=shadow","src/components/oem-mnms-modal/oem-mnms-modal.tsx"],"sourcesContent":["/* 标签和按钮容器 */\r\n.label-with-button {\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n margin-bottom: 8px;\r\n}\r\n\r\n.button-group {\r\n display: flex;\r\n gap: 8px;\r\n align-items: center;\r\n}\r\n\r\n.position-selector-button {\r\n padding: 4px 12px;\r\n background: #1890ff;\r\n color: white;\r\n border: none;\r\n border-radius: 4px;\r\n font-size: 13px;\r\n cursor: pointer;\r\n transition: all 0.3s ease;\r\n white-space: nowrap;\r\n height: 28px;\r\n line-height: 20px;\r\n}\r\n\r\n.position-selector-button:hover {\r\n background: #40a9ff;\r\n}\r\n\r\n.position-selector-button:active {\r\n background: #096dd9;\r\n}\r\n\r\n/* 手机号输入框样式 */\r\n.phone-input {\r\n width: calc(100% - 16px);\r\n padding: 8px;\r\n border: 1px solid #ddd;\r\n border-radius: 4px;\r\n font-family: inherit;\r\n font-size: 14px;\r\n line-height: 1.5;\r\n transition: border-color 0.3s;\r\n}\r\n\r\n.phone-input:focus {\r\n outline: none;\r\n border-color: #1890ff;\r\n box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);\r\n}\r\n\r\n/* 验证码容器样式 */\r\n.verification-code-container {\r\n display: flex;\r\n gap: 8px;\r\n align-items: center;\r\n}\r\n\r\n.verification-code-input {\r\n flex: 1;\r\n padding: 8px;\r\n border: 1px solid #ddd;\r\n border-radius: 4px;\r\n font-family: inherit;\r\n font-size: 14px;\r\n line-height: 1.5;\r\n transition: border-color 0.3s;\r\n}\r\n\r\n.verification-code-input:focus {\r\n outline: none;\r\n border-color: #1890ff;\r\n box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);\r\n}\r\n\r\n/* 发送验证码按钮样式 */\r\n.send-code-button {\r\n padding: 8px 16px;\r\n background: #0D75FB;\r\n color: white;\r\n border: none;\r\n border-radius: 4px;\r\n font-size: 14px;\r\n cursor: pointer;\r\n transition: all 0.3s ease;\r\n white-space: nowrap;\r\n min-width: 100px;\r\n}\r\n\r\n.send-code-button:hover:not(:disabled) {\r\n background-color: #40a9ff;\r\n}\r\n\r\n.send-code-button:disabled {\r\n background-color: rgba(0, 0, 0, 0.04);\r\n color: rgba(0, 0, 0, 0.25);\r\n cursor: not-allowed;\r\n}\r\n\r\n/* 职位选择器弹窗样式 */\r\n.position-selector-modal-overlay {\r\n position: fixed;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n background-color: rgba(0, 0, 0, 0.5);\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n z-index: 3000;\r\n padding: 20px;\r\n}\r\n\r\n.position-selector-modal {\r\n background: white;\r\n border-radius: 8px;\r\n width: 90%;\r\n max-width: 900px;\r\n height: 80vh;\r\n max-height: 600px;\r\n display: flex;\r\n flex-direction: column;\r\n box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\r\n animation: modalFadeIn 0.3s ease;\r\n}\r\n\r\n@keyframes modalFadeIn {\r\n from {\r\n opacity: 0;\r\n transform: scale(0.95) translateY(-20px);\r\n }\r\n to {\r\n opacity: 1;\r\n transform: scale(1) translateY(0);\r\n }\r\n}\r\n\r\n.position-selector-modal-header {\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n padding: 16px 24px;\r\n border-bottom: 1px solid #f0f0f0;\r\n flex-shrink: 0;\r\n}\r\n\r\n.position-selector-modal-header h3 {\r\n margin: 0;\r\n font-size: 16px;\r\n font-weight: 500;\r\n color: rgba(0, 0, 0, 0.85);\r\n}\r\n\r\n.position-selector-close-btn {\r\n background: none;\r\n border: none;\r\n font-size: 24px;\r\n line-height: 1;\r\n cursor: pointer;\r\n color: rgba(0, 0, 0, 0.45);\r\n padding: 0;\r\n width: 32px;\r\n height: 32px;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n border-radius: 4px;\r\n transition: all 0.3s;\r\n}\r\n\r\n.position-selector-close-btn:hover {\r\n background-color: rgba(0, 0, 0, 0.06);\r\n color: rgba(0, 0, 0, 0.85);\r\n}\r\n\r\n.position-selector-modal-body {\r\n flex: 1;\r\n overflow: hidden;\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n\r\n/* 移动端适配 */\r\n@media screen and (max-width: 768px) {\r\n .position-selector-modal {\r\n width: 100%;\r\n height: 90vh;\r\n max-height: none;\r\n border-radius: 8px 8px 0 0;\r\n margin-top: auto;\r\n }\r\n\r\n .position-selector-modal-overlay {\r\n padding: 0;\r\n align-items: flex-end;\r\n }\r\n}\r\n","/* 模态框基础样式 */\r\n.modal-overlay {\r\n position: fixed;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n background-color: rgba(0, 0, 0, 0.5);\r\n display: flex;\r\n z-index: 1000;\r\n overflow-y: auto;\r\n}\r\n\r\n/* 全屏模式下的overlay样式 - 改为基于父组件 */\r\n.fullscreen-overlay {\r\n padding: 0;\r\n background-color: rgba(0, 0, 0, 0.7);\r\n /* 改为基于父组件的全屏 */\r\n position: absolute;\r\n width: 100%;\r\n height: auto;\r\n}\r\n\r\n.modal-container {\r\n background-color: #fff;\r\n border-radius: 8px;\r\n width: 100%;\r\n display: flex;\r\n flex-direction: column;\r\n position: relative;\r\n margin: auto;\r\n transition: all 0.3s ease-out;\r\n overflow: hidden;\r\n}\r\n\r\n/* 全屏模式样式 - 改为基于父组件 */\r\n.modal-container.fullscreen {\r\n width: 100%;\r\n max-width: none;\r\n height: 100%;\r\n border-radius: 0;\r\n margin: 0;\r\n display: flex;\r\n flex-direction: column;\r\n max-height: 100%;\r\n}\r\n\r\n/* PC端布局 */\r\n.pc-layout {\r\n width: 80%;\r\n max-width: 600px;\r\n min-width: 320px;\r\n}\r\n\r\n/* 响应式布局 */\r\n@media screen and (max-width: 768px) {\r\n .pc-layout {\r\n width: 95%;\r\n }\r\n\r\n .modal-overlay {\r\n padding: 10px 0px 0px 0px;\r\n }\r\n\r\n .modal-container.fullscreen {\r\n /* 移动端也基于父组件尺寸 */\r\n width: 100%;\r\n height: 100%;\r\n max-height: 100%;\r\n border-radius: 16px 16px 0 0;\r\n /* 保留安全区域支持 */\r\n padding: env(safe-area-inset-top) 0 env(safe-area-inset-bottom);\r\n }\r\n}\r\n\r\n/* 模态框头部样式 */\r\n.modal-header {\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n padding: 4px 16px;\r\n height: 50px;\r\n border-bottom: 1px solid #e8e8e8;\r\n flex-shrink: 0;\r\n}\r\n\r\n.header-left {\r\n display: flex;\r\n align-items: center;\r\n gap: 8px;\r\n font-size: 16px;\r\n font-weight: 600;\r\n color: #333;\r\n}\r\n\r\n.header-icon {\r\n width: 24px;\r\n height: 24px;\r\n}\r\n\r\n.close-button {\r\n background: transparent;\r\n border: none;\r\n cursor: pointer;\r\n padding: 8px;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 32px;\r\n height: 32px;\r\n border-radius: 4px;\r\n}\r\n\r\n.close-button:hover {\r\n background-color: rgba(0, 0, 0, 0.04);\r\n}\r\n\r\n.close-button span {\r\n font-size: 24px;\r\n line-height: 1;\r\n color: #999;\r\n}\r\n\r\n.close-button:hover span {\r\n color: #666;\r\n}\r\n\r\n\r\n/* 文件上传区域通用样式 */\r\n.upload-area {\r\n cursor: pointer;\r\n width: 100%;\r\n}\r\n\r\n\r\n.upload-placeholder {\r\n transition: all 0.3s ease;\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n background: rgba(0, 0, 0, 0.02);\r\n border: 1px dashed #d9d9d9;\r\n border-radius: 8px;\r\n cursor: pointer;\r\n}\r\n\r\n.upload-placeholder:hover {\r\n border: 1px dashed #1890ff;\r\n}\r\n\r\n.upload-placeholder img {\r\n margin-top: 8px;\r\n width: 50px;\r\n height: 50px;\r\n}\r\n\r\n.upload-placeholder .upload-text {\r\n margin: 4px 0;\r\n color: #332F39;\r\n font-size: 14px;\r\n}\r\n\r\n.upload-placeholder .upload-hint {\r\n font-size: 14px;\r\n color: #949AA5;\r\n margin-top: 8px;\r\n padding: 0px 10px;\r\n text-align: center;\r\n white-space: normal;\r\n word-wrap: break-word;\r\n word-break: break-all;\r\n line-height: 1.4;\r\n}\r\n\r\n\r\n/* 文件项样式 */\r\n.file-item {\r\n position: relative;\r\n padding: 16px;\r\n border: 1px solid #e2e8f0;\r\n border-radius: 8px;\r\n transition: border-color 0.3s;\r\n cursor: pointer;\r\n margin-bottom: 16px;\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n}\r\n\r\n.file-item:hover {\r\n border-color: #0D75FB;\r\n}\r\n\r\n.file-item-content {\r\n display: flex;\r\n align-items: center;\r\n gap: 8px;\r\n flex: 1;\r\n min-width: 0;\r\n overflow: hidden;\r\n}\r\n\r\n.file-icon {\r\n color: #0D75FB;\r\n flex-shrink: 0;\r\n}\r\n\r\n.file-name {\r\n font-weight: 500;\r\n white-space: nowrap;\r\n overflow: hidden;\r\n text-overflow: ellipsis;\r\n max-width: calc(100% - 50px);\r\n}\r\n\r\n.remove-file {\r\n background: transparent;\r\n border: none;\r\n color: #94a3b8;\r\n cursor: pointer;\r\n font-size: 18px;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n padding: 4px;\r\n margin-left: 8px;\r\n border-radius: 4px;\r\n transition: all 0.2s;\r\n min-width: 30px;\r\n min-height: 30px;\r\n}\r\n\r\n.remove-file:hover {\r\n background-color: #f1f5f9;\r\n color: #475569;\r\n}\r\n\r\n.file-input {\r\n display: none;\r\n}\r\n\r\n\r\n\r\n/* 输入容器样式 */\r\n.input-container {\r\n padding: 20px;\r\n display: flex;\r\n flex-direction: column;\r\n height: calc(100% - 50px);\r\n background: linear-gradient(150deg, #2a6ee933, #0000 50%) 0 0 / 400px 200px no-repeat, #fff;\r\n /* 减去header高度 */\r\n overflow-y: auto;\r\n}\r\n\r\n.input-container h3 {\r\n margin-top: 0;\r\n margin-bottom: 20px;\r\n font-size: 18px;\r\n color: #333;\r\n text-align: center;\r\n}\r\n\r\n/* JD输入区域样式 */\r\n.jd-input-section {\r\n margin-bottom: 20px;\r\n}\r\n\r\n.jd-input-section label {\r\n display: block;\r\n margin-bottom: 8px;\r\n font-weight: 500;\r\n color: #333;\r\n}\r\n\r\n.job-description-textarea {\r\n width: calc(100% - 16px);\r\n border: 1px solid #ddd;\r\n border-radius: 4px;\r\n resize: vertical;\r\n font-family: inherit;\r\n font-size: 14px;\r\n line-height: 1.5;\r\n transition: border-color 0.3s;\r\n padding: 8px;\r\n}\r\n\r\n.job-description-textarea:focus {\r\n outline: none;\r\n border-color: #1890ff;\r\n box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);\r\n}\r\n\r\n/* 简历上传区域样式 */\r\n.resume-upload-section {\r\n margin-bottom: 20px;\r\n width: 100%;\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n\r\n.resume-upload-section label {\r\n display: block;\r\n margin-bottom: 8px;\r\n font-weight: 500;\r\n color: #333;\r\n}\r\n\r\n/* 面试设置区域 */\r\n.interview-settings-section {\r\n margin-bottom: 24px;\r\n padding: 20px;\r\n background-color: #f8f9fa;\r\n border-radius: 8px;\r\n border: 1px solid #e9ecef;\r\n}\r\n\r\n.settings-row {\r\n margin-bottom: 20px;\r\n}\r\n\r\n.settings-row:last-child {\r\n margin-bottom: 0;\r\n}\r\n\r\n.settings-label {\r\n display: block;\r\n font-weight: 500;\r\n color: #333;\r\n margin-bottom: 12px;\r\n font-size: 14px;\r\n}\r\n\r\n/* 题目数量滑块样式 */\r\n.question-number-section {\r\n width: 100%;\r\n}\r\n\r\n.slider-container {\r\n display: flex;\r\n align-items: center;\r\n gap: 12px;\r\n}\r\n\r\n.question-slider {\r\n flex: 1;\r\n height: 6px;\r\n background: #ddd;\r\n outline: none;\r\n border-radius: 3px;\r\n -webkit-appearance: none;\r\n appearance: none;\r\n}\r\n\r\n.question-slider::-webkit-slider-thumb {\r\n -webkit-appearance: none;\r\n appearance: none;\r\n width: 20px;\r\n height: 20px;\r\n background: #0D75FB;\r\n cursor: pointer;\r\n border-radius: 50%;\r\n box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);\r\n}\r\n\r\n.question-slider::-moz-range-thumb {\r\n width: 20px;\r\n height: 20px;\r\n background: #0D75FB;\r\n cursor: pointer;\r\n border-radius: 50%;\r\n border: none;\r\n box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);\r\n}\r\n\r\n.slider-value {\r\n font-weight: 600;\r\n color: #0D75FB;\r\n font-size: 14px;\r\n min-width: 30px;\r\n text-align: center;\r\n}\r\n\r\n/* 开关样式 */\r\n.analysis-toggle-section {\r\n width: 100%;\r\n}\r\n\r\n.toggle-container {\r\n display: flex;\r\n align-items: center;\r\n gap: 12px;\r\n}\r\n\r\n.toggle-switch {\r\n position: relative;\r\n display: inline-block;\r\n width: 44px;\r\n height: 24px;\r\n}\r\n\r\n.toggle-switch input {\r\n opacity: 0;\r\n width: 0;\r\n height: 0;\r\n}\r\n\r\n.toggle-slider {\r\n position: absolute;\r\n cursor: pointer;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n background-color: #ccc;\r\n transition: 0.3s;\r\n border-radius: 24px;\r\n}\r\n\r\n.toggle-slider:before {\r\n position: absolute;\r\n content: \"\";\r\n height: 18px;\r\n width: 18px;\r\n left: 3px;\r\n bottom: 3px;\r\n background-color: white;\r\n transition: 0.3s;\r\n border-radius: 50%;\r\n box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);\r\n}\r\n\r\ninput:checked + .toggle-slider {\r\n background-color: #0D75FB;\r\n}\r\n\r\ninput:checked + .toggle-slider:before {\r\n transform: translateX(20px);\r\n}\r\n\r\n.toggle-text {\r\n font-size: 14px;\r\n color: #666;\r\n font-weight: 500;\r\n}\r\n\r\n/* 提交按钮通用样式 */\r\n.submit-button {\r\n margin-top: 10px;\r\n padding: 10px 30px;\r\n background: #0D75FB;\r\n color: white;\r\n border: none;\r\n border-radius: 4px;\r\n font-size: 16px;\r\n cursor: pointer;\r\n transition: all 0.3s ease;\r\n width: 100%;\r\n max-width: 400px;\r\n align-self: center;\r\n}\r\n\r\n.submit-button:hover {\r\n background-color: #40a9ff;\r\n}\r\n\r\n.submit-button:disabled {\r\n background-color: rgba(0,0,0,0.04);\r\n color: rgba(0,0,0,0.25);\r\n cursor: not-allowed;\r\n}\r\n\r\n\r\n\r\n/* AI免责声明和备案信息样式 */\r\n.ai-disclaimer {\r\n margin-top: 16px;\r\n text-align: center;\r\n font-size: 12px;\r\n color: #999;\r\n line-height: 1.5;\r\n}\r\n\r\n.ai-disclaimer p {\r\n margin: 4px 0;\r\n}\r\n\r\n.beian-info {\r\n display: flex;\r\n justify-content: center;\r\n flex-wrap: wrap;\r\n gap: 4px;\r\n}\r\n\r\n.ai-disclaimer a {\r\n color: #666;\r\n text-decoration: none;\r\n transition: color 0.2s ease;\r\n}\r\n\r\n.ai-disclaimer a:hover {\r\n color: #1890ff;\r\n text-decoration: underline;\r\n}\r\n\r\n/* 添加加载状态的样式 */\r\n.loading-container {\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n justify-content: center;\r\n height: 100%;\r\n padding: 24px;\r\n }\r\n \r\n .loading-spinner {\r\n width: 40px;\r\n height: 40px;\r\n border: 4px solid rgba(0, 0, 0, 0.1);\r\n border-radius: 50%;\r\n border-top-color: var(--pcm-primary-color, #1890ff);\r\n animation: spin 1s linear infinite;\r\n margin-bottom: 16px;\r\n }\r\n \r\n .loading-text {\r\n font-size: 16px;\r\n color: var(--pcm-text-color, #333);\r\n }\r\n \r\n @keyframes spin {\r\n 0% { transform: rotate(0deg); }\r\n 100% { transform: rotate(360deg); }\r\n }\r\n\r\n/* 历史会话按钮 */\r\n.header-history-button {\r\n display: inline-flex;\r\n align-items: center;\r\n justify-content: center;\r\n gap: 4px;\r\n padding: 4px 8px;\r\n border: none;\r\n border-radius: 4px;\r\n background: transparent;\r\n color: #bbb;\r\n cursor: pointer;\r\n font-size: 13px;\r\n font-family: inherit;\r\n line-height: 1;\r\n transition: all 0.2s ease;\r\n flex-shrink: 0;\r\n white-space: nowrap;\r\n}\r\n\r\n.header-history-button svg {\r\n width: 16px;\r\n height: 16px;\r\n display: block;\r\n flex-shrink: 0;\r\n}\r\n\r\n.header-history-button:hover {\r\n background-color: rgba(0, 0, 0, 0.04);\r\n color: #666;\r\n}\r\n\r\n.header-history-button:focus-visible {\r\n outline: 2px solid rgba(13, 117, 251, 0.35);\r\n outline-offset: 2px;\r\n}\r\n\r\n/* 历史会话抽屉内容 */\r\n.history-drawer-content {\r\n height: 100%;\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n\r\n.conversation-list {\r\n flex: 1;\r\n overflow-y: auto;\r\n padding: 8px 0;\r\n}\r\n\r\n.loading-conversations,\r\n.empty-conversations {\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n justify-content: center;\r\n padding: 40px 20px;\r\n color: #999;\r\n}\r\n\r\n.loading-conversations .loading-spinner-small {\r\n margin-bottom: 12px;\r\n}\r\n\r\n.conversation-item {\r\n display: flex;\r\n align-items: center;\r\n justify-content: space-between;\r\n padding: 12px 16px;\r\n margin: 0 8px 4px 8px;\r\n border-radius: 8px;\r\n cursor: pointer;\r\n transition: all 0.2s ease;\r\n border: 1px solid transparent;\r\n}\r\n\r\n.conversation-item:hover {\r\n background: #f5f5f5;\r\n}\r\n\r\n.conversation-info {\r\n flex: 1;\r\n min-width: 0;\r\n}\r\n\r\n.conversation-title {\r\n font-size: 14px;\r\n font-weight: 500;\r\n color: #262626;\r\n margin-bottom: 4px;\r\n overflow: hidden;\r\n text-overflow: ellipsis;\r\n white-space: nowrap;\r\n}\r\n\r\n.conversation-meta {\r\n display: flex;\r\n align-items: center;\r\n gap: 8px;\r\n font-size: 12px;\r\n color: #8c8c8c;\r\n}\r\n\r\n.conversation-time {\r\n flex-shrink: 0;\r\n}\r\n\r\n.message-count {\r\n flex-shrink: 0;\r\n}\r\n\r\n.conversation-status {\r\n padding: 2px 6px;\r\n border-radius: 4px;\r\n font-size: 11px;\r\n font-weight: 500;\r\n}\r\n\r\n.conversation-status.completed {\r\n background: #f6ffed;\r\n color: #52c41a;\r\n}\r\n\r\n.conversation-status.running {\r\n background: #fff7e6;\r\n color: #fa8c16;\r\n}\r\n\r\n.header-history-entry {\r\n display: inline-flex;\r\n align-items: center;\r\n justify-content: center;\r\n margin-left: 8px;\r\n padding: 4px 8px;\r\n border: none;\r\n border-radius: 4px;\r\n background: #e6f7ff;\r\n color: #1890ff;\r\n cursor: pointer;\r\n font-size: 12px;\r\n font-family: inherit;\r\n line-height: 1;\r\n white-space: nowrap;\r\n transition: all 0.2s ease;\r\n}\r\n\r\n.header-history-entry:hover {\r\n background: #bae7ff;\r\n color: #0958d9;\r\n transform: translateY(-1px);\r\n}\r\n\r\n.header-history-entry:focus-visible {\r\n outline: 2px solid rgba(13, 117, 251, 0.35);\r\n outline-offset: 2px;\r\n}\r\n\r\n.history-pagination {\r\n display: flex;\r\n align-items: center;\r\n justify-content: space-between;\r\n gap: 12px;\r\n padding: 12px 16px 16px;\r\n border-top: 1px solid #f0f0f0;\r\n}\r\n\r\n.history-pagination-button {\r\n min-width: 72px;\r\n padding: 6px 12px;\r\n border: 1px solid #d9d9d9;\r\n border-radius: 6px;\r\n background: #fff;\r\n color: #595959;\r\n cursor: pointer;\r\n font-size: 12px;\r\n transition: all 0.2s ease;\r\n}\r\n\r\n.history-pagination-button:hover:not(:disabled) {\r\n border-color: #1890ff;\r\n color: #1890ff;\r\n background: #e6f7ff;\r\n}\r\n\r\n.history-pagination-button:disabled {\r\n border-color: #f0f0f0;\r\n color: #bfbfbf;\r\n background: #fafafa;\r\n cursor: not-allowed;\r\n}\r\n\r\n.history-pagination-info {\r\n flex: 1;\r\n text-align: center;\r\n color: #8c8c8c;\r\n font-size: 12px;\r\n}\r\n \r\n",":host {\r\n /* 字体相关属性 */\r\n font-size: 16px;\r\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;\r\n font-weight: normal;\r\n font-style: normal;\r\n font-variant: normal;\r\n line-height: 1.5;\r\n letter-spacing: normal;\r\n word-spacing: normal;\r\n \r\n /* 文本相关属性 */\r\n color: #333; /* 重置文字颜色,避免继承外部的red */\r\n text-align: left;\r\n text-indent: 0;\r\n text-transform: none;\r\n white-space: normal;\r\n word-wrap: normal;\r\n word-break: normal;\r\n \r\n /* 可见性 */\r\n visibility: visible;\r\n \r\n /* 列表相关 */\r\n list-style: none;\r\n list-style-image: none;\r\n list-style-position: outside;\r\n list-style-type: disc;\r\n \r\n /* 表格相关 */\r\n border-collapse: separate;\r\n border-spacing: 0;\r\n caption-side: top;\r\n empty-cells: show;\r\n table-layout: auto;\r\n \r\n /* 引用 */\r\n quotes: auto;\r\n \r\n /* 光标 */\r\n cursor: auto;\r\n \r\n /* 文本方向 */\r\n direction: ltr;\r\n \r\n /* 打印相关 */\r\n orphans: 2;\r\n widows: 2;\r\n page-break-inside: auto;\r\n \r\n /* 其他可能被继承的属性 */\r\n text-decoration: none;\r\n text-shadow: none;\r\n \r\n /* 确保组件有独立的盒模型 */\r\n box-sizing: border-box;\r\n \r\n /* 重置一些可能影响布局的继承属性 */\r\n writing-mode: horizontal-tb;\r\n text-orientation: mixed;\r\n}","import { Component, Prop, h, State, Element, Event, EventEmitter, Watch } from '@stencil/core';\r\nimport { FileUploadResponse, verifyApiKey, verifySmsCode, Position } from '../../utils/utils';\r\nimport {\r\n StreamCompleteEventData,\r\n ConversationStartEventData,\r\n InterviewCompleteEventData,\r\n RecordingErrorEventData,\r\n} from '../../interfaces/events';\r\nimport { ErrorEventBus, ErrorEventDetail } from '../../utils/error-event';\r\nimport { authStore } from '../../../store/auth.store';\r\nimport { configStore } from '../../../store/config.store';\r\nimport { Message } from '../../services/message.service';\r\n\r\n/**\r\n * 模拟面试(求职)\r\n */\r\n\r\n@Component({\r\n tag: 'oem-mnms-modal',\r\n styleUrls: ['oem-mnms-modal.css', '../../global/global.css', '../../global/host.css'],\r\n shadow: true,\r\n})\r\nexport class OemMnmsModal {\r\n /**\r\n * 模态框标题\r\n */\r\n @Prop() modalTitle: string = '模拟面试';\r\n\r\n /**\r\n * SDK鉴权密钥\r\n */\r\n @Prop({ attribute: 'token' }) token!: string;\r\n\r\n /**\r\n * 是否显示聊天模态框\r\n */\r\n @Prop({ mutable: true }) isOpen: boolean = false;\r\n\r\n /**\r\n * 当点击模态框关闭时触发\r\n */\r\n @Event() modalClosed: EventEmitter<void>;\r\n\r\n /**\r\n * 应用图标URL\r\n */\r\n @Prop() icon?: string;\r\n\r\n /**\r\n * 聊天框的页面层级\r\n */\r\n @Prop() zIndex?: number = 1000;\r\n\r\n /**\r\n * 是否展示顶部标题栏\r\n */\r\n @Prop() isShowHeader: boolean = true;\r\n\r\n /**\r\n * 是否展示右上角的关闭按钮\r\n */\r\n @Prop() isNeedClose: boolean = true;\r\n\r\n /**\r\n * 会话ID,传入继续对话,否则创建新会话\r\n */\r\n @Prop({ mutable: true }) conversationId?: string;\r\n\r\n /**\r\n * 默认查询文本\r\n */\r\n @Prop() defaultQuery: string = '请开始模拟面试';\r\n\r\n /**\r\n * 视频录制最大时长(秒)默认120\r\n */\r\n @Prop() maxRecordingTime: number = 120;\r\n\r\n /**\r\n * 等待录制时间(秒)默认10\r\n */\r\n @Prop() waitingTime: number = 10;\r\n\r\n /**\r\n * 是否以全屏模式打开,移动端建议设置为true\r\n */\r\n @Prop() fullscreen: boolean = false;\r\n\r\n /**\r\n * 自定义输入参数,可传入以下参数:<br>\r\n * customInputs.job_info时,JD输入区域不可修改。<br>\r\n * customInputs.file_url或customInputs.resume_content时,简历上传区域不可修改。<br>\r\n * customInputs.resume_content:可传入json字符串,或纯文本字符串,字符串内容为简历内容。<br>\r\n * customInputs.url_callback:可传入url字符串,当报告生成后,会调用该url进行回调。该url请使用post请求,接收报告字段为report_content,会话id字段为conversation_id。<br>\r\n * customInputs.question_number时,会设置面试题总数量, 默认5题。<br>\r\n */\r\n @Prop() customInputs: Record<string, string> = {};\r\n\r\n /**\r\n * 是否显示工作区历史会话按钮\r\n */\r\n @Prop() showWorkspaceHistory: boolean = false;\r\n\r\n /**\r\n * OEM ID,用于获取职位列表\r\n */\r\n @Prop() oemId?: string;\r\n\r\n /**\r\n * 虚拟数字人ID,指定则开启虚拟数字人功能</br>\r\n * 智琪:79202506752 </br>\r\n * 艾綝:79402522624 </br>\r\n * 艾姮:79597885440 </br>\r\n */\r\n @Prop() digitalId?: string;\r\n\r\n /**\r\n * 数字人开场白索引,用于选择开场白和开场视频(可选:0, 1, 2)\r\n * 0、您好,我是聘才猫 AI 面试助手。很高兴为你主持这场面试!在开始前,请确保:身处安静、光线充足的环境。网络顺畅,摄像头和麦克风工作正常。现在我正在查看本次面试的相关信息,为您生成专属面试题,马上就好,请稍等片刻。</br>\r\n * 1、您好,我是您的 AI 面试助手。欢迎参加本次AI面试!为了获得最佳效果,请确认:您在安静、明亮的环境中。您的网络稳定,摄像头和麦克风已开启。我们正在后台为您准备本次专属面试内容,很快开始,请稍候。<br>\r\n * 2、您好,我是您的 AI 面试助手。面试马上开始。趁此片刻,请快速确认:周围安静吗?光线足够吗?网络没问题?摄像头和麦克风准备好了吗?我们正在为您加载个性化的面试环节,稍等就好!\r\n */\r\n @Prop() openingIndex: number = 0;\r\n\r\n /**\r\n * 是否启用全屏数字人模式,此模式下面试报告只会通过url_callback回调返回\r\n */\r\n @Prop() enableVirtualHuman: boolean = false;\r\n\r\n /**\r\n * 是否显示结束面试按钮\r\n */\r\n @Prop() showEndInterviewButton: boolean = false;\r\n\r\n /**\r\n * 数字人视频播放速率(0.5-2.0之间),默认1.0\r\n */\r\n @Prop() playbackRate: number = 1.0;\r\n\r\n /**\r\n * 上传成功事件\r\n */\r\n @Event() uploadSuccess: EventEmitter<FileUploadResponse>;\r\n\r\n /**\r\n * 流式输出完成事件\r\n */\r\n @Event() streamComplete: EventEmitter<StreamCompleteEventData>;\r\n\r\n /**\r\n * 新会话开始的回调,只会在一轮对话开始时触发一次\r\n */\r\n @Event() conversationStart: EventEmitter<ConversationStartEventData>;\r\n\r\n /**\r\n * 当聊天完成时触发\r\n */\r\n @Event() interviewComplete: EventEmitter<InterviewCompleteEventData>;\r\n\r\n /**\r\n * SDK密钥验证失败事件\r\n */\r\n @Event() tokenInvalid: EventEmitter<void>;\r\n\r\n /**\r\n * 错误事件\r\n */\r\n @Event() someErrorEvent: EventEmitter<ErrorEventDetail>;\r\n\r\n /**\r\n * 发送短信验证码请求事件\r\n * 接入方需要监听此事件并调用自己的短信接口发送验证码\r\n * 事件参数: { mobile: string, codeType: number }\r\n */\r\n @Event() sendSmsCodeRequest: EventEmitter<{ mobile: string; codeType: number }>;\r\n\r\n /**\r\n * 面试模式:text - 文本模式,video - 视频模式\r\n */\r\n @Prop() interviewMode: 'text' | 'video' = 'text';\r\n\r\n /**\r\n * 录制错误事件\r\n */\r\n @Event() recordingError: EventEmitter<RecordingErrorEventData>;\r\n\r\n /**\r\n * 是否显示复制按钮\r\n */\r\n @Prop() showCopyButton: boolean = true;\r\n\r\n /**\r\n * 是否显示点赞点踩按钮\r\n */\r\n @Prop() showFeedbackButtons: boolean = true;\r\n\r\n\r\n @State() selectedFile: File | null = null;\r\n @State() isUploading: boolean = false;\r\n @State() uploadedFileInfo: FileUploadResponse | null = null;\r\n @State() showChatModal: boolean = false;\r\n\r\n // 手机号和验证码相关状态\r\n @State() phoneNumber: string = '';\r\n @State() verificationCode: string = '';\r\n @State() isSendingCode: boolean = false;\r\n @State() countdown: number = 0;\r\n @State() isVerifyingCode: boolean = false;\r\n\r\n // 职位选择器相关状态\r\n @State() showPositionSelector: boolean = false;\r\n\r\n // 使用 @Element 装饰器获取组件的 host 元素\r\n @Element() hostElement: HTMLElement;\r\n\r\n @State() jobDescription: string = '';\r\n @State() isSubmitting: boolean = false;\r\n\r\n private tokenInvalidListener: () => void;\r\n private removeErrorListener: () => void;\r\n\r\n // 添加对子组件的引用\r\n private virtualChatModalRef: HTMLPcmVirtualChatModalElement | null = null;\r\n\r\n @Watch('token')\r\n handleTokenChange(newToken: string) {\r\n // 当传入的 token 变化时,更新 authStore 中的 token\r\n if (newToken && newToken !== authStore.getToken()) {\r\n authStore.setToken(newToken);\r\n }\r\n }\r\n\r\n\r\n @Watch('isOpen')\r\n async handleIsOpenChange(newValue: boolean) {\r\n if (!newValue) {\r\n // 当模态框关闭时,调用子组件方法停止所有活动\r\n if (this.virtualChatModalRef) {\r\n await this.virtualChatModalRef.setComponentActive(false);\r\n }\r\n // 重置状态\r\n this.resetState();\r\n } else {\r\n await verifyApiKey(this.token);\r\n\r\n // 如果有会话ID,直接显示聊天模态框\r\n if (this.conversationId) {\r\n this.showChatModal = true;\r\n }\r\n }\r\n }\r\n\r\n\r\n\r\n componentWillLoad() {\r\n // 将 zIndex 存入配置缓存\r\n if (this.zIndex) {\r\n configStore.setItem('modal-zIndex', this.zIndex);\r\n }\r\n\r\n // 初始化时设置 token,但不验证(验证在 @Watch('isOpen') 中处理)\r\n if (this.token) {\r\n authStore.setToken(this.token);\r\n }\r\n\r\n // 添加全局token无效事件监听器\r\n this.tokenInvalidListener = () => {\r\n this.tokenInvalid.emit();\r\n };\r\n\r\n // 添加全局错误监听\r\n this.removeErrorListener = ErrorEventBus.addErrorListener((errorDetail) => {\r\n this.someErrorEvent.emit(errorDetail);\r\n });\r\n\r\n document.addEventListener('pcm-token-invalid', this.tokenInvalidListener);\r\n }\r\n\r\n disconnectedCallback() {\r\n // 组件销毁时移除事件监听器\r\n document.removeEventListener('pcm-token-invalid', this.tokenInvalidListener);\r\n // 移除错误监听器\r\n if (this.removeErrorListener) {\r\n this.removeErrorListener();\r\n }\r\n // 清理倒计时定时器\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n this.countdownTimer = null;\r\n }\r\n }\r\n\r\n /**\r\n * 重置组件状态\r\n */\r\n private resetState() {\r\n this.showChatModal = false;\r\n this.jobDescription = '';\r\n this.isSubmitting = false;\r\n this.selectedFile = null;\r\n this.isUploading = false;\r\n this.uploadedFileInfo = null;\r\n // 重置手机号和验证码相关状态\r\n this.phoneNumber = '';\r\n this.verificationCode = '';\r\n this.isSendingCode = false;\r\n this.isVerifyingCode = false;\r\n this.countdown = 0;\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n this.countdownTimer = null;\r\n }\r\n }\r\n\r\n private handleClose = () => {\r\n this.modalClosed.emit();\r\n };\r\n\r\n private handleJobDescriptionChange = (event: Event) => {\r\n const textarea = event.target as HTMLTextAreaElement;\r\n this.jobDescription = textarea.value;\r\n };\r\n\r\n private handlePhoneNumberChange = (event: Event) => {\r\n const input = event.target as HTMLInputElement;\r\n this.phoneNumber = input.value;\r\n };\r\n\r\n private handleVerificationCodeChange = (event: Event) => {\r\n const input = event.target as HTMLInputElement;\r\n this.verificationCode = input.value;\r\n };\r\n\r\n private countdownTimer: NodeJS.Timeout | null = null;\r\n\r\n private startCountdown = () => {\r\n this.countdown = 60;\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n }\r\n this.countdownTimer = setInterval(() => {\r\n this.countdown--;\r\n if (this.countdown <= 0) {\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n this.countdownTimer = null;\r\n }\r\n }\r\n }, 1000);\r\n };\r\n\r\n private handleSendCode = () => {\r\n if (!this.phoneNumber.trim()) {\r\n Message.error('请输入手机号');\r\n return;\r\n }\r\n\r\n // 验证手机号格式\r\n const phoneRegex = /^1[3-9]\\d{9}$/;\r\n if (!phoneRegex.test(this.phoneNumber.trim())) {\r\n Message.error('请输入正确的手机号');\r\n return;\r\n }\r\n\r\n // 设置发送中状态\r\n this.isSendingCode = true;\r\n\r\n // 触发事件,让接入方去调用自己的短信接口\r\n this.sendSmsCodeRequest.emit({\r\n mobile: this.phoneNumber.trim(),\r\n codeType: 8\r\n });\r\n\r\n // 开始倒计时并重置发送状态\r\n this.startCountdown();\r\n this.isSendingCode = false;\r\n };\r\n\r\n private handleOpenPositionSelector = () => {\r\n if (!this.oemId) {\r\n Message.info('未配置OEM ID,无法选择职位');\r\n return;\r\n }\r\n this.showPositionSelector = true;\r\n };\r\n\r\n private handleClosePositionSelector = () => {\r\n this.showPositionSelector = false;\r\n };\r\n\r\n private handlePositionSelected = (event: CustomEvent<Position & { formatted_text?: string }>) => {\r\n const position = event.detail;\r\n // 直接使用职位选择器组件提供的格式化文本\r\n this.jobDescription = position.formatted_text || '';\r\n this.showPositionSelector = false;\r\n Message.success('职位信息已填充');\r\n };\r\n\r\n private handleStartInterview = async () => {\r\n // 检查职位描述是否填写(优先使用customInputs.job_info)\r\n if (!this.customInputs?.job_info && !this.jobDescription.trim()) {\r\n Message.error('请输入职位描述');\r\n return;\r\n }\r\n\r\n // 如果填写了手机号,则需要验证验证码\r\n if (this.phoneNumber.trim()) {\r\n // 验证手机号格式\r\n const phoneRegex = /^1[3-9]\\d{9}$/;\r\n if (!phoneRegex.test(this.phoneNumber.trim())) {\r\n Message.error('请输入正确的手机号');\r\n return;\r\n }\r\n\r\n // 验证验证码是否填写\r\n if (!this.verificationCode.trim()) {\r\n Message.error('请输入验证码');\r\n return;\r\n }\r\n\r\n // 验证验证码是否正确\r\n this.isVerifyingCode = true;\r\n try {\r\n const result = await verifySmsCode(this.phoneNumber.trim(), this.verificationCode.trim(), 0);\r\n if (!result.success) {\r\n Message.error(result.error.detail || '验证码错误');\r\n this.isVerifyingCode = false;\r\n return;\r\n }\r\n } catch (error) {\r\n Message.error('验证码校验失败');\r\n this.isVerifyingCode = false;\r\n return;\r\n }\r\n this.isVerifyingCode = false;\r\n }\r\n\r\n // 判断文件是否正在上传\r\n if (await this.pcmUploadRef?.getIsUploading?.()) {\r\n Message.info('文件上传中,请稍后');\r\n return;\r\n }\r\n this.isSubmitting = true;\r\n this.showChatModal = true;\r\n };\r\n\r\n private pcmUploadRef;\r\n\r\n render() {\r\n if (!this.isOpen) return null;\r\n\r\n const modalStyle = {\r\n zIndex: String(this.zIndex)\r\n };\r\n\r\n\r\n const containerClass = {\r\n 'modal-container': true,\r\n 'fullscreen': this.fullscreen,\r\n 'pc-layout': true,\r\n };\r\n\r\n const overlayClass = {\r\n 'modal-overlay': true,\r\n 'fullscreen-overlay': this.fullscreen\r\n };\r\n\r\n // 显示加载状态\r\n const isLoading = this.conversationId && !this.showChatModal;\r\n\r\n // 判断是否禁用JD输入区域(有预设job_info时禁用)\r\n const disableJdInput = Boolean(this.customInputs && this.customInputs.job_info);\r\n\r\n // 判断是否禁用简历上传区域(有预设file_url或resume_content时禁用)\r\n const disableResumeUpload = Boolean(this.customInputs && (this.customInputs.file_url || this.customInputs.resume_content));\r\n\r\n return (\r\n <div class={overlayClass} style={modalStyle}>\r\n <div class={containerClass}>\r\n {this.isShowHeader && (\r\n <div class=\"modal-header\">\r\n <div class=\"header-left\">\r\n {this.icon && <img src={this.icon} class=\"header-icon\" alt=\"应用图标\" />}\r\n <div>{this.modalTitle}</div>\r\n </div>\r\n {this.isNeedClose && (\r\n <button class=\"close-button\" onClick={this.handleClose}>\r\n <span>×</span>\r\n </button>\r\n )}\r\n </div>\r\n )}\r\n\r\n {/* 输入界面 - 仅在不显示聊天模态框且没有会话ID时显示 */}\r\n {!this.showChatModal && !this.conversationId && (\r\n <div class=\"input-container\">\r\n {/* JD输入区域 */}\r\n <div class=\"jd-input-section\">\r\n <div class=\"label-with-button\">\r\n <label htmlFor=\"job-description\">请输入职位描述 (JD)</label>\r\n {!disableJdInput && (\r\n <div class=\"button-group\">\r\n {this.oemId && (\r\n <button\r\n class=\"position-selector-button\"\r\n onClick={this.handleOpenPositionSelector}\r\n type=\"button\"\r\n >\r\n 选择职位\r\n </button>\r\n )}\r\n <pcm-mobile-input-btn\r\n name=\"职位描述\"\r\n onOk={(e) => {\r\n this.jobDescription = e.detail;\r\n }}\r\n />\r\n </div>\r\n )}\r\n </div>\r\n <textarea\r\n id=\"job-description\"\r\n class=\"job-description-textarea\"\r\n placeholder=\"请输入职位描述,包括职责、要求等信息...\"\r\n rows={6}\r\n value={this.customInputs?.job_info || this.jobDescription}\r\n onInput={this.handleJobDescriptionChange}\r\n disabled={disableJdInput}\r\n ></textarea>\r\n </div>\r\n\r\n {/* 简历上传区域 */}\r\n <div class=\"jd-input-section\">\r\n <label>上传简历(选填)</label>\r\n <pcm-upload\r\n ref={el => this.pcmUploadRef = el}\r\n maxFileSize={15 * 1024 * 1024}\r\n multiple={false}\r\n mobileUploadAble={true}\r\n acceptFileSuffixList={['.txt', '.md', '.pdf', '.docx', '.doc']}\r\n uploadParams={{\r\n tags: ['resume'],\r\n }}\r\n defaultFileList={this.customInputs?.file_url ? [{\r\n cos_key: this.customInputs.file_url,\r\n file_name: this.customInputs?.file_name || '预设简历',\r\n file_size: 0,\r\n ext: '',\r\n }] : undefined}\r\n disabled={disableResumeUpload}\r\n onUploadChange={(e) => {\r\n const result: FileUploadResponse[] = e.detail ?? [];\r\n this.uploadedFileInfo = result[0];\r\n this.uploadSuccess.emit(this.uploadedFileInfo);\r\n }}\r\n />\r\n </div>\r\n\r\n {/* 手机号和验证码区域(可选) */}\r\n <div class=\"jd-input-section\">\r\n <label>手机号(选填,填写后报告结果将通过短信发送)</label>\r\n <input\r\n type=\"tel\"\r\n class=\"phone-input\"\r\n placeholder=\"请输入手机号\"\r\n maxLength={11}\r\n value={this.phoneNumber}\r\n onInput={this.handlePhoneNumberChange}\r\n />\r\n </div>\r\n\r\n {/* 验证码输入区域 - 仅在填写了手机号时显示 */}\r\n {this.phoneNumber.trim().length === 11 && (\r\n <div class=\"jd-input-section\">\r\n <label>验证码</label>\r\n <div class=\"verification-code-container\">\r\n <input\r\n type=\"text\"\r\n class=\"verification-code-input\"\r\n placeholder=\"请输入验证码\"\r\n maxLength={6}\r\n value={this.verificationCode}\r\n onInput={this.handleVerificationCodeChange}\r\n />\r\n <button\r\n class=\"send-code-button\"\r\n disabled={this.isSendingCode || this.countdown > 0 || !this.phoneNumber.trim()}\r\n onClick={this.handleSendCode}\r\n >\r\n {this.isSendingCode ? '发送中...' : this.countdown > 0 ? `${this.countdown}秒` : '发送验证码'}\r\n </button>\r\n </div>\r\n </div>\r\n )}\r\n\r\n <button\r\n class=\"submit-button\"\r\n disabled={!(this.customInputs?.job_info || this.jobDescription.trim()) || this.isUploading || this.isSubmitting || this.isVerifyingCode}\r\n onClick={this.handleStartInterview}\r\n >\r\n {this.isUploading ? '上传中...' : this.isVerifyingCode ? '验证中...' : this.isSubmitting ? '处理中...' : '开始面试'}\r\n </button>\r\n\r\n <div class=\"ai-disclaimer\">\r\n <p>所有内容均由AI生成仅供参考</p>\r\n <p class=\"beian-info\">\r\n <span>中央网信办生成式人工智能服务备案号</span>:\r\n <a href=\"https://www.pincaimao.com\" target=\"_blank\" rel=\"noopener noreferrer\">Hunan-PinCaiMao-202412310003</a>\r\n </p>\r\n </div>\r\n </div>\r\n )}\r\n\r\n {/* 加载状态 - 在有会话ID但聊天模态框尚未显示时展示 */}\r\n {isLoading && (\r\n <div class=\"loading-container\">\r\n <div class=\"loading-spinner\"></div>\r\n <p class=\"loading-text\">正在加载对话...</p>\r\n </div>\r\n )}\r\n\r\n {/* 聊天界面 - 在显示聊天模态框时显示 */}\r\n {this.showChatModal && (\r\n <div >\r\n {this.enableVirtualHuman && this.digitalId ? (\r\n <pcm-virtual-chat-modal\r\n ref={el => this.virtualChatModalRef = el}\r\n isOpen={true}\r\n fullscreen={this.fullscreen}\r\n botId=\"3022316191018884\"\r\n digitalId={this.digitalId}\r\n openingIndex={this.openingIndex}\r\n conversationId={this.conversationId}\r\n defaultQuery={this.defaultQuery}\r\n waitingTime={this.waitingTime}\r\n playbackRate={this.playbackRate}\r\n showEndInterviewButton={this.showEndInterviewButton}\r\n customInputs={{\r\n ...this.customInputs,\r\n file_url: this.customInputs?.file_url || this.uploadedFileInfo?.cos_key,\r\n file_name: this.customInputs?.file_name || this.uploadedFileInfo?.file_name,\r\n job_info: this.customInputs?.job_info || this.jobDescription,\r\n question_number: this.customInputs?.question_number || 5,\r\n ai_interview_type: this.digitalId,\r\n phone: this.phoneNumber.trim() || undefined\r\n }}\r\n ></pcm-virtual-chat-modal>\r\n ) : (\r\n <pcm-app-chat-modal\r\n isOpen={true}\r\n modalTitle={this.modalTitle}\r\n icon={this.icon}\r\n isShowHeader={this.isShowHeader}\r\n isNeedClose={this.isShowHeader}\r\n fullscreen={this.fullscreen}\r\n showWorkspaceHistory={this.showWorkspaceHistory}\r\n botId=\"3022316191018884\"\r\n digitalId={this.digitalId}\r\n conversationId={this.conversationId}\r\n defaultQuery={this.defaultQuery}\r\n maxRecordingTime={this.maxRecordingTime}\r\n waitingTime={this.waitingTime}\r\n playbackRate={this.playbackRate}\r\n showCopyButton={this.showCopyButton}\r\n showFeedbackButtons={this.showFeedbackButtons}\r\n showEndInterviewButton={this.showEndInterviewButton}\r\n customInputs={{\r\n ...this.customInputs,\r\n file_url: this.customInputs?.file_url || this.uploadedFileInfo?.cos_key,\r\n file_name: this.customInputs?.file_name || this.uploadedFileInfo?.file_name,\r\n job_info: this.customInputs?.job_info || this.jobDescription,\r\n question_number: this.customInputs?.question_number || 5,\r\n ai_interview_type: this.digitalId || (this.interviewMode === 'video' ? 2 : 1),\r\n phone: this.phoneNumber.trim() || undefined\r\n }}\r\n interviewMode={this.interviewMode}\r\n ></pcm-app-chat-modal>\r\n )}\r\n </div>\r\n )}\r\n </div>\r\n\r\n {/* 职位选择器弹窗 */}\r\n {this.showPositionSelector && (\r\n <div class=\"position-selector-modal-overlay\">\r\n <div class=\"position-selector-modal\">\r\n <div class=\"position-selector-modal-header\">\r\n <h3>选择职位</h3>\r\n <button class=\"position-selector-close-btn\" onClick={this.handleClosePositionSelector}>\r\n ×\r\n </button>\r\n </div>\r\n <div class=\"position-selector-modal-body\">\r\n <components-position-selector\r\n oemId={this.oemId}\r\n multiple={false}\r\n onPositionSelected={this.handlePositionSelected}\r\n ></components-position-selector>\r\n </div>\r\n </div>\r\n </div>\r\n )}\r\n </div>\r\n );\r\n }\r\n} "],"mappings":"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","ignoreList":[]}
1
+ {"version":3,"names":["oemMnmsModalCss","globalCss","hostCss","OemMnmsModal","modalTitle","token","isOpen","modalClosed","icon","zIndex","isShowHeader","isNeedClose","conversationId","defaultQuery","maxRecordingTime","waitingTime","fullscreen","customInputs","showWorkspaceHistory","oemId","digitalId","openingIndex","enableVirtualHuman","showEndInterviewButton","playbackRate","uploadSuccess","streamComplete","conversationStart","interviewComplete","tokenInvalid","someErrorEvent","sendSmsCodeRequest","interviewMode","recordingError","showCopyButton","showFeedbackButtons","selectedFile","isUploading","uploadedFileInfo","showChatModal","phoneNumber","verificationCode","isSendingCode","countdown","isVerifyingCode","showPositionSelector","jobDescription","isSubmitting","tokenInvalidListener","removeErrorListener","virtualChatModalRef","handleTokenChange","newToken","authStore","getToken","setToken","handleIsOpenChange","newValue","this","setComponentActive","resetState","verifyApiKey","componentWillLoad","configStore","setItem","emit","ErrorEventBus","addErrorListener","errorDetail","document","addEventListener","disconnectedCallback","removeEventListener","countdownTimer","clearInterval","handleClose","handleJobDescriptionChange","event","textarea","target","value","handlePhoneNumberChange","input","handleVerificationCodeChange","startCountdown","setInterval","handleSendCode","trim","Message","error","phoneRegex","test","mobile","codeType","handleOpenPositionSelector","info","handleClosePositionSelector","handlePositionSelected","position","detail","formatted_text","success","handleStartInterview","async","job_info","result","verifySmsCode","pcmUploadRef","getIsUploading","render","modalStyle","String","containerClass","overlayClass","isLoading","disableJdInput","Boolean","disableResumeUpload","file_url","resume_content","h","class","style","src","alt","onClick","htmlFor","type","name","onOk","e","id","placeholder","rows","onInput","disabled","ref","el","maxFileSize","multiple","mobileUploadAble","acceptFileSuffixList","uploadParams","tags","defaultFileList","cos_key","file_name","file_size","ext","undefined","onUploadChange","maxLength","length","href","rel","botId","question_number","ai_interview_type","phone","onPositionSelected"],"sources":["src/components/oem-mnms-modal/oem-mnms-modal.css?tag=oem-mnms-modal&encapsulation=shadow","src/global/global.css?tag=oem-mnms-modal&encapsulation=shadow","src/global/host.css?tag=oem-mnms-modal&encapsulation=shadow","src/components/oem-mnms-modal/oem-mnms-modal.tsx"],"sourcesContent":["/* 标签和按钮容器 */\r\n.label-with-button {\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n margin-bottom: 8px;\r\n}\r\n\r\n.button-group {\r\n display: flex;\r\n gap: 8px;\r\n align-items: center;\r\n}\r\n\r\n.position-selector-button {\r\n padding: 4px 12px;\r\n background: #1890ff;\r\n color: white;\r\n border: none;\r\n border-radius: 4px;\r\n font-size: 13px;\r\n cursor: pointer;\r\n transition: all 0.3s ease;\r\n white-space: nowrap;\r\n height: 28px;\r\n line-height: 20px;\r\n}\r\n\r\n.position-selector-button:hover {\r\n background: #40a9ff;\r\n}\r\n\r\n.position-selector-button:active {\r\n background: #096dd9;\r\n}\r\n\r\n/* 手机号输入框样式 */\r\n.phone-input {\r\n width: calc(100% - 16px);\r\n padding: 8px;\r\n border: 1px solid #ddd;\r\n border-radius: 4px;\r\n font-family: inherit;\r\n font-size: 14px;\r\n line-height: 1.5;\r\n transition: border-color 0.3s;\r\n}\r\n\r\n.phone-input:focus {\r\n outline: none;\r\n border-color: #1890ff;\r\n box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);\r\n}\r\n\r\n/* 验证码容器样式 */\r\n.verification-code-container {\r\n display: flex;\r\n gap: 8px;\r\n align-items: center;\r\n}\r\n\r\n.verification-code-input {\r\n flex: 1;\r\n padding: 8px;\r\n border: 1px solid #ddd;\r\n border-radius: 4px;\r\n font-family: inherit;\r\n font-size: 14px;\r\n line-height: 1.5;\r\n transition: border-color 0.3s;\r\n}\r\n\r\n.verification-code-input:focus {\r\n outline: none;\r\n border-color: #1890ff;\r\n box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);\r\n}\r\n\r\n/* 发送验证码按钮样式 */\r\n.send-code-button {\r\n padding: 8px 16px;\r\n background: #0D75FB;\r\n color: white;\r\n border: none;\r\n border-radius: 4px;\r\n font-size: 14px;\r\n cursor: pointer;\r\n transition: all 0.3s ease;\r\n white-space: nowrap;\r\n min-width: 100px;\r\n}\r\n\r\n.send-code-button:hover:not(:disabled) {\r\n background-color: #40a9ff;\r\n}\r\n\r\n.send-code-button:disabled {\r\n background-color: rgba(0, 0, 0, 0.04);\r\n color: rgba(0, 0, 0, 0.25);\r\n cursor: not-allowed;\r\n}\r\n\r\n/* 职位选择器弹窗样式 */\r\n.position-selector-modal-overlay {\r\n position: fixed;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n background-color: rgba(0, 0, 0, 0.5);\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n z-index: 3000;\r\n padding: 20px;\r\n}\r\n\r\n.position-selector-modal {\r\n background: white;\r\n border-radius: 8px;\r\n width: 90%;\r\n max-width: 900px;\r\n height: 80vh;\r\n max-height: 600px;\r\n display: flex;\r\n flex-direction: column;\r\n box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);\r\n animation: modalFadeIn 0.3s ease;\r\n}\r\n\r\n@keyframes modalFadeIn {\r\n from {\r\n opacity: 0;\r\n transform: scale(0.95) translateY(-20px);\r\n }\r\n to {\r\n opacity: 1;\r\n transform: scale(1) translateY(0);\r\n }\r\n}\r\n\r\n.position-selector-modal-header {\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n padding: 16px 24px;\r\n border-bottom: 1px solid #f0f0f0;\r\n flex-shrink: 0;\r\n}\r\n\r\n.position-selector-modal-header h3 {\r\n margin: 0;\r\n font-size: 16px;\r\n font-weight: 500;\r\n color: rgba(0, 0, 0, 0.85);\r\n}\r\n\r\n.position-selector-close-btn {\r\n background: none;\r\n border: none;\r\n font-size: 24px;\r\n line-height: 1;\r\n cursor: pointer;\r\n color: rgba(0, 0, 0, 0.45);\r\n padding: 0;\r\n width: 32px;\r\n height: 32px;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n border-radius: 4px;\r\n transition: all 0.3s;\r\n}\r\n\r\n.position-selector-close-btn:hover {\r\n background-color: rgba(0, 0, 0, 0.06);\r\n color: rgba(0, 0, 0, 0.85);\r\n}\r\n\r\n.position-selector-modal-body {\r\n flex: 1;\r\n overflow: hidden;\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n\r\n/* 移动端适配 */\r\n@media screen and (max-width: 768px) {\r\n .position-selector-modal {\r\n width: 100%;\r\n height: 90vh;\r\n max-height: none;\r\n border-radius: 8px 8px 0 0;\r\n margin-top: auto;\r\n }\r\n\r\n .position-selector-modal-overlay {\r\n padding: 0;\r\n align-items: flex-end;\r\n }\r\n}\r\n","/* 模态框基础样式 */\r\n.modal-overlay {\r\n position: fixed;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n background-color: rgba(0, 0, 0, 0.5);\r\n display: flex;\r\n z-index: 1000;\r\n overflow-y: auto;\r\n}\r\n\r\n/* 全屏模式下的overlay样式 - 改为基于父组件 */\r\n.fullscreen-overlay {\r\n padding: 0;\r\n background-color: rgba(0, 0, 0, 0.7);\r\n /* 改为基于父组件的全屏 */\r\n position: absolute;\r\n width: 100%;\r\n height: auto;\r\n}\r\n\r\n.modal-container {\r\n background-color: #fff;\r\n border-radius: 8px;\r\n width: 100%;\r\n display: flex;\r\n flex-direction: column;\r\n position: relative;\r\n margin: auto;\r\n transition: all 0.3s ease-out;\r\n overflow: hidden;\r\n}\r\n\r\n/* 全屏模式样式 - 改为基于父组件 */\r\n.modal-container.fullscreen {\r\n width: 100%;\r\n max-width: none;\r\n height: 100%;\r\n border-radius: 0;\r\n margin: 0;\r\n display: flex;\r\n flex-direction: column;\r\n max-height: 100%;\r\n}\r\n\r\n/* PC端布局 */\r\n.pc-layout {\r\n width: 80%;\r\n max-width: 600px;\r\n min-width: 320px;\r\n}\r\n\r\n/* 响应式布局 */\r\n@media screen and (max-width: 768px) {\r\n .pc-layout {\r\n width: 95%;\r\n }\r\n\r\n .modal-overlay {\r\n padding: 10px 0px 0px 0px;\r\n }\r\n\r\n .modal-container.fullscreen {\r\n /* 移动端也基于父组件尺寸 */\r\n width: 100%;\r\n height: 100%;\r\n max-height: 100%;\r\n border-radius: 16px 16px 0 0;\r\n /* 保留安全区域支持 */\r\n padding: env(safe-area-inset-top) 0 env(safe-area-inset-bottom);\r\n }\r\n}\r\n\r\n/* 模态框头部样式 */\r\n.modal-header {\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n padding: 4px 16px;\r\n height: 50px;\r\n border-bottom: 1px solid #e8e8e8;\r\n flex-shrink: 0;\r\n}\r\n\r\n.header-left {\r\n display: flex;\r\n align-items: center;\r\n gap: 8px;\r\n font-size: 16px;\r\n font-weight: 600;\r\n color: #333;\r\n}\r\n\r\n.header-icon {\r\n width: 24px;\r\n height: 24px;\r\n}\r\n\r\n.close-button {\r\n background: transparent;\r\n border: none;\r\n cursor: pointer;\r\n padding: 8px;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 32px;\r\n height: 32px;\r\n border-radius: 4px;\r\n}\r\n\r\n.close-button:hover {\r\n background-color: rgba(0, 0, 0, 0.04);\r\n}\r\n\r\n.close-button span {\r\n font-size: 24px;\r\n line-height: 1;\r\n color: #999;\r\n}\r\n\r\n.close-button:hover span {\r\n color: #666;\r\n}\r\n\r\n\r\n/* 文件上传区域通用样式 */\r\n.upload-area {\r\n cursor: pointer;\r\n width: 100%;\r\n}\r\n\r\n\r\n.upload-placeholder {\r\n transition: all 0.3s ease;\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n background: rgba(0, 0, 0, 0.02);\r\n border: 1px dashed #d9d9d9;\r\n border-radius: 8px;\r\n cursor: pointer;\r\n}\r\n\r\n.upload-placeholder:hover {\r\n border: 1px dashed #1890ff;\r\n}\r\n\r\n.upload-placeholder img {\r\n margin-top: 8px;\r\n width: 50px;\r\n height: 50px;\r\n}\r\n\r\n.upload-placeholder .upload-text {\r\n margin: 4px 0;\r\n color: #332F39;\r\n font-size: 14px;\r\n}\r\n\r\n.upload-placeholder .upload-hint {\r\n font-size: 14px;\r\n color: #949AA5;\r\n margin-top: 8px;\r\n padding: 0px 10px;\r\n text-align: center;\r\n white-space: normal;\r\n word-wrap: break-word;\r\n word-break: break-all;\r\n line-height: 1.4;\r\n}\r\n\r\n\r\n/* 文件项样式 */\r\n.file-item {\r\n position: relative;\r\n padding: 16px;\r\n border: 1px solid #e2e8f0;\r\n border-radius: 8px;\r\n transition: border-color 0.3s;\r\n cursor: pointer;\r\n margin-bottom: 16px;\r\n display: flex;\r\n justify-content: space-between;\r\n align-items: center;\r\n}\r\n\r\n.file-item:hover {\r\n border-color: #0D75FB;\r\n}\r\n\r\n.file-item-content {\r\n display: flex;\r\n align-items: center;\r\n gap: 8px;\r\n flex: 1;\r\n min-width: 0;\r\n overflow: hidden;\r\n}\r\n\r\n.file-icon {\r\n color: #0D75FB;\r\n flex-shrink: 0;\r\n}\r\n\r\n.file-name {\r\n font-weight: 500;\r\n white-space: nowrap;\r\n overflow: hidden;\r\n text-overflow: ellipsis;\r\n max-width: calc(100% - 50px);\r\n}\r\n\r\n.remove-file {\r\n background: transparent;\r\n border: none;\r\n color: #94a3b8;\r\n cursor: pointer;\r\n font-size: 18px;\r\n display: flex;\r\n align-items: center;\r\n justify-content: center;\r\n padding: 4px;\r\n margin-left: 8px;\r\n border-radius: 4px;\r\n transition: all 0.2s;\r\n min-width: 30px;\r\n min-height: 30px;\r\n}\r\n\r\n.remove-file:hover {\r\n background-color: #f1f5f9;\r\n color: #475569;\r\n}\r\n\r\n.file-input {\r\n display: none;\r\n}\r\n\r\n\r\n\r\n/* 输入容器样式 */\r\n.input-container {\r\n padding: 20px;\r\n display: flex;\r\n flex-direction: column;\r\n height: calc(100% - 50px);\r\n background: linear-gradient(150deg, #2a6ee933, #0000 50%) 0 0 / 400px 200px no-repeat, #fff;\r\n /* 减去header高度 */\r\n overflow-y: auto;\r\n}\r\n\r\n.input-container h3 {\r\n margin-top: 0;\r\n margin-bottom: 20px;\r\n font-size: 18px;\r\n color: #333;\r\n text-align: center;\r\n}\r\n\r\n/* JD输入区域样式 */\r\n.jd-input-section {\r\n margin-bottom: 20px;\r\n}\r\n\r\n.jd-input-section label {\r\n display: block;\r\n margin-bottom: 8px;\r\n font-weight: 500;\r\n color: #333;\r\n}\r\n\r\n.job-description-textarea {\r\n width: calc(100% - 16px);\r\n border: 1px solid #ddd;\r\n border-radius: 4px;\r\n resize: vertical;\r\n font-family: inherit;\r\n font-size: 14px;\r\n line-height: 1.5;\r\n transition: border-color 0.3s;\r\n padding: 8px;\r\n}\r\n\r\n.job-description-textarea:focus {\r\n outline: none;\r\n border-color: #1890ff;\r\n box-shadow: 0 0 0 2px rgba(24, 144, 255, 0.2);\r\n}\r\n\r\n/* 简历上传区域样式 */\r\n.resume-upload-section {\r\n margin-bottom: 20px;\r\n width: 100%;\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n\r\n.resume-upload-section label {\r\n display: block;\r\n margin-bottom: 8px;\r\n font-weight: 500;\r\n color: #333;\r\n}\r\n\r\n/* 面试设置区域 */\r\n.interview-settings-section {\r\n margin-bottom: 24px;\r\n padding: 20px;\r\n background-color: #f8f9fa;\r\n border-radius: 8px;\r\n border: 1px solid #e9ecef;\r\n}\r\n\r\n.settings-row {\r\n margin-bottom: 20px;\r\n}\r\n\r\n.settings-row:last-child {\r\n margin-bottom: 0;\r\n}\r\n\r\n.settings-label {\r\n display: block;\r\n font-weight: 500;\r\n color: #333;\r\n margin-bottom: 12px;\r\n font-size: 14px;\r\n}\r\n\r\n/* 题目数量滑块样式 */\r\n.question-number-section {\r\n width: 100%;\r\n}\r\n\r\n.slider-container {\r\n display: flex;\r\n align-items: center;\r\n gap: 12px;\r\n}\r\n\r\n.question-slider {\r\n flex: 1;\r\n height: 6px;\r\n background: #ddd;\r\n outline: none;\r\n border-radius: 3px;\r\n -webkit-appearance: none;\r\n appearance: none;\r\n}\r\n\r\n.question-slider::-webkit-slider-thumb {\r\n -webkit-appearance: none;\r\n appearance: none;\r\n width: 20px;\r\n height: 20px;\r\n background: #0D75FB;\r\n cursor: pointer;\r\n border-radius: 50%;\r\n box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);\r\n}\r\n\r\n.question-slider::-moz-range-thumb {\r\n width: 20px;\r\n height: 20px;\r\n background: #0D75FB;\r\n cursor: pointer;\r\n border-radius: 50%;\r\n border: none;\r\n box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);\r\n}\r\n\r\n.slider-value {\r\n font-weight: 600;\r\n color: #0D75FB;\r\n font-size: 14px;\r\n min-width: 30px;\r\n text-align: center;\r\n}\r\n\r\n/* 开关样式 */\r\n.analysis-toggle-section {\r\n width: 100%;\r\n}\r\n\r\n.toggle-container {\r\n display: flex;\r\n align-items: center;\r\n gap: 12px;\r\n}\r\n\r\n.toggle-switch {\r\n position: relative;\r\n display: inline-block;\r\n width: 44px;\r\n height: 24px;\r\n}\r\n\r\n.toggle-switch input {\r\n opacity: 0;\r\n width: 0;\r\n height: 0;\r\n}\r\n\r\n.toggle-slider {\r\n position: absolute;\r\n cursor: pointer;\r\n top: 0;\r\n left: 0;\r\n right: 0;\r\n bottom: 0;\r\n background-color: #ccc;\r\n transition: 0.3s;\r\n border-radius: 24px;\r\n}\r\n\r\n.toggle-slider:before {\r\n position: absolute;\r\n content: \"\";\r\n height: 18px;\r\n width: 18px;\r\n left: 3px;\r\n bottom: 3px;\r\n background-color: white;\r\n transition: 0.3s;\r\n border-radius: 50%;\r\n box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);\r\n}\r\n\r\ninput:checked + .toggle-slider {\r\n background-color: #0D75FB;\r\n}\r\n\r\ninput:checked + .toggle-slider:before {\r\n transform: translateX(20px);\r\n}\r\n\r\n.toggle-text {\r\n font-size: 14px;\r\n color: #666;\r\n font-weight: 500;\r\n}\r\n\r\n/* 提交按钮通用样式 */\r\n.submit-button {\r\n margin-top: 10px;\r\n padding: 10px 30px;\r\n background: #0D75FB;\r\n color: white;\r\n border: none;\r\n border-radius: 4px;\r\n font-size: 16px;\r\n cursor: pointer;\r\n transition: all 0.3s ease;\r\n width: 100%;\r\n max-width: 400px;\r\n align-self: center;\r\n}\r\n\r\n.submit-button:hover {\r\n background-color: #40a9ff;\r\n}\r\n\r\n.submit-button:disabled {\r\n background-color: rgba(0,0,0,0.04);\r\n color: rgba(0,0,0,0.25);\r\n cursor: not-allowed;\r\n}\r\n\r\n\r\n\r\n/* AI免责声明和备案信息样式 */\r\n.ai-disclaimer {\r\n margin-top: 16px;\r\n text-align: center;\r\n font-size: 12px;\r\n color: #999;\r\n line-height: 1.5;\r\n}\r\n\r\n.ai-disclaimer p {\r\n margin: 4px 0;\r\n}\r\n\r\n.beian-info {\r\n display: flex;\r\n justify-content: center;\r\n flex-wrap: wrap;\r\n gap: 4px;\r\n}\r\n\r\n.ai-disclaimer a {\r\n color: #666;\r\n text-decoration: none;\r\n transition: color 0.2s ease;\r\n}\r\n\r\n.ai-disclaimer a:hover {\r\n color: #1890ff;\r\n text-decoration: underline;\r\n}\r\n\r\n/* 添加加载状态的样式 */\r\n.loading-container {\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n justify-content: center;\r\n height: 100%;\r\n padding: 24px;\r\n }\r\n \r\n .loading-spinner {\r\n width: 40px;\r\n height: 40px;\r\n border: 4px solid rgba(0, 0, 0, 0.1);\r\n border-radius: 50%;\r\n border-top-color: var(--pcm-primary-color, #1890ff);\r\n animation: spin 1s linear infinite;\r\n margin-bottom: 16px;\r\n }\r\n \r\n .loading-text {\r\n font-size: 16px;\r\n color: var(--pcm-text-color, #333);\r\n }\r\n \r\n @keyframes spin {\r\n 0% { transform: rotate(0deg); }\r\n 100% { transform: rotate(360deg); }\r\n }\r\n\r\n/* 历史会话按钮 */\r\n.header-history-button {\r\n display: inline-flex;\r\n align-items: center;\r\n justify-content: center;\r\n gap: 4px;\r\n padding: 4px 8px;\r\n border: none;\r\n border-radius: 4px;\r\n background: transparent;\r\n color: #bbb;\r\n cursor: pointer;\r\n font-size: 13px;\r\n font-family: inherit;\r\n line-height: 1;\r\n transition: all 0.2s ease;\r\n flex-shrink: 0;\r\n white-space: nowrap;\r\n}\r\n\r\n.header-history-button svg {\r\n width: 16px;\r\n height: 16px;\r\n display: block;\r\n flex-shrink: 0;\r\n}\r\n\r\n.header-history-button:hover {\r\n background-color: rgba(0, 0, 0, 0.04);\r\n color: #666;\r\n}\r\n\r\n.header-history-button:focus-visible {\r\n outline: 2px solid rgba(13, 117, 251, 0.35);\r\n outline-offset: 2px;\r\n}\r\n\r\n/* 历史会话抽屉内容 */\r\n.history-drawer-content {\r\n height: 100%;\r\n display: flex;\r\n flex-direction: column;\r\n}\r\n\r\n.conversation-list {\r\n flex: 1;\r\n overflow-y: auto;\r\n padding: 8px 0;\r\n}\r\n\r\n.loading-conversations,\r\n.empty-conversations {\r\n display: flex;\r\n flex-direction: column;\r\n align-items: center;\r\n justify-content: center;\r\n padding: 40px 20px;\r\n color: #999;\r\n}\r\n\r\n.loading-conversations .loading-spinner-small {\r\n margin-bottom: 12px;\r\n}\r\n\r\n.conversation-item {\r\n display: flex;\r\n align-items: center;\r\n justify-content: space-between;\r\n padding: 12px 16px;\r\n margin: 0 8px 4px 8px;\r\n border-radius: 8px;\r\n cursor: pointer;\r\n transition: all 0.2s ease;\r\n border: 1px solid transparent;\r\n}\r\n\r\n.conversation-item:hover {\r\n background: #f5f5f5;\r\n}\r\n\r\n.conversation-info {\r\n flex: 1;\r\n min-width: 0;\r\n}\r\n\r\n.conversation-title {\r\n font-size: 14px;\r\n font-weight: 500;\r\n color: #262626;\r\n margin-bottom: 4px;\r\n overflow: hidden;\r\n text-overflow: ellipsis;\r\n white-space: nowrap;\r\n}\r\n\r\n.conversation-meta {\r\n display: flex;\r\n align-items: center;\r\n gap: 8px;\r\n font-size: 12px;\r\n color: #8c8c8c;\r\n}\r\n\r\n.conversation-time {\r\n flex-shrink: 0;\r\n}\r\n\r\n.message-count {\r\n flex-shrink: 0;\r\n}\r\n\r\n.conversation-status {\r\n padding: 2px 6px;\r\n border-radius: 4px;\r\n font-size: 11px;\r\n font-weight: 500;\r\n}\r\n\r\n.conversation-status.completed {\r\n background: #f6ffed;\r\n color: #52c41a;\r\n}\r\n\r\n.conversation-status.running {\r\n background: #fff7e6;\r\n color: #fa8c16;\r\n}\r\n\r\n.header-history-entry {\r\n display: inline-flex;\r\n align-items: center;\r\n justify-content: center;\r\n margin-left: 8px;\r\n padding: 4px 8px;\r\n border: none;\r\n border-radius: 4px;\r\n background: #e6f7ff;\r\n color: #1890ff;\r\n cursor: pointer;\r\n font-size: 12px;\r\n font-family: inherit;\r\n line-height: 1;\r\n white-space: nowrap;\r\n transition: all 0.2s ease;\r\n}\r\n\r\n.header-history-entry:hover {\r\n background: #bae7ff;\r\n color: #0958d9;\r\n transform: translateY(-1px);\r\n}\r\n\r\n.header-history-entry:focus-visible {\r\n outline: 2px solid rgba(13, 117, 251, 0.35);\r\n outline-offset: 2px;\r\n}\r\n\r\n.history-pagination {\r\n display: flex;\r\n align-items: center;\r\n justify-content: space-between;\r\n gap: 12px;\r\n padding: 12px 16px 16px;\r\n border-top: 1px solid #f0f0f0;\r\n}\r\n\r\n.history-pagination-button {\r\n min-width: 72px;\r\n padding: 6px 12px;\r\n border: 1px solid #d9d9d9;\r\n border-radius: 6px;\r\n background: #fff;\r\n color: #595959;\r\n cursor: pointer;\r\n font-size: 12px;\r\n transition: all 0.2s ease;\r\n}\r\n\r\n.history-pagination-button:hover:not(:disabled) {\r\n border-color: #1890ff;\r\n color: #1890ff;\r\n background: #e6f7ff;\r\n}\r\n\r\n.history-pagination-button:disabled {\r\n border-color: #f0f0f0;\r\n color: #bfbfbf;\r\n background: #fafafa;\r\n cursor: not-allowed;\r\n}\r\n\r\n.history-pagination-info {\r\n flex: 1;\r\n text-align: center;\r\n color: #8c8c8c;\r\n font-size: 12px;\r\n}\r\n \r\n",":host {\r\n /* 字体相关属性 */\r\n font-size: 16px;\r\n font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;\r\n font-weight: normal;\r\n font-style: normal;\r\n font-variant: normal;\r\n line-height: 1.5;\r\n letter-spacing: normal;\r\n word-spacing: normal;\r\n \r\n /* 文本相关属性 */\r\n color: #333; /* 重置文字颜色,避免继承外部的red */\r\n text-align: left;\r\n text-indent: 0;\r\n text-transform: none;\r\n white-space: normal;\r\n word-wrap: normal;\r\n word-break: normal;\r\n \r\n /* 可见性 */\r\n visibility: visible;\r\n \r\n /* 列表相关 */\r\n list-style: none;\r\n list-style-image: none;\r\n list-style-position: outside;\r\n list-style-type: disc;\r\n \r\n /* 表格相关 */\r\n border-collapse: separate;\r\n border-spacing: 0;\r\n caption-side: top;\r\n empty-cells: show;\r\n table-layout: auto;\r\n \r\n /* 引用 */\r\n quotes: auto;\r\n \r\n /* 光标 */\r\n cursor: auto;\r\n \r\n /* 文本方向 */\r\n direction: ltr;\r\n \r\n /* 打印相关 */\r\n orphans: 2;\r\n widows: 2;\r\n page-break-inside: auto;\r\n \r\n /* 其他可能被继承的属性 */\r\n text-decoration: none;\r\n text-shadow: none;\r\n \r\n /* 确保组件有独立的盒模型 */\r\n box-sizing: border-box;\r\n \r\n /* 重置一些可能影响布局的继承属性 */\r\n writing-mode: horizontal-tb;\r\n text-orientation: mixed;\r\n}","import { Component, Prop, h, State, Element, Event, EventEmitter, Watch } from '@stencil/core';\r\nimport { FileUploadResponse, verifyApiKey, verifySmsCode, Position } from '../../utils/utils';\r\nimport {\r\n StreamCompleteEventData,\r\n ConversationStartEventData,\r\n InterviewCompleteEventData,\r\n RecordingErrorEventData,\r\n} from '../../interfaces/events';\r\nimport { ErrorEventBus, ErrorEventDetail } from '../../utils/error-event';\r\nimport { authStore } from '../../../store/auth.store';\r\nimport { configStore } from '../../../store/config.store';\r\nimport { Message } from '../../services/message.service';\r\n\r\n/**\r\n * 模拟面试(求职)\r\n */\r\n\r\n@Component({\r\n tag: 'oem-mnms-modal',\r\n styleUrls: ['oem-mnms-modal.css', '../../global/global.css', '../../global/host.css'],\r\n shadow: true,\r\n})\r\nexport class OemMnmsModal {\r\n /**\r\n * 模态框标题\r\n */\r\n @Prop() modalTitle: string = '模拟面试';\r\n\r\n /**\r\n * SDK鉴权密钥\r\n */\r\n @Prop({ attribute: 'token' }) token!: string;\r\n\r\n /**\r\n * 是否显示聊天模态框\r\n */\r\n @Prop({ mutable: true }) isOpen: boolean = false;\r\n\r\n /**\r\n * 当点击模态框关闭时触发\r\n */\r\n @Event() modalClosed: EventEmitter<void>;\r\n\r\n /**\r\n * 应用图标URL\r\n */\r\n @Prop() icon?: string;\r\n\r\n /**\r\n * 聊天框的页面层级\r\n */\r\n @Prop() zIndex?: number = 1000;\r\n\r\n /**\r\n * 是否展示顶部标题栏\r\n */\r\n @Prop() isShowHeader: boolean = true;\r\n\r\n /**\r\n * 是否展示右上角的关闭按钮\r\n */\r\n @Prop() isNeedClose: boolean = true;\r\n\r\n /**\r\n * 会话ID,传入继续对话,否则创建新会话\r\n */\r\n @Prop({ mutable: true }) conversationId?: string;\r\n\r\n /**\r\n * 默认查询文本\r\n */\r\n @Prop() defaultQuery: string = '请开始模拟面试';\r\n\r\n /**\r\n * 视频录制最大时长(秒)默认120\r\n */\r\n @Prop() maxRecordingTime: number = 120;\r\n\r\n /**\r\n * 等待录制时间(秒)默认10\r\n */\r\n @Prop() waitingTime: number = 10;\r\n\r\n /**\r\n * 是否以全屏模式打开,移动端建议设置为true\r\n */\r\n @Prop() fullscreen: boolean = false;\r\n\r\n /**\r\n * 自定义输入参数,可传入以下参数:<br>\r\n * customInputs.job_info时,JD输入区域不可修改。<br>\r\n * customInputs.file_url或customInputs.resume_content时,简历上传区域不可修改。<br>\r\n * customInputs.resume_content:可传入json字符串,或纯文本字符串,字符串内容为简历内容。<br>\r\n * customInputs.url_callback:可传入url字符串,当报告生成后,会调用该url进行回调。该url请使用post请求,接收报告字段为report_content,会话id字段为conversation_id。<br>\r\n * customInputs.question_number时,会设置面试题总数量, 默认5题。<br>\r\n */\r\n @Prop() customInputs: Record<string, string> = {};\r\n\r\n /**\r\n * 是否显示工作区历史会话按钮\r\n */\r\n @Prop() showWorkspaceHistory: boolean = false;\r\n\r\n /**\r\n * OEM ID,用于获取职位列表\r\n */\r\n @Prop() oemId?: string;\r\n\r\n /**\r\n * 虚拟数字人ID,指定则开启虚拟数字人功能</br>\r\n * 智琪:79202506752 </br>\r\n * 艾綝:79402522624 </br>\r\n * 艾姮:79597885440 </br>\r\n */\r\n @Prop() digitalId?: string;\r\n\r\n /**\r\n * 数字人开场白索引,用于选择开场白和开场视频(可选:0, 1, 2)\r\n * 0、您好,我是聘才猫 AI 面试助手。很高兴为你主持这场面试!在开始前,请确保:身处安静、光线充足的环境。网络顺畅,摄像头和麦克风工作正常。现在我正在查看本次面试的相关信息,为您生成专属面试题,马上就好,请稍等片刻。</br>\r\n * 1、您好,我是您的 AI 面试助手。欢迎参加本次AI面试!为了获得最佳效果,请确认:您在安静、明亮的环境中。您的网络稳定,摄像头和麦克风已开启。我们正在后台为您准备本次专属面试内容,很快开始,请稍候。<br>\r\n * 2、您好,我是您的 AI 面试助手。面试马上开始。趁此片刻,请快速确认:周围安静吗?光线足够吗?网络没问题?摄像头和麦克风准备好了吗?我们正在为您加载个性化的面试环节,稍等就好!\r\n */\r\n @Prop() openingIndex: number = 0;\r\n\r\n /**\r\n * 是否启用全屏数字人模式,此模式下面试报告只会通过url_callback回调返回\r\n */\r\n @Prop() enableVirtualHuman: boolean = false;\r\n\r\n /**\r\n * 是否显示结束面试按钮\r\n */\r\n @Prop() showEndInterviewButton: boolean = false;\r\n\r\n /**\r\n * 数字人视频播放速率(0.5-2.0之间),默认1.0\r\n */\r\n @Prop() playbackRate: number = 1.0;\r\n\r\n /**\r\n * 上传成功事件\r\n */\r\n @Event() uploadSuccess: EventEmitter<FileUploadResponse>;\r\n\r\n /**\r\n * 流式输出完成事件\r\n */\r\n @Event() streamComplete: EventEmitter<StreamCompleteEventData>;\r\n\r\n /**\r\n * 新会话开始的回调,只会在一轮对话开始时触发一次\r\n */\r\n @Event() conversationStart: EventEmitter<ConversationStartEventData>;\r\n\r\n /**\r\n * 当聊天完成时触发\r\n */\r\n @Event() interviewComplete: EventEmitter<InterviewCompleteEventData>;\r\n\r\n /**\r\n * SDK密钥验证失败事件\r\n */\r\n @Event() tokenInvalid: EventEmitter<void>;\r\n\r\n /**\r\n * 错误事件\r\n */\r\n @Event() someErrorEvent: EventEmitter<ErrorEventDetail>;\r\n\r\n /**\r\n * 发送短信验证码请求事件\r\n * 接入方需要监听此事件并调用自己的短信接口发送验证码\r\n * 事件参数: { mobile: string, codeType: number }\r\n */\r\n @Event() sendSmsCodeRequest: EventEmitter<{ mobile: string; codeType: number }>;\r\n\r\n /**\r\n * 面试模式:text - 文本模式,video - 视频模式\r\n */\r\n @Prop() interviewMode: 'text' | 'video' = 'text';\r\n\r\n /**\r\n * 录制错误事件\r\n */\r\n @Event() recordingError: EventEmitter<RecordingErrorEventData>;\r\n\r\n /**\r\n * 是否显示复制按钮\r\n */\r\n @Prop() showCopyButton: boolean = true;\r\n\r\n /**\r\n * 是否显示点赞点踩按钮\r\n */\r\n @Prop() showFeedbackButtons: boolean = true;\r\n\r\n\r\n @State() selectedFile: File | null = null;\r\n @State() isUploading: boolean = false;\r\n @State() uploadedFileInfo: FileUploadResponse | null = null;\r\n @State() showChatModal: boolean = false;\r\n\r\n // 手机号和验证码相关状态\r\n @State() phoneNumber: string = '';\r\n @State() verificationCode: string = '';\r\n @State() isSendingCode: boolean = false;\r\n @State() countdown: number = 0;\r\n @State() isVerifyingCode: boolean = false;\r\n\r\n // 职位选择器相关状态\r\n @State() showPositionSelector: boolean = false;\r\n\r\n // 使用 @Element 装饰器获取组件的 host 元素\r\n @Element() hostElement: HTMLElement;\r\n\r\n @State() jobDescription: string = '';\r\n @State() isSubmitting: boolean = false;\r\n\r\n private tokenInvalidListener: () => void;\r\n private removeErrorListener: () => void;\r\n\r\n // 添加对子组件的引用\r\n private virtualChatModalRef: HTMLPcmVirtualChatModalElement | null = null;\r\n\r\n @Watch('token')\r\n handleTokenChange(newToken: string) {\r\n // 当传入的 token 变化时,更新 authStore 中的 token\r\n if (newToken && newToken !== authStore.getToken()) {\r\n authStore.setToken(newToken);\r\n }\r\n }\r\n\r\n\r\n @Watch('isOpen')\r\n async handleIsOpenChange(newValue: boolean) {\r\n if (!newValue) {\r\n // 当模态框关闭时,调用子组件方法停止所有活动\r\n if (this.virtualChatModalRef) {\r\n await this.virtualChatModalRef.setComponentActive(false);\r\n }\r\n // 重置状态\r\n this.resetState();\r\n } else {\r\n await verifyApiKey(this.token);\r\n\r\n // 如果有会话ID,直接显示聊天模态框\r\n if (this.conversationId) {\r\n this.showChatModal = true;\r\n }\r\n }\r\n }\r\n\r\n\r\n\r\n componentWillLoad() {\r\n // 将 zIndex 存入配置缓存\r\n if (this.zIndex) {\r\n configStore.setItem('modal-zIndex', this.zIndex);\r\n }\r\n\r\n // 初始化时设置 token,但不验证(验证在 @Watch('isOpen') 中处理)\r\n if (this.token) {\r\n authStore.setToken(this.token);\r\n }\r\n\r\n // 添加全局token无效事件监听器\r\n this.tokenInvalidListener = () => {\r\n this.tokenInvalid.emit();\r\n };\r\n\r\n // 添加全局错误监听\r\n this.removeErrorListener = ErrorEventBus.addErrorListener((errorDetail) => {\r\n this.someErrorEvent.emit(errorDetail);\r\n });\r\n\r\n document.addEventListener('pcm-token-invalid', this.tokenInvalidListener);\r\n }\r\n\r\n disconnectedCallback() {\r\n // 组件销毁时移除事件监听器\r\n document.removeEventListener('pcm-token-invalid', this.tokenInvalidListener);\r\n // 移除错误监听器\r\n if (this.removeErrorListener) {\r\n this.removeErrorListener();\r\n }\r\n // 清理倒计时定时器\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n this.countdownTimer = null;\r\n }\r\n }\r\n\r\n /**\r\n * 重置组件状态\r\n */\r\n private resetState() {\r\n this.showChatModal = false;\r\n this.jobDescription = '';\r\n this.isSubmitting = false;\r\n this.selectedFile = null;\r\n this.isUploading = false;\r\n this.uploadedFileInfo = null;\r\n // 重置手机号和验证码相关状态\r\n this.phoneNumber = '';\r\n this.verificationCode = '';\r\n this.isSendingCode = false;\r\n this.isVerifyingCode = false;\r\n this.countdown = 0;\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n this.countdownTimer = null;\r\n }\r\n }\r\n\r\n private handleClose = () => {\r\n this.modalClosed.emit();\r\n };\r\n\r\n private handleJobDescriptionChange = (event: Event) => {\r\n const textarea = event.target as HTMLTextAreaElement;\r\n this.jobDescription = textarea.value;\r\n };\r\n\r\n private handlePhoneNumberChange = (event: Event) => {\r\n const input = event.target as HTMLInputElement;\r\n this.phoneNumber = input.value;\r\n };\r\n\r\n private handleVerificationCodeChange = (event: Event) => {\r\n const input = event.target as HTMLInputElement;\r\n this.verificationCode = input.value;\r\n };\r\n\r\n private countdownTimer: NodeJS.Timeout | null = null;\r\n\r\n private startCountdown = () => {\r\n this.countdown = 60;\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n }\r\n this.countdownTimer = setInterval(() => {\r\n this.countdown--;\r\n if (this.countdown <= 0) {\r\n if (this.countdownTimer) {\r\n clearInterval(this.countdownTimer);\r\n this.countdownTimer = null;\r\n }\r\n }\r\n }, 1000);\r\n };\r\n\r\n private handleSendCode = () => {\r\n if (!this.phoneNumber.trim()) {\r\n Message.error('请输入手机号');\r\n return;\r\n }\r\n\r\n // 验证手机号格式\r\n const phoneRegex = /^1[3-9]\\d{9}$/;\r\n if (!phoneRegex.test(this.phoneNumber.trim())) {\r\n Message.error('请输入正确的手机号');\r\n return;\r\n }\r\n\r\n // 设置发送中状态\r\n this.isSendingCode = true;\r\n\r\n // 触发事件,让接入方去调用自己的短信接口\r\n this.sendSmsCodeRequest.emit({\r\n mobile: this.phoneNumber.trim(),\r\n codeType: 8\r\n });\r\n\r\n // 开始倒计时并重置发送状态\r\n this.startCountdown();\r\n this.isSendingCode = false;\r\n };\r\n\r\n private handleOpenPositionSelector = () => {\r\n if (!this.oemId) {\r\n Message.info('未配置OEM ID,无法选择职位');\r\n return;\r\n }\r\n this.showPositionSelector = true;\r\n };\r\n\r\n private handleClosePositionSelector = () => {\r\n this.showPositionSelector = false;\r\n };\r\n\r\n private handlePositionSelected = (event: CustomEvent<Position & { formatted_text?: string }>) => {\r\n const position = event.detail;\r\n // 直接使用职位选择器组件提供的格式化文本\r\n this.jobDescription = position.formatted_text || '';\r\n this.showPositionSelector = false;\r\n Message.success('职位信息已填充');\r\n };\r\n\r\n private handleStartInterview = async () => {\r\n // 检查职位描述是否填写(优先使用customInputs.job_info)\r\n if (!this.customInputs?.job_info && !this.jobDescription.trim()) {\r\n Message.error('请输入职位描述');\r\n return;\r\n }\r\n\r\n // 如果填写了手机号,则需要验证验证码\r\n if (this.phoneNumber.trim()) {\r\n // 验证手机号格式\r\n const phoneRegex = /^1[3-9]\\d{9}$/;\r\n if (!phoneRegex.test(this.phoneNumber.trim())) {\r\n Message.error('请输入正确的手机号');\r\n return;\r\n }\r\n\r\n // 验证验证码是否填写\r\n if (!this.verificationCode.trim()) {\r\n Message.error('请输入验证码');\r\n return;\r\n }\r\n\r\n // 验证验证码是否正确\r\n this.isVerifyingCode = true;\r\n try {\r\n const result = await verifySmsCode(this.phoneNumber.trim(), this.verificationCode.trim(), 0);\r\n if (!result.success) {\r\n Message.error(result.error.detail || '验证码错误');\r\n this.isVerifyingCode = false;\r\n return;\r\n }\r\n } catch (error) {\r\n Message.error('验证码校验失败');\r\n this.isVerifyingCode = false;\r\n return;\r\n }\r\n this.isVerifyingCode = false;\r\n }\r\n\r\n // 判断文件是否正在上传\r\n if (await this.pcmUploadRef?.getIsUploading?.()) {\r\n Message.info('文件上传中,请稍后');\r\n return;\r\n }\r\n this.isSubmitting = true;\r\n this.showChatModal = true;\r\n };\r\n\r\n private pcmUploadRef;\r\n\r\n render() {\r\n if (!this.isOpen) return null;\r\n\r\n const modalStyle = {\r\n zIndex: String(this.zIndex)\r\n };\r\n\r\n\r\n const containerClass = {\r\n 'modal-container': true,\r\n 'fullscreen': this.fullscreen,\r\n 'pc-layout': true,\r\n };\r\n\r\n const overlayClass = {\r\n 'modal-overlay': true,\r\n 'fullscreen-overlay': this.fullscreen\r\n };\r\n\r\n // 显示加载状态\r\n const isLoading = this.conversationId && !this.showChatModal;\r\n\r\n // 判断是否禁用JD输入区域(有预设job_info时禁用)\r\n const disableJdInput = Boolean(this.customInputs && this.customInputs.job_info);\r\n\r\n // 判断是否禁用简历上传区域(有预设file_url或resume_content时禁用)\r\n const disableResumeUpload = Boolean(this.customInputs && (this.customInputs.file_url || this.customInputs.resume_content));\r\n\r\n return (\r\n <div class={overlayClass} style={modalStyle}>\r\n <div class={containerClass}>\r\n {this.isShowHeader && (\r\n <div class=\"modal-header\">\r\n <div class=\"header-left\">\r\n {this.icon && <img src={this.icon} class=\"header-icon\" alt=\"应用图标\" />}\r\n <div>{this.modalTitle}</div>\r\n </div>\r\n {this.isNeedClose && (\r\n <button class=\"close-button\" onClick={this.handleClose}>\r\n <span>×</span>\r\n </button>\r\n )}\r\n </div>\r\n )}\r\n\r\n {/* 输入界面 - 仅在不显示聊天模态框且没有会话ID时显示 */}\r\n {!this.showChatModal && !this.conversationId && (\r\n <div class=\"input-container\">\r\n {/* JD输入区域 */}\r\n <div class=\"jd-input-section\">\r\n <div class=\"label-with-button\">\r\n <label htmlFor=\"job-description\">请输入职位描述 (JD)</label>\r\n {!disableJdInput && (\r\n <div class=\"button-group\">\r\n {this.oemId && (\r\n <button\r\n class=\"position-selector-button\"\r\n onClick={this.handleOpenPositionSelector}\r\n type=\"button\"\r\n >\r\n 选择职位\r\n </button>\r\n )}\r\n <pcm-mobile-input-btn\r\n name=\"职位描述\"\r\n onOk={(e) => {\r\n this.jobDescription = e.detail;\r\n }}\r\n />\r\n </div>\r\n )}\r\n </div>\r\n <textarea\r\n id=\"job-description\"\r\n class=\"job-description-textarea\"\r\n placeholder=\"请输入职位描述,包括职责、要求等信息...\"\r\n rows={6}\r\n value={this.customInputs?.job_info || this.jobDescription}\r\n onInput={this.handleJobDescriptionChange}\r\n disabled={disableJdInput}\r\n ></textarea>\r\n </div>\r\n\r\n {/* 简历上传区域 */}\r\n <div class=\"jd-input-section\">\r\n <label>上传简历(选填)</label>\r\n <pcm-upload\r\n ref={el => this.pcmUploadRef = el}\r\n maxFileSize={15 * 1024 * 1024}\r\n multiple={false}\r\n mobileUploadAble={true}\r\n acceptFileSuffixList={['.txt', '.md', '.pdf', '.docx', '.doc']}\r\n uploadParams={{\r\n tags: ['resume'],\r\n }}\r\n defaultFileList={this.customInputs?.file_url ? [{\r\n cos_key: this.customInputs.file_url,\r\n file_name: this.customInputs?.file_name || '预设简历',\r\n file_size: 0,\r\n ext: '',\r\n }] : undefined}\r\n disabled={disableResumeUpload}\r\n onUploadChange={(e) => {\r\n const result: FileUploadResponse[] = e.detail ?? [];\r\n this.uploadedFileInfo = result[0];\r\n this.uploadSuccess.emit(this.uploadedFileInfo);\r\n }}\r\n />\r\n </div>\r\n\r\n {/* 手机号和验证码区域(可选) */}\r\n <div class=\"jd-input-section\">\r\n <label>手机号(选填,填写后报告结果将通过短信发送)</label>\r\n <input\r\n type=\"tel\"\r\n class=\"phone-input\"\r\n placeholder=\"请输入手机号\"\r\n maxLength={11}\r\n value={this.phoneNumber}\r\n onInput={this.handlePhoneNumberChange}\r\n />\r\n </div>\r\n\r\n {/* 验证码输入区域 - 仅在填写了手机号时显示 */}\r\n {this.phoneNumber.trim().length === 11 && (\r\n <div class=\"jd-input-section\">\r\n <label>验证码</label>\r\n <div class=\"verification-code-container\">\r\n <input\r\n type=\"text\"\r\n class=\"verification-code-input\"\r\n placeholder=\"请输入验证码\"\r\n maxLength={6}\r\n value={this.verificationCode}\r\n onInput={this.handleVerificationCodeChange}\r\n />\r\n <button\r\n class=\"send-code-button\"\r\n disabled={this.isSendingCode || this.countdown > 0 || !this.phoneNumber.trim()}\r\n onClick={this.handleSendCode}\r\n >\r\n {this.isSendingCode ? '发送中...' : this.countdown > 0 ? `${this.countdown}秒` : '发送验证码'}\r\n </button>\r\n </div>\r\n </div>\r\n )}\r\n\r\n <button\r\n class=\"submit-button\"\r\n disabled={!(this.customInputs?.job_info || this.jobDescription.trim()) || this.isUploading || this.isSubmitting || this.isVerifyingCode}\r\n onClick={this.handleStartInterview}\r\n >\r\n {this.isUploading ? '上传中...' : this.isVerifyingCode ? '验证中...' : this.isSubmitting ? '处理中...' : '开始面试'}\r\n </button>\r\n\r\n <div class=\"ai-disclaimer\">\r\n <p>所有内容均由AI生成仅供参考</p>\r\n <p class=\"beian-info\">\r\n <span>中央网信办生成式人工智能服务备案号</span>:\r\n <a href=\"https://www.pincaimao.com\" target=\"_blank\" rel=\"noopener noreferrer\">Hunan-PinCaiMao-202412310003</a>\r\n </p>\r\n </div>\r\n </div>\r\n )}\r\n\r\n {/* 加载状态 - 在有会话ID但聊天模态框尚未显示时展示 */}\r\n {isLoading && (\r\n <div class=\"loading-container\">\r\n <div class=\"loading-spinner\"></div>\r\n <p class=\"loading-text\">正在加载对话...</p>\r\n </div>\r\n )}\r\n\r\n {/* 聊天界面 - 在显示聊天模态框时显示 */}\r\n {this.showChatModal && (\r\n <div >\r\n {this.enableVirtualHuman && this.digitalId ? (\r\n <pcm-virtual-chat-modal\r\n ref={el => this.virtualChatModalRef = el}\r\n isOpen={true}\r\n fullscreen={this.fullscreen}\r\n botId=\"3022316191018884\"\r\n digitalId={this.digitalId}\r\n openingIndex={this.openingIndex}\r\n conversationId={this.conversationId}\r\n defaultQuery={this.defaultQuery}\r\n waitingTime={this.waitingTime}\r\n playbackRate={this.playbackRate}\r\n showEndInterviewButton={this.showEndInterviewButton}\r\n customInputs={{\r\n ...this.customInputs,\r\n file_url: this.customInputs?.file_url || this.uploadedFileInfo?.cos_key,\r\n file_name: this.customInputs?.file_name || this.uploadedFileInfo?.file_name,\r\n job_info: this.customInputs?.job_info || this.jobDescription,\r\n question_number: this.customInputs?.question_number || 5,\r\n ai_interview_type: this.digitalId,\r\n phone: this.phoneNumber.trim() || undefined\r\n }}\r\n ></pcm-virtual-chat-modal>\r\n ) : (\r\n <pcm-app-chat-modal\r\n isOpen={true}\r\n modalTitle={this.modalTitle}\r\n icon={this.icon}\r\n isShowHeader={this.isShowHeader}\r\n isNeedClose={this.isShowHeader}\r\n fullscreen={this.fullscreen}\r\n showWorkspaceHistory={this.showWorkspaceHistory}\r\n botId=\"3022316191018884\"\r\n digitalId={this.digitalId}\r\n conversationId={this.conversationId}\r\n defaultQuery={this.defaultQuery}\r\n maxRecordingTime={this.maxRecordingTime}\r\n waitingTime={this.waitingTime}\r\n playbackRate={this.playbackRate}\r\n showCopyButton={this.showCopyButton}\r\n showFeedbackButtons={this.showFeedbackButtons}\r\n showEndInterviewButton={this.showEndInterviewButton}\r\n customInputs={{\r\n ...this.customInputs,\r\n file_url: this.customInputs?.file_url || this.uploadedFileInfo?.cos_key,\r\n file_name: this.customInputs?.file_name || this.uploadedFileInfo?.file_name,\r\n job_info: this.customInputs?.job_info || this.jobDescription,\r\n question_number: this.customInputs?.question_number || 5,\r\n ai_interview_type: this.digitalId || (this.interviewMode === 'video' ? \"2\" : \"1\"),\r\n phone: this.phoneNumber.trim() || undefined\r\n }}\r\n interviewMode={this.interviewMode}\r\n ></pcm-app-chat-modal>\r\n )}\r\n </div>\r\n )}\r\n </div>\r\n\r\n {/* 职位选择器弹窗 */}\r\n {this.showPositionSelector && (\r\n <div class=\"position-selector-modal-overlay\">\r\n <div class=\"position-selector-modal\">\r\n <div class=\"position-selector-modal-header\">\r\n <h3>选择职位</h3>\r\n <button class=\"position-selector-close-btn\" onClick={this.handleClosePositionSelector}>\r\n ×\r\n </button>\r\n </div>\r\n <div class=\"position-selector-modal-body\">\r\n <components-position-selector\r\n oemId={this.oemId}\r\n multiple={false}\r\n onPositionSelected={this.handlePositionSelected}\r\n ></components-position-selector>\r\n </div>\r\n </div>\r\n </div>\r\n )}\r\n </div>\r\n );\r\n }\r\n} "],"mappings":"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","ignoreList":[]}