♻️(component): 优化datepicker 弹出后内部元素的定位

This commit is contained in:
0o张不歪o0 2022-07-14 13:28:25 +08:00
parent d697f0b100
commit 3d5b738104
2 changed files with 76 additions and 16 deletions

View File

@ -12,24 +12,36 @@ const getYears = () => {
/** /**
* *
*/ */
const getDate = () => { const getDate = (val='') => {
return new Date(); if(val){
return new Date(val);
}else{
return new Date();
}
}; };
/** /**
* *
*/ */
const getYear = () => { const getYear = (val='') => {
return getDate().getFullYear(); return getDate(val).getFullYear();
}; };
/** /**
* *
*/ */
const getMonth = () => { const getMonth = (val='') => {
return getDate().getMonth(); return getDate(val).getMonth();
}; };
const getDay=(val='')=>{
if(val){
return new Date(getDate(val).toDateString()).getTime();
}else{
return -1;
}
}
/** /**
* *
* *
@ -40,4 +52,4 @@ const getDayLength = (year: number, month: number): number => {
return new Date(year, month + 1, 0).getDate(); return new Date(year, month + 1, 0).getDate();
}; };
export { getDayLength, getYears, getDate, getMonth, getYear }; export { getDayLength, getYears, getDate, getMonth, getYear,getDay };

View File

@ -1,12 +1,13 @@
<template> <template>
<div> <div>
<lay-dropdown ref="dropdownRef" :disabled="props.disabled"> <lay-dropdown ref="dropdownRef" :disabled="disabled" @open="openDropDown">
<lay-input <lay-input
readonly readonly
:name="name" :name="name"
:model-value="dateValue || modelValue" :model-value="dateValue || modelValue"
:placeholder="placeholder" :placeholder="placeholder"
prefix-icon="layui-icon-date" prefix-icon="layui-icon-date"
:disabled="disabled"
> >
</lay-input> </lay-input>
<template #content> <template #content>
@ -83,7 +84,7 @@
<div class="layui-laydate-footer"> <div class="layui-laydate-footer">
<span <span
v-if="type === 'datetime'" v-if="type === 'datetime'"
@click="showPane = 'time'" @click="showPane = 'time';scrollToCurrentTime()"
class="laydate-btns-time" class="laydate-btns-time"
>选择时间</span >选择时间</span
> >
@ -114,6 +115,7 @@
<div <div
class="layui-laydate-content" class="layui-laydate-content"
style="height: 220px; overflow-y: auto" style="height: 220px; overflow-y: auto"
ref="yearmonthScrollRef"
> >
<ul class="layui-laydate-list laydate-year-list"> <ul class="layui-laydate-list laydate-year-list">
<li <li
@ -216,8 +218,8 @@
</div> </div>
</div> </div>
<div class="layui-laydate-content" style="height: 210px"> <div class="layui-laydate-content" style="height: 210px">
<ul class="layui-laydate-list laydate-time-list"> <ul class="layui-laydate-list laydate-time-list" ref="timePanelRef">
<li class="num-list" v-for="item in els" :key="item.type"> <li class="num-list" v-for="item in els" :key="item.type" :data-type="item.type">
<ol class="scroll" @click="choseTime"> <ol class="scroll" @click="choseTime">
<li <li
v-for="(it, index) in item.count" v-for="(it, index) in item.count"
@ -276,8 +278,9 @@ import dayjs from "dayjs";
import { LayIcon } from "@layui/icons-vue"; import { LayIcon } from "@layui/icons-vue";
import LayInput from "../input/index.vue"; import LayInput from "../input/index.vue";
import LayDropdown from "../dropdown/index.vue"; import LayDropdown from "../dropdown/index.vue";
import { getDayLength, getYears, getMonth, getYear } from "./day"; import { getDayLength, getYears, getMonth, getYear,getDay } from "./day";
import { ref, watch, computed, defineProps, defineEmits, onMounted } from "vue"; import { ref, watch, computed, defineProps, defineEmits, onMounted, nextTick } from "vue";
import { anyTypeAnnotation } from "@babel/types";
export interface LayDatePickerProps { export interface LayDatePickerProps {
type?: "date" | "datetime" | "year" | "time" | "month" | "yearmonth"; type?: "date" | "datetime" | "year" | "time" | "month" | "yearmonth";
@ -328,13 +331,16 @@ const els = [
{ count: 60, type: "ss" }, { count: 60, type: "ss" },
]; ];
const currentYear = ref(getYear()); const currentYear = ref(getYear(props.modelValue));
const currentMonth = ref(getMonth()); const currentMonth = ref(getMonth(props.modelValue));
const currentDay = ref<number>(props.modelValue? new Date(props.modelValue).getTime(): -1); const currentDay = ref<number>(getDay(props.modelValue));
const yearList = ref<number[]>(getYears()); const yearList = ref<number[]>(getYears());
const dateList = ref<any[]>([]); const dateList = ref<any[]>([]);
const showPane = ref("date"); const showPane = ref("date");
const yearmonthScrollRef=ref()
watch( watch(
() => props.type, () => props.type,
() => { () => {
@ -555,4 +561,46 @@ const choseTime = (e: any) => {
hms.value[type as keyof typeof hms.value] = value; hms.value[type as keyof typeof hms.value] = value;
} }
}; };
const openDropDown=()=>{
nextTick(()=>{
if(showPane.value=='year'||showPane.value=='yearmonth'){
let scrollTop=0;
for (const child of yearmonthScrollRef.value.firstElementChild.childNodes) {
if(child.classList&&child.classList.contains('layui-this')){
scrollTop=child.offsetTop-(yearmonthScrollRef.value.offsetHeight-child.offsetHeight)/2;
break;
}
}
yearmonthScrollRef.value.scrollTo(0,scrollTop)
}else if(showPane.value=='time'){
scrollToCurrentTime()
}
})
}
//
const timePanelRef=ref()
const scrollToCurrentTime = () => {
nextTick(() => {
timePanelRef.value.childNodes.forEach((element: HTMLElement) => {
if (element.nodeName === 'LI') {
let scrollTop = 0;
let parentDom = element.firstElementChild as HTMLElement;
let childList = parentDom.childNodes;
for (let index = 0; index < childList.length; index++) {
const child = childList[index] as HTMLElement;
if (child.nodeName !== 'LI') {
continue;
}
if (child.classList && child.classList.contains('layui-this')) {
scrollTop = child.offsetTop - (parentDom.offsetHeight - child.offsetHeight) / 2;
parentDom.scrollTo(0, scrollTop)
break;
}
}
}
});
})
}
</script> </script>