实现el-select的远程搜索,并且展示最近搜索的十条数据

项目场景:

使用el-select实现远程搜索,并且可以输入进行搜索,首次点击的时候显示十条历史记录,在输入之后就展示远程搜索返回的数据


问题描述

1、el-select的远程搜索如果实现

2、怎么记录历史搜索的数据

3、如果实现输入之后点击回车就进入到结果页面


问题分析:

远程搜索的实现

远程搜索的实现,el-select组件中有远程搜索的API

 <el-select
          popper-class="social-economy-search-input"
          v-model="keywords"
          placeholder="请输入关键词"
          filterable
          remote
          :remote-method="remoteMethod"
          :loading="loadingOption"
          @click="getHistorySearch"
          @change="handleChange"
          @keyup.enter="keyupEnter"
        >

remote:其中的选项是否从服务器远程加载

remote-method:自定义远程搜索方法

loading:是否正在从远程获取数据

利用remote-method这个API自定义远程搜索的方法,当进行远程搜索的时候就开启loading,有数据结束之后就关闭loading

这里我为什么要定义这三个方法呢?

@click="getHistorySearch"  ,点击select选择器的时候,获取历史记录
@change="handleChange",当select选择器的数据发生变化的时候触发
@keyup.enter="keyupEnter",当回车的时候触发

//远程搜索
const remoteMethod = async (val) => {
  if (val) {
    loadingOption.value = true;
    keywords.value = val;
    let params = {
      keywords: keywords.value,
      pageSize: pageSize.value,
    };
    let res = await httpPost(
      "economicStatistics/v1/searchIndicatorListByKeywords",
      params
    );
    if (res.code == 0) {
      options.value.label = "";
      options.value.list = res.data;
      loadingOption.value = false;
    } else {
      loadingOption.value = false;
    }
  } else {
    keywords.value = "";
    loadingOption.value = false;
  }
};

 历史记录的实现

要实现历史记录就要记录用户输入后请求的每一次数据,将用户的输入存储localStorage中,在用户点击select的时候,展示最近十条历史记录

注意:获取最近十条历史记录,当用户有输入与之前记录相同的关键字,就删除掉之前的记录只展示最新的相同的记录

//历史记录列表
const historySearchKeyWordsList = ref([]);
//删除历史搜索中与当前输入相同的关键字
const removeDuplicate = () => {
  historySearchKeyWordsList.value =
    JSON.parse(localStorage.getItem("keywords")) || [];
  const index = historySearchKeyWordsList.value.indexOf(keywords.value); // 查找新元素在数组中的索引
  if (index !== -1) {
    historySearchKeyWordsList.value.splice(index, 1); // 删除数组中与新元素相同的旧元素
  }
  return historySearchKeyWordsList.value;
};
//存储历史搜索关键字
const saveKeyWords = () => {
  if (historySearchKeyWordsList.value.length >= 10) {
    historySearchKeyWordsList.value.pop(); // 删除最早存储的关键字
  }
  historySearchKeyWordsList.value.unshift(keywords.value);
  localStorage.setItem(
    "keywords",
    JSON.stringify(historySearchKeyWordsList.value)
  );
};
//获取历史搜索
const getHistorySearch = () => {
  const keywords = JSON.parse(localStorage.getItem("keywords")) || [];
  options.value.label = "历史搜索";
  options.value.list = keywords.map((item) => {
    return { name: item };
  });
};

解决方案:

<template>
  <div class="database-index">
    <div class="social-economy">
      <div class="social-economy-logo">
        <img
          src="@/assets/imgs/database/social-economy/socialEconomy-logo.png"
          alt=""
        />
      </div>
      <div class="social-economy-search">
        <el-select
          popper-class="social-economy-search-input"
          v-model="keywords"
          placeholder="请输入关键词"
          filterable
          remote
          :remote-method="remoteMethod"
          :loading="loadingOption"
          @click="getHistorySearch"
          @change="handleChange"
          @keyup.enter="keyupEnter"
        >
          <template #prefix>
            <el-icon class="el-input__icon"><search /></el-icon>
          </template>
          <el-option-group :key="options.label" :label="options.label">
            <el-option
              v-for="item in options.list"
              :key="item.name"
              :label="item.name"
              :value="item.name"
            >
              <div
                class="option-list"
                @click="getCurrentId(item.indicatorUniqueId)"
              >
                <el-icon class="el-input__icon">
                  <search />
                </el-icon>
                <span
                  style="padding-left: 8px"
                  v-html="highlightKeywords(item.name, keywords)"
                ></span>
              </div>
            </el-option>
          </el-option-group>
        </el-select>
      </div>
      <div class="social-economy-list"></div>
    </div>
  </div>
</template>

<script setup>
import { ref } from "vue";
import { httpPost } from "@/api/httpService.js";

import { useRoute, useRouter } from "vue-router";
import { highlightKeywords } from "@/utils/highlightKeywords.js";
import { Debounce } from "@/utils/utils";
import { ElMessage } from "element-plus";

const route = useRoute();
const router = useRouter();
const loadingOption = ref(false); //是否正在加载选项数据
const options = ref({ label: "", list: [] });
const keywords = ref("");
const pageNum = ref(1);
const pageSize = ref(10);
const currentId = ref("");
//获取远程搜索点击时当前指标id
const getCurrentId = (id) => {
  currentId.value = id;
};
//远程搜索
const remoteMethod = async (val) => {
  if (val) {
    loadingOption.value = true;
    keywords.value = val;
    let params = {
      keywords: keywords.value,
      pageSize: pageSize.value,
    };
    let res = await httpPost(
      "economicStatistics/v1/searchIndicatorListByKeywords",
      params
    );
    if (res.code == 0) {
      options.value.label = "";
      options.value.list = res.data;
      loadingOption.value = false;
    } else {
      loadingOption.value = false;
    }
  } else {
    keywords.value = "";
    loadingOption.value = false;
  }
};
//前往搜索结果页
const goSearchResult = () => {
  if (keywords.value && keywords.value !== "") {
    router.push({
      path: "/database/social-economy/search-result",
      query: {
        keywords: keywords.value,
      },
    });
    removeDuplicate();
    saveKeyWords();
  } else {
    ElMessage.warning("请输入搜索关键字!");
  }
};
const handleChange = async (value) => {
  keywords.value = value != null && value != "" ? value : "";
  if (options.value.label) {
    goSearchResult();
  } else {
    await saveSearchKeyWords();
    router.push({
      path: "/database/social-economy/indicator-result",
      query: {
        type: 3,
        id: currentId.value,
      },
    });
  }
};
const keyupEnter = () => {
  goSearchResult();
};
//历史记录列表
const historySearchKeyWordsList = ref([]);
//删除历史搜索中与当前输入相同的关键字
const removeDuplicate = () => {
  historySearchKeyWordsList.value =
    JSON.parse(localStorage.getItem("keywords")) || [];
  const index = historySearchKeyWordsList.value.indexOf(keywords.value); // 查找新元素在数组中的索引
  if (index !== -1) {
    historySearchKeyWordsList.value.splice(index, 1); // 删除数组中与新元素相同的旧元素
  }
  return historySearchKeyWordsList.value;
};
//存储历史搜索关键字
const saveKeyWords = () => {
  if (historySearchKeyWordsList.value.length >= 10) {
    historySearchKeyWordsList.value.pop(); // 删除最早存储的关键字
  }
  historySearchKeyWordsList.value.unshift(keywords.value);
  localStorage.setItem(
    "keywords",
    JSON.stringify(historySearchKeyWordsList.value)
  );
};
//获取历史搜索
const getHistorySearch = () => {
  const keywords = JSON.parse(localStorage.getItem("keywords")) || [];
  options.value.label = "历史搜索";
  options.value.list = keywords.map((item) => {
    return { name: item };
  });
};
//保存远程搜索关键字
const saveSearchKeyWords = async () => {
  let params = {
    keywords: keywords.value,
  };
  httpPost("economicStatistics/v1/saveSearchKeywords", params);
};
</script>

  • 11
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
form generator是一个表单生成器,可以帮助我们快速生成各种表单元素。而el-selectelement-ui框架中的一个下拉选择组件。要扩展el-select远程搜索功能,我们可以按照以下步骤进行。 首先,在form generator的代码中找到生成el-select组件的部分。一般来说,我们可以为el-select设置一个远程搜索的属性,比如remote-search。然后,在该属性的处理逻辑中,我们可以使用element-ui提供的el-autocomplete组件,实现远程搜索的功能。 其次,为了实现远程搜索,我们需要在el-select远程搜索事件中发送Ajax请求。可以使用Vue.js提供的axios库或者使用element-ui的el-request库来发送请求。在请求中,我们需要传递远程搜索的关键字,一般是通过输入框实时获取用户输入的值。将这个关键字发送到后台,进行搜索操作。搜索的结果可以是JSON格式的数据,包含了符合条件的选项。 最后,当接收到后台返回的搜索结果后,我们可以根据结果来更新el-select的下拉选项列表。可以通过修改el-select的options属性来实现。将搜索结果的数据添加到options列表中,然后重新渲染el-select,用户就可以看到符合搜索条件的选项。 综上所述,要扩展form generator中的el-select远程搜索功能,我们需要为el-select添加一个remote-search属性,处理该属性对应的远程搜索逻辑,并在搜索结果返回后更新el-select的下拉选项列表。这样,用户在使用表单生成器时可以方便地使用el-select远程搜索功能,提高了表单的灵活性和用户体验。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值