SearchPage.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. <template>
  2. <div class="container">
  3. <!-- 左侧知识库列表 -->
  4. <div class="knowledge-base">
  5. <h3>选择知识库</h3>
  6. <div
  7. v-for="item in knowledgeBaseList"
  8. :key="item.dataset_id"
  9. :class="['knowledge-item', { 'active': selectedDatasetIds.includes(item.dataset_id) }]"
  10. @click="toggleDatasetSelection(item.dataset_id)"
  11. >
  12. {{ item.name }}
  13. </div>
  14. </div>
  15. <!-- 右侧搜索框和搜索结果 -->
  16. <div class="search-area">
  17. <div class="search-container">
  18. <el-input
  19. v-model="query"
  20. placeholder="请输入搜索内容"
  21. suffix-icon="el-icon-search"
  22. @keyup.enter="search"
  23. class="search-input"
  24. ></el-input>
  25. <el-button @click="search" type="primary" style="margin-left: 15px">搜索</el-button>
  26. </div>
  27. <!-- 搜索中的提示 -->
  28. <div v-if="loading" class="loading-spinner">搜索中...</div>
  29. <div class="search-results">
  30. <el-card
  31. v-for="result in searchResults"
  32. :key="result.contentSummary"
  33. class="result-card"
  34. @click="handleDetails(result)"
  35. >
  36. <h3>{{ result.contentSummary }}</h3>
  37. <p>{{ result.content.substring(0, 100) }}...</p>
  38. <div class="meta">
  39. <span>相似度: {{ result.score.toFixed(2) }}</span>
  40. <span>知识库: {{ result.datasetName }}</span>
  41. </div>
  42. </el-card>
  43. </div>
  44. </div>
  45. </div>
  46. <!-- 弹窗:展示完整内容 -->
  47. <el-dialog v-model="dialogVisible" width="80%">
  48. <div>
  49. <h3>{{ selectedResult.contentSummary }}</h3>
  50. <p>{{ selectedResult.content }}</p>
  51. <hr />
  52. <div>
  53. <h4>原文内容:</h4>
  54. <p>{{ originalContent }}</p>
  55. </div>
  56. </div>
  57. <template #footer>
  58. <el-button @click="dialogVisible = false">关闭</el-button>
  59. </template>
  60. </el-dialog>
  61. </template>
  62. <script setup>
  63. import { ref, onMounted } from 'vue';
  64. import { ElMessage } from 'element-plus';
  65. // 存储选择的知识库数据
  66. const knowledgeBaseList = ref([]);
  67. const selectedDatasetIds = ref([]);
  68. // 搜索框输入内容
  69. const query = ref('');
  70. // 存储搜索结果
  71. const searchResults = ref([]);
  72. // 弹窗显示状态
  73. const dialogVisible = ref(false);
  74. // 存储选中的搜索结果
  75. const selectedResult = ref({});
  76. // 存储原文内容
  77. const originalContent = ref('');
  78. // 搜索加载状态
  79. const loading = ref(false);
  80. // 请求知识库列表
  81. const getKnowledgeBaseList = async () => {
  82. try {
  83. const response = await fetch('http://127.0.0.1:8001/api/dataset/list');
  84. const data = await response.json();
  85. knowledgeBaseList.value = data.data;
  86. } catch (error) {
  87. ElMessage.error('获取知识库列表失败');
  88. }
  89. };
  90. // 选择或取消选择知识库
  91. const toggleDatasetSelection = (datasetId) => {
  92. const index = selectedDatasetIds.value.indexOf(datasetId);
  93. if (index === -1) {
  94. selectedDatasetIds.value.push(datasetId);
  95. } else {
  96. selectedDatasetIds.value.splice(index, 1);
  97. }
  98. };
  99. // 执行搜索操作
  100. const search = async () => {
  101. if (!query.value.trim()) {
  102. ElMessage.warning('请输入搜索内容');
  103. return;
  104. }
  105. if (selectedDatasetIds.value.length === 0) {
  106. ElMessage.warning('请先选择知识库');
  107. return;
  108. }
  109. loading.value = true; // 开始搜索时显示加载提示
  110. const datasetIds = selectedDatasetIds.value.join(',');
  111. try {
  112. const response = await fetch(`http://127.0.0.1:8001/api/query?query=${query.value}&datasetIds=${datasetIds}`);
  113. const data = await response.json();
  114. searchResults.value = data.data.results.map((item) => ({
  115. ...item,
  116. // 这里只是简化了展示,可以将 datasetName 从返回结果中提取
  117. }));
  118. } catch (error) {
  119. ElMessage.error('搜索失败');
  120. } finally {
  121. loading.value = false; // 搜索结束后隐藏加载提示
  122. }
  123. };
  124. // 展示选中的搜索结果的完整内容
  125. const handleDetails = async (result) => {
  126. selectedResult.value = result;
  127. dialogVisible.value = true; // 打开弹窗
  128. // 请求完整内容
  129. try {
  130. const response = await fetch(`http://127.0.0.1:8001/api/content/get?docId=${result.docId}`);
  131. const data = await response.json();
  132. if (data.status_code === 200) {
  133. originalContent.value = data.data.text; // 显示原文内容
  134. } else {
  135. ElMessage.error('获取原文内容失败');
  136. }
  137. } catch (error) {
  138. ElMessage.error('请求原文内容失败');
  139. }
  140. };
  141. // 页面初始化加载知识库列表
  142. onMounted(() => {
  143. getKnowledgeBaseList();
  144. });
  145. </script>
  146. <style scoped>
  147. .container {
  148. display: flex;
  149. justify-content: space-between;
  150. padding: 20px;
  151. height: 100vh; /* 设置容器高度为视口高度 */
  152. }
  153. .knowledge-base {
  154. width: 20%;
  155. background-color: #f9f9f9;
  156. padding: 15px;
  157. border-radius: 8px;
  158. height: 100%; /* 确保高度为100% */
  159. overflow-y: auto; /* 启用垂直滚动 */
  160. }
  161. .knowledge-item {
  162. padding: 10px;
  163. border-radius: 5px;
  164. cursor: pointer;
  165. margin: 5px 0;
  166. transition: background-color 0.3s ease;
  167. }
  168. .knowledge-item:hover {
  169. background-color: #e6f7ff;
  170. }
  171. .knowledge-item.active {
  172. background-color: #b3d8ff;
  173. }
  174. .search-area {
  175. width: 75%;
  176. background-color: #ffffff;
  177. padding: 20px;
  178. border-radius: 8px;
  179. box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1);
  180. height: 100%; /* 确保高度为100% */
  181. overflow-y: auto; /* 启用垂直滚动 */
  182. }
  183. .search-container {
  184. display: flex;
  185. justify-content: center;
  186. align-items: center;
  187. margin-bottom: 20px;
  188. }
  189. .search-input {
  190. width: 60%; /* 控制搜索框的宽度 */
  191. }
  192. .result-card {
  193. margin-top: 10px;
  194. cursor: pointer;
  195. }
  196. .result-card:hover {
  197. background-color: #f5f5f5;
  198. }
  199. .meta {
  200. display: flex;
  201. justify-content: space-between;
  202. font-size: 12px;
  203. color: #888;
  204. }
  205. .loading-spinner {
  206. text-align: center;
  207. font-size: 16px;
  208. color: #888;
  209. margin-top: 20px;
  210. }
  211. </style>