index.vue 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. <script lang="ts">
  2. export default {
  3. name: "Search",
  4. };
  5. </script>
  6. <script setup lang="ts">
  7. import { ref, computed, watch } from "vue";
  8. import { Search } from "@element-plus/icons-vue";
  9. import { watchDebounced, useFocus } from "@vueuse/core";
  10. import { useI18n } from "vue-i18n";
  11. import _ from "lodash";
  12. const { t } = useI18n();
  13. const input = ref("");
  14. const input$ = ref();
  15. const { focused } = useFocus(computed(() => input$.value?.input));
  16. const suggestions = ref<unknown[]>([]);
  17. const loading = ref(false);
  18. const suggestionVisible = computed(() => {
  19. // TEST
  20. // return true
  21. const isValidData = suggestions.value.length > 0;
  22. return focused.value && (isValidData || loading.value);
  23. });
  24. watch(
  25. () => input.value,
  26. (val) => {
  27. console.log(input$.value);
  28. loading.value = !!val;
  29. if (!val) {
  30. suggestions.value = [];
  31. }
  32. }
  33. );
  34. const fetchSuggestions = (mock) => {
  35. return new Promise((resolve, reject) => {
  36. setTimeout(() => {
  37. // MOCK
  38. // TODO should we use local search and gpt search together?
  39. resolve(["vue", "react", mock]);
  40. }, 1000);
  41. });
  42. };
  43. watchDebounced(
  44. () => input.value,
  45. async (taggedInput) => {
  46. if (!taggedInput) {
  47. return;
  48. }
  49. const result = await fetchSuggestions(taggedInput);
  50. if (taggedInput === input.value) {
  51. suggestions.value = result as unknown[];
  52. loading.value = false;
  53. }
  54. },
  55. { debounce: 500 }
  56. );
  57. </script>
  58. <template>
  59. <div class="search-container">
  60. <el-popover
  61. :visible="suggestionVisible"
  62. :show-arrow="false"
  63. :offset="0"
  64. :teleported="false"
  65. width="100%"
  66. >
  67. <template #reference>
  68. <div class="search-trigger" :class="{ 'has-content': suggestionVisible }">
  69. <el-input
  70. :ref="(el) => (input$ = el)"
  71. v-model="input"
  72. clearable
  73. :prefix-icon="Search"
  74. :placeholder="t('请输入关键词,如:课程、协同、AI')"
  75. ></el-input>
  76. </div>
  77. </template>
  78. <div class="search-content">
  79. <template v-if="loading">
  80. <span>loading</span>
  81. </template>
  82. <template v-else>
  83. <ul>
  84. <li v-for="(suggest, _index) in suggestions" :key="_index">{{ suggest }}</li>
  85. </ul>
  86. </template>
  87. </div>
  88. </el-popover>
  89. </div>
  90. </template>
  91. <i18n locale="zh-HK">
  92. {
  93. "请输入关键词,如:课程、协同、AI": "TODO",
  94. }
  95. </i18n>
  96. <style lang="scss" scoped>
  97. .search-container {
  98. width: 514px;
  99. position: relative;
  100. .search-trigger {
  101. border: 1px solid #aeccfe;
  102. padding: 1px;
  103. width: 100%;
  104. height: 52px;
  105. border-radius: 26px;
  106. display: flex;
  107. align-items: center;
  108. padding: 0 10px;
  109. overflow: hidden;
  110. transition: all 0.2s;
  111. :deep(.el-input) {
  112. .el-input__wrapper {
  113. box-shadow: none;
  114. }
  115. }
  116. &:has(input:focus) {
  117. border: none;
  118. box-shadow: var(--el-box-shadow-light);
  119. }
  120. &.has-content {
  121. border-bottom: 1px solid #e2eeff;
  122. border-radius: 26px 26px 0 0;
  123. }
  124. }
  125. :deep(.el-popover) {
  126. border-radius: 0 0 26px 26px;
  127. border: none;
  128. clip-path: inset(0px -10px -10px -10px);
  129. }
  130. .search-content {
  131. }
  132. }
  133. </style>