element-ui 二次封装搜索组件

如何使用此插件


import Search from "@/components/search";
// resourceSearchKey   resourceSearchId 都是权限 若权限编码格式不同自行修改
// formTypeData 需要的控件集合,type是 类型 name 是lable  key 是返回数据的key 常和接口查询字段相同

 <Search
   class="search-area"
   :formTypeData="formTypeData"
   :resourceSearchKey="'.5.1'"
   :resourceSearchId="3"
   @handleSearch="handleSearch"
   @resetForm="resetSearchForm"
 ></Search>
传参
 formTypeData: [
        {
          name:'公司名称',
          placeholderText:'请输入要查询的公司名称',
          type:'input',
          key:'companyName'
        },
        {
          name:'合同编号',
          placeholderText:'请输入要查询的合同编号',
          type:'input',
          key:'contractCode'
        },
        {
          name:'合同名称',
          placeholderText:'请输入要查询的合同名称',
          type:'input',
          key:'contractName'
        },
          {
          name: "合同类型",
          placeholderText: "请选择合同类型",
          type: "select",
          data: [
            { name: "全部", id: "" },
            { name: "执行中", id: 10 },
            { name: "已结束", id: 20 }
          ],
          key: "acContractStatus"
        },
        {
          name: "合同状态",
          placeholderText: "请选择合同状态",
          type: "select",
          data: [
            { name: "全部", id: "" },
            { name: "执行中", id: 10 },
            { name: "已结束", id: 20 }
          ],
          key: "acContractStatus"
        },
         {
          name:'制单人',
          placeholderText:'制单人',
          type:'input',
          key:'maker'
        },
      ],

一行最多三个,多出有收起
展开
element-ui 二次封装搜索组件_第1张图片

<!--
   Description:条件搜索组件
 -->
<template>
  <div class="search-area container" id="myDIV">
    <!--条件选择区域-->
    <div :style="{'width':getContentWidth()}">
      <el-form
        class="search-area-left"
        :model="form"
        label-width="100px"
        label-position="right"
        @submit.native.prevent
      >
        <div class="search-item-parent" v-for="(item,index) of formTypeData" :key="index">
          <transition name="condition">
            <div class="search-item-child" v-show="item.isVisibility">
              <el-form-item :label="item.name" v-if="item.custom_widget">
                <!--自定义控件插槽-->
                <slot name="custom_widget" :item="item"></slot>
              </el-form-item>
              <el-form-item :label="item.name" v-else-if="item.type === 'date'">
                <el-date-picker
                  v-model="form.value[index].modelVal"
                  :type="item.dateType"
                  :value-format="item.dateValueFormat"
                  :placeholder="item.placeholderText"
                ></el-date-picker>
              </el-form-item>
              <el-form-item :label="item.name" v-else-if="item.type === 'range'">
                <el-date-picker
                  v-model="form.value[index].modelVal"
                  :type="item.dateType"
                  :value-format="item.dateValueFormat"
                  :placeholder="item.placeholderText"
                  :default-time="item.defaultTime"
                  range-separator="至"
                  :start-placeholder="item.startPlaceholder"
                  :end-placeholder="item.endPlaceholder"
                ></el-date-picker>
              </el-form-item>
              <el-form-item :label="item.name" v-else-if="item.type === 'select'">
                <el-select v-model="form.value[index].modelVal" :placeholder="item.placeholderText">
                  <el-option
                    v-for="childItem in item.data"
                    :key="childItem.id"
                    :label="childItem.name"
                    :value="childItem.id"
                  ></el-option>
                </el-select>
              </el-form-item>
              <el-form-item :label="item.name" v-else-if="item.type === 'input'">
                <el-input
                  v-model.trim="form.value[index].modelVal"
                  :placeholder="item.placeholderText"
                  auto-complete="off"
                  @keyup.enter.native.stop="handleSearch()"
                ></el-input>
              </el-form-item>
            </div>
          </transition>
        </div>
      </el-form>
    </div>
    <!--按钮操作区域-->
    <div class="button-area">
      <el-button
        type="primary"
        v-if="vertifyOperateCode(resourceSearchKey,resourceSearchId)"
        @click="handleSearch()"
      >查询</el-button>
      <el-button
        @click="resetSearchForm()"
        v-if="vertifyOperateCode(resourceSearchKey,resourceSearchId)"
        style="margin-left: 10px"
      >清空</el-button>
      <span class="search-arrow" v-if="formTypeData.length>SINGLE_LINE_NUM" @click="openOrClose()">
        <img :src="imageSrc" :class="[isOpen?'arrow-open':'arrow-close','arrow-normal']" />
      </span>
    </div>
  </div>
</template>
<script>
export default {
  name: "searchView",
  props: {
    /* 源数据*/
    formTypeData: {
      type: Array,
      default: function() {
        return [];
      }
    },
    resourceSearchKey: {
      type: String,
      default: ""
    },
    resourceSearchId: {
      type: Number,
      default: null
    },

  },

  data: function() {
    return {
      // 搜索条件表单
      form: {
        // v-model绑定的数据
        value: [],
        // 条目组件是否可见
        visibility: []
      },
      // 是否展示更多搜索
      isOpen: false,
      // 上一次的浏览器宽度 默认1920
      lastBrowserWidth: 1920,
      // 控件适配浏览器左右放大缩小的分界限
      BOUND_WIDTH: 1200,
      // 每一个搜索条件控件的高度
      SINGLE_ITEM_HEIGHT: 60,
      // 单行展示的个数
      SINGLE_LINE_NUM: 3,
      // 上下内部间距各为10px
      VERTICAL_PADDING: 10,
      // 容器高度过渡时间
      HEIGHT_TRANSITION_TIME: 0.3,
      imageSrc: "../../../static/images/arrow-down.png"
    };
  },
  created: function() {
    /* 初始化数据,增加条件默认值*/
    this.formTypeData.forEach((item, index) => {
      this.form.value.push({
        key: item.key,
        modelVal: ""
      });
      if (index < this.SINGLE_LINE_NUM) {
        item.isVisibility = true;
      } else {
        item.isVisibility = false;
      }
    });
  },
  mounted: function() {
    this.$nextTick(() => {
      // 根据浏览器宽度 初始化样式
      let browserWidth = window.innerWidth;
      this.calculateParentHeight(browserWidth);

      /* 监听浏览器宽度变化*/
      window.addEventListener("resize", this.resizeBrowser, false);
    });
  },

  methods: {
    // 处理浏览器变化
    resizeBrowser: function(argument) {
      let browserWidth = argument.target.innerWidth;
      // 浏览器由宽变窄并越过分界线
      if (
        browserWidth <= this.BOUND_WIDTH &&
        this.lastBrowserWidth > this.BOUND_WIDTH
      ) {
        this.lastBrowserWidth = browserWidth;
        this.calculateParentHeight(browserWidth);
      }
      // 浏览器由窄变宽并越过分界线
      if (
        this.lastBrowserWidth <= this.BOUND_WIDTH &&
        browserWidth > this.BOUND_WIDTH
      ) {
        this.lastBrowserWidth = browserWidth;
        this.calculateParentHeight(browserWidth);
      }
    },

    // 根据数据内容长度调整条件区域的宽度
    getContentWidth: function() {
      if (this.formTypeData.length >= this.SINGLE_LINE_NUM) {
        return "80%";
      } else {
        return this.formTypeData.length * 29 + "%";
      }
    },
    // 打开或关闭更多搜索区域
    openOrClose: function() {
      if (this.isOpen) {
        this.formTypeData.forEach((item, index) => {
          if (index < this.SINGLE_LINE_NUM) {
            item.isVisibility = true;
          } else {
            item.isVisibility = false;
          }
        });
      } else {
        this.formTypeData.forEach((item, index) => {
          item.isVisibility = true;
        });
      }
      this.isOpen = !this.isOpen;

      /* 设置控件样式*/
      let browserWidth = window.innerWidth;
      this.calculateParentHeight(browserWidth);
    },

    /**
     * 根据当前浏览器宽度计算外层应有的高度
     * @param browserWidth 当前浏览器的宽度
     */
    calculateParentHeight: function(browserWidth) {
      let array = document.getElementsByClassName("search-item-parent");
      let elParent = document.getElementsByClassName("container")[0];
      if (browserWidth > this.BOUND_WIDTH) {
        // 当视口为较宽状态
        elParent.style.transition =
          "height " + this.HEIGHT_TRANSITION_TIME + "s";
        if (this.isOpen) {
          let size = this.formTypeData.length;
          // 获取当前展开的所有条目所占的行数  容器高度=单行高度*行数+上下内间距
          let lineNum =
            size % this.SINGLE_LINE_NUM > 0
              ? parseInt(size / this.SINGLE_LINE_NUM + 1)
              : parseInt(size / this.SINGLE_LINE_NUM);
          elParent.style.height =
            this.SINGLE_ITEM_HEIGHT * lineNum +
            this.VERTICAL_PADDING * 2 +
            "px";
        } else {
          elParent.style.height =
            this.SINGLE_ITEM_HEIGHT + this.VERTICAL_PADDING * 2 + "px";
        }
        // 设置单个搜索条件的宽度
        for (let i = 0; i < array.length; i++) {
          let item = array[i];
          if (this.formTypeData.length >= this.SINGLE_LINE_NUM) {
            item.style.width = 100 / this.SINGLE_LINE_NUM + "%";
          } else {
            item.style.width = 100 / this.formTypeData.length + "%";
          }
        }
      } else {
        // 当视口为较窄状态
        elParent.style.transition = "initial";
        if (this.isOpen) {
          let size = this.formTypeData.length;
          elParent.style.height =
            this.SINGLE_ITEM_HEIGHT * size + this.VERTICAL_PADDING * 2 + "px";
        } else {
          // 容器高度=单行高度*行数+上下内间距
          let lineNum =
            this.formTypeData.length >= this.SINGLE_LINE_NUM
              ? this.SINGLE_LINE_NUM
              : this.formTypeData.length % this.SINGLE_LINE_NUM;
          elParent.style.height =
            this.SINGLE_ITEM_HEIGHT * lineNum +
            this.VERTICAL_PADDING * 2 +
            "px";
        }
        // 设置单个搜索条件的宽度
        for (let i = 0; i < array.length; i++) {
          let item = array[i];
          item.style.width = "100%";
        }
      }
    },
    // 生成接口对应的数据结构进行搜索
    handleSearch: function() {
      let formValue = {};
      if (this.formTypeData) {
        this.formTypeData.forEach((item, index) => {
          let key = item.key;
          let value = this.form.value[index].modelVal;
          if (value === 0 || value) {
            formValue[key] = value;
          }
        });
      }
      this.$emit("handleSearch", 1, formValue, true);
    },
    // 重置搜索条件
    resetSearchForm: function() {
      this.form.value = [];
      this.formTypeData.forEach((item, index) => {
        this.form.value.push({
          key: item.key,
          modelVal: ""
        });
      });
      this.$emit("resetForm");
      this.$emit("handleSearch", 1, {});
    }
  },
  beforeDestroy: function() {
    window.removeEventListener("resize", this.resizeBrowser);
  }
};
</script>
<style lang="scss" scoped>
/*列表页搜索区域*/

.search-area {
    box-sizing: border-box;
    -webkit-box-sizing: border-box;
    padding: 10px 25px 10px 20px;
    height: auto;
    background-color: rgb(252, 252, 252);
    .el-form-item {
        float: left;
        margin-bottom: 0;
        padding-bottom: 10px;
        padding-top: 10px;
        width: 100%;
    }
    &-button {
        .el-form-item__content {
            margin-left: 0px !important;
        }
    }
}

/*最外层包裹上下过渡*/
.container {
  display: flex;
  height: 80px;
  transition-property: height;
  transition-duration: 0.3s;
}
/*条件条目的显示隐藏过渡*/
.condition-enter,
.condition-leave-to {
  opacity: 0;
}
.condition-enter-active {
  transition-property: opacity;
  transition-duration: 1.2s;
}
.condition-leave-active {
  transition-property: opacity;
  transition-duration: 0.3s;
}
/*按钮操作区域*/
.button-area {
  display: flex;
  align-items: center;
  height: 60px;
  margin-left: 30px;
  /*展开关闭箭头*/
  .search-arrow {
    border: 1px solid #d9d9d9;
    border-radius: 4px;
    background: #ffffff;
    margin-left: 10px;
    width: 28px;
    height: 28px;
    display: flex;
    align-items: center;
    justify-content: center;
    cursor: pointer;
  }
  /*箭头正常状态*/
  .arrow-normal {
    width: 18px;
    height: 14px;
  }
  /*展开箭头过渡*/
  .arrow-open {
    transform: rotate(-180deg);
    transition: transform 0.3s;
  }
  /*关闭箭头过渡*/
  .arrow-close {
    transform: rotate(0deg);
    transition: transform 0.3s;
  }
}
/*左侧搜索条件区域*/
.search-area-left {
  display: flex;
  flex-wrap: wrap;
  flex-direction: row;
  width: 100%;
  box-sizing: border-box;
  .search-item-parent {
    overflow: hidden;
  }
  .search-item-parent >>> .el-input__inner {
    // height: 40px !important;
  }
  .search-item-range >>> .el-input__inner {
    // height: 40px !important;
  }
  .search-item-range {
    // width: 66%;
    width: 33.3%;
  }
  .search-item-child {
    width: 100%;
    box-sizing: border-box;
    // padding-right: 45px;
    >>> .el-form-item__label {
      color: #4a4a4a;
      letter-spacing: 0;
    }
  }
}
.inquireDiv .el-form-item {
}
</style>

你可能感兴趣的:(element-ui 二次封装搜索组件)