ChatAgentInput.vue 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393
  1. <script setup>
  2. import { ref, unref, onMounted, onUnmounted, computed, watch } from 'vue';
  3. import { useMessage, NInput, NSwitch, NPopover, NScrollbar } from 'naive-ui';
  4. import dayjs from 'dayjs';
  5. import { getFormatYesterDay } from '@/utils/format';
  6. import { helperApi } from '@/api/helper';
  7. import SvgIcon from '@/components/SvgIcon';
  8. import 'load-awesome/css/ball-running-dots.min.css';
  9. const props = defineProps({
  10. activeItem: {
  11. type: Object,
  12. default: () => ({})
  13. }
  14. });
  15. const emit = defineEmits(['onClick', 'onEnter']);
  16. const MAX_NUM = 5;
  17. const modelLoading = defineModel('loading');
  18. const switchStatus = defineModel('switch');
  19. const message = useMessage();
  20. const inpVal = ref('');
  21. const inpRef = ref(null);
  22. const isFocusState = ref(false);
  23. const isOpen = ref(false);
  24. const highlightedIndex = ref(0);
  25. const selectedOption = ref(null);
  26. const helperList = ref([]);
  27. const popoverTriggerRef = ref(null);
  28. const popoverInnerRef = ref(null);
  29. const scrollRef = ref(null);
  30. const agentOptions = computed(() => helperList.value.filter(({ tools }) => tools));
  31. const focusInput = _ => isFocusState.value = true;
  32. const blurInput = _ => isFocusState.value = false;
  33. watch(inpVal, (curVal) => {
  34. if (curVal === "@" && curVal.length === 1) {
  35. if ( !unref(agentOptions).length ) {
  36. return message.warning('当前未配置智能体');
  37. }
  38. if ( modelLoading.value ) {
  39. return message.warning('当前对话进行中');
  40. }
  41. isOpen.value = true;
  42. } else {
  43. isOpen.value = false;
  44. }
  45. })
  46. watch(() => props.activeItem, (curVal) => {
  47. selectedOption.value = curVal?.tools ? curVal : null;
  48. })
  49. const handleInpFocus = () => {
  50. inpRef.value?.focus();
  51. }
  52. const commonEmitEvent = (eventName) => {
  53. const val = unref(inpVal);
  54. const len = val.trim().length;
  55. if ( !len ) {
  56. return message.warning('请输入您的问题或需求');
  57. }
  58. if ( len > 2000 ) {
  59. return message.warning('问题限制2000个字以内');
  60. }
  61. if ( modelLoading.value ) {
  62. return message.warning('当前对话进行中');
  63. }
  64. emit(eventName, { question: val, selectedOption: selectedOption.value || {} });
  65. inpVal.value = '';
  66. }
  67. // 回车事件
  68. const handleInpEnter = (event) => {
  69. if (event.key === 'Enter' && !event.shiftKey && inpVal.value) {
  70. event.preventDefault();
  71. commonEmitEvent('onEnter');
  72. inpRef.value?.blur();
  73. }
  74. }
  75. // 点击事件
  76. const handleBtnClick = () => {
  77. commonEmitEvent("onClick");
  78. }
  79. const clearInpVal = () => {
  80. inpVal.value = '';
  81. }
  82. // 键盘上键无限滚动
  83. const scrollKeyUp = (index) => {
  84. const itemLength = agentOptions.value.length;
  85. if ( itemLength - MAX_NUM > index ) {
  86. scrollRef.value.scrollBy({
  87. top: -40 ,
  88. behavior: 'smooth'
  89. })
  90. }
  91. if ( index == itemLength - 1 ) {
  92. scrollRef.value.scrollTo({
  93. top: itemLength * 40,
  94. behavior: 'smooth'
  95. })
  96. }
  97. }
  98. // 键盘下键无限滚动
  99. const scrollKeyDown = (index) => {
  100. if ( index >= MAX_NUM ) {
  101. scrollRef.value.scrollBy({
  102. top: 40,
  103. behavior: 'smooth'
  104. })
  105. }
  106. if ( index == 0 ) {
  107. scrollRef.value.scrollTo({
  108. top: 0,
  109. behavior: 'smooth'
  110. })
  111. }
  112. }
  113. // 键盘事件
  114. const handleKeyDown = (event) => {
  115. const len = unref(agentOptions).length;
  116. if ( !isOpen.value ) return;
  117. switch (event.key) {
  118. case 'ArrowUp':
  119. event.preventDefault();
  120. highlightedIndex.value = (unref(highlightedIndex) - 1 + len) % len;
  121. scrollKeyUp(unref(highlightedIndex), 1)
  122. break;
  123. case 'ArrowDown':
  124. event.preventDefault();
  125. highlightedIndex.value = (unref(highlightedIndex) + 1) % len;
  126. scrollKeyDown(unref(highlightedIndex))
  127. break;
  128. case 'Enter':
  129. event.preventDefault();
  130. selectOption(unref(highlightedIndex));
  131. break;
  132. default:
  133. break;
  134. }
  135. }
  136. // 处理点击空白处关闭
  137. const closePopoverOutside =(event) => {
  138. if (!isOpen.value) return;
  139. const triggerResult = popoverTriggerRef.value.contains(event.target);
  140. const innerResult = popoverInnerRef.value.contains(event.target);
  141. isOpen.value = triggerResult || innerResult;
  142. }
  143. // 选中选项
  144. const selectOption = (index) => {
  145. selectedOption.value = agentOptions.value[index];
  146. highlightedIndex.value = index;
  147. isOpen.value = false;
  148. inpVal.value = selectedOption.value.content;
  149. }
  150. onMounted(async () => {
  151. const { data } = await helperApi.getHelperList();
  152. const result = getFormatYesterDay(data)
  153. console.log("result", result);
  154. helperList.value = result;
  155. document.addEventListener('keydown', handleKeyDown);
  156. document.addEventListener('click', closePopoverOutside);
  157. })
  158. onUnmounted(() => {
  159. document.removeEventListener('keydown', handleKeyDown);
  160. document.removeEventListener('click', closePopoverOutside);
  161. })
  162. defineExpose({
  163. clearInpVal,
  164. handleInpFocus,
  165. inpVal,
  166. })
  167. </script>
  168. <template>
  169. <NPopover
  170. trigger="hover"
  171. width="trigger"
  172. display-directive="show"
  173. content-style="padding: 0;"
  174. :show-arrow="false"
  175. :show="isOpen"
  176. >
  177. <template #trigger>
  178. <div class="popover-trigger" ref="popoverTriggerRef">
  179. <div class="chat-inp-outer border-[1px]" :class="[{ 'border-[#2454FF]': isFocusState }]">
  180. <ul class="chat-tools-inner py-[10px] px-[10px] bg-[#fcfcfc]" v-show="selectedOption">
  181. <li class="tools-tips space-x-[10px]">
  182. <span>与</span>
  183. <p class="agent-name space-x-[5px]" @click="isOpen = true">
  184. <img src="https://static.fuxicarbon.com/userupload/db77ffe0cef843278a23b0d2db9505fa.png" alt="">
  185. <span>{{ selectedOption?.title }}</span>
  186. </p>
  187. <span>对话中</span>
  188. </li>
  189. <li class="tools-close" @click="selectedOption = null">
  190. <SvgIcon name="chat-icon-close-btn"></SvgIcon>
  191. </li>
  192. </ul>
  193. <div class="chat-inp-inner">
  194. <div class="inp-wrapper flex-1" @click="handleInpFocus">
  195. <NInput
  196. class="flex-1"
  197. ref="inpRef"
  198. type="textarea"
  199. size="medium"
  200. placeholder="输入@,召唤智能体"
  201. v-model:value="inpVal"
  202. :autosize="{ minRows: 1, maxRows: 5 }"
  203. @focus="focusInput"
  204. @blur="blurInput"
  205. @keypress="handleInpEnter"
  206. />
  207. </div>
  208. <div class="submit-btn">
  209. <button class="btn bg-[#1A2029] hover:bg-[#3C4148]" @click="handleBtnClick">
  210. <SvgIcon name="tool-send-plane" size="22" v-show="!modelLoading"></SvgIcon>
  211. <div style="color: #fff" class="la-ball-running-dots la-sm" v-show="modelLoading">
  212. <div v-for="item in 5" :key="item"></div>
  213. </div>
  214. </button>
  215. </div>
  216. </div>
  217. </div>
  218. <div class="switch-inner pt-[8px] space-x-[6px]">
  219. <NSwitch size="small" v-model:value="switchStatus"></NSwitch>
  220. <span class="text-[12px] text-[#9E9E9E]">使用搜索增强</span>
  221. </div>
  222. <div class="masking-inner text-center text-[#2454FF]"></div>
  223. </div>
  224. </template>
  225. <div class="popover-inner" ref="popoverInnerRef">
  226. <div class="header">
  227. <span>选择智能体</span>
  228. <p class="tools-close" @click="isOpen = false">
  229. <SvgIcon name="chat-icon-close-btn"></SvgIcon>
  230. </p>
  231. </div>
  232. <NScrollbar style="max-height: 200px;" ref="scrollRef" trigger="none">
  233. <div class="item" v-for="item, index in agentOptions" :class="['item', { active: highlightedIndex === index }]" @click="selectOption(index)">
  234. <p class="icon">
  235. <img :src="item.banner" alt="">
  236. </p>
  237. <p class="ml-[10px] space-x-[5px] text">
  238. <span class="text-[15px]">{{item.title}}</span>
  239. <!-- <span class="text-[#888] text-[14px]">这里可以补充个描述</span> -->
  240. </p>
  241. </div>
  242. </NScrollbar>
  243. </div>
  244. </NPopover>
  245. </template>
  246. <style scoped lang="scss">
  247. .chat-inp-outer {
  248. border-radius: 8px;
  249. overflow: hidden;
  250. box-shadow: 0px 3px 12px 0px #97D3FF40;
  251. .chat-tools-inner {
  252. @include flex(x, center, between);
  253. .tools-tips {
  254. @include flex(x, center, start);
  255. color: #666;
  256. font-size: 14px;
  257. .agent-name {
  258. @include flex(x, center, start);
  259. font-weight: bold;
  260. color: #333;
  261. cursor: pointer;
  262. img {
  263. width: 14px;
  264. height: 14px;
  265. }
  266. }
  267. }
  268. }
  269. .chat-inp-inner {
  270. position: relative;
  271. @include flex(x, center, between);
  272. background: #fff;
  273. .inp-wrapper {
  274. padding: 17px 0px 17px 34px;
  275. }
  276. .submit-btn {
  277. @include flex(x, center, center);
  278. width: 84px;
  279. .btn {
  280. @include flex(x, center, center);
  281. width: 50px;
  282. height: 32px;
  283. border-radius: 32px;
  284. transition: all .3s;
  285. }
  286. }
  287. }
  288. }
  289. .popover-inner {
  290. .header {
  291. @include flex(x, center, between);
  292. padding-bottom: 8px;
  293. font-size: 14px;
  294. color: #666;
  295. }
  296. .item {
  297. @include flex(x, center, start);
  298. padding: 8px 10px;
  299. cursor: pointer;
  300. &:hover {
  301. background: #f0fafe;
  302. }
  303. .icon {
  304. @include flex(x, center, center);
  305. width: 24px;
  306. height: 24px;
  307. border-radius: 100%;
  308. background: #e9eef8;
  309. img {
  310. width: 16px;
  311. height: 16px;
  312. }
  313. }
  314. .text {
  315. text-align: left;
  316. overflow: hidden;
  317. white-space: nowrap;
  318. text-overflow: ellipsis;
  319. }
  320. }
  321. .active {
  322. background: #f0fafe;
  323. }
  324. }
  325. .tools-close {
  326. @include flex(x, center, center);
  327. width: 28px;
  328. height: 28px;
  329. border-radius: 6px;
  330. background: #fff;
  331. cursor: pointer;
  332. &:hover {
  333. background: #e9eef8;
  334. }
  335. }
  336. .masking-inner {
  337. position: absolute;
  338. top: -30px;
  339. left: 0;
  340. width: 100%;
  341. height: 30px;
  342. background: linear-gradient(180deg, rgba(232, 241, 250, 0) 0%, #E7F0FA 95%);
  343. }
  344. </style>