ChatInputCopy.vue 9.0 KB

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