185 lines
4.3 KiB
Plaintext
185 lines
4.3 KiB
Plaintext
<script lang="ts">
|
|
export default {
|
|
name: "StandardRange",
|
|
};
|
|
</script>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, toRef, Ref } from "vue";
|
|
import { on, off } from "evtd";
|
|
import LayTooltip from "../tooltip/index.vue";
|
|
import { throttle, makeDots } from "./utils/index";
|
|
|
|
export interface StandardRangeProps {
|
|
rangeValue: Array<number>;
|
|
disabled?: boolean;
|
|
step?: number;
|
|
min?: number;
|
|
max?: number;
|
|
showDots?: boolean;
|
|
}
|
|
|
|
const props = withDefaults(defineProps<StandardRangeProps>(), {
|
|
step: 0,
|
|
min: 0,
|
|
max: 100,
|
|
disabled: false,
|
|
showDots: true,
|
|
});
|
|
|
|
let rv = toRef(props, "rangeValue");
|
|
|
|
const moveAction = throttle(rangeMove);
|
|
let currbtn = -1;
|
|
|
|
function handle_mousedown() {
|
|
currbtn = -1;
|
|
tooptipHide.value = false;
|
|
on("selectstart", window, handle_select, { once: true });
|
|
on("mouseup", window, handle_mouseup);
|
|
on("mousemove", window, moveAction);
|
|
}
|
|
|
|
function handle_mouseup() {
|
|
tooptipHide.value = true;
|
|
off("selectstart", document, handle_select);
|
|
off("mouseup", window, handle_mouseup);
|
|
off("mousemove", window, moveAction);
|
|
}
|
|
function handle_select(e: Event): void {
|
|
e.preventDefault();
|
|
}
|
|
|
|
const tracker = ref<HTMLElement | null>(null);
|
|
const emit = defineEmits(["link-val-hook"]);
|
|
|
|
const tooptipHide = ref<boolean>(true);
|
|
|
|
function rangeMove(e: MouseEvent) {
|
|
// tooptipHide.value = false;
|
|
if (!tracker.value) {
|
|
return;
|
|
}
|
|
let tracker_rect = tracker.value.getBoundingClientRect();
|
|
let origin_left = tracker_rect.left;
|
|
let point_left = e.clientX;
|
|
let distance = point_left - origin_left;
|
|
if (distance < props.min) {
|
|
rv.value[0] = props.min;
|
|
} else {
|
|
let rate = (distance / tracker_rect.width) * 100;
|
|
let idx = -1;
|
|
if (currbtn === -1) {
|
|
currbtn = moveNeighbors(Math.floor(rate), rv);
|
|
idx = currbtn;
|
|
} else {
|
|
idx = currbtn;
|
|
}
|
|
|
|
calcWithStep(rate, rv, idx);
|
|
if (rv.value[1] > props.max) {
|
|
rv.value[1] = props.max;
|
|
}
|
|
if (rv.value[0] < props.min) {
|
|
rv.value[0] = props.min;
|
|
}
|
|
}
|
|
emit("link-val-hook", rv.value);
|
|
}
|
|
|
|
function moveNeighbors(rate: number, rangeValues: any) {
|
|
let d1 = Math.abs(rate - rangeValues.value[0]);
|
|
let d2 = Math.abs(rate - rangeValues.value[1]);
|
|
if (d1 > d2) {
|
|
return 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
function calcWithStep(
|
|
rate: number | undefined,
|
|
val: Ref<number> | Ref<number[]>,
|
|
idx: number = -1
|
|
) {
|
|
if (typeof rate === "undefined") return false;
|
|
|
|
if (typeof val.value === "object") {
|
|
let r = rate - val.value[idx];
|
|
if (Math.abs(r) < props.step) {
|
|
return false;
|
|
}
|
|
|
|
if (props.step === 0) val.value[idx] = Math.floor(rate);
|
|
|
|
if (Array.isArray(val.value)) {
|
|
if (r < 0 && props.step !== 0) {
|
|
val.value[idx] -= props.step;
|
|
} else {
|
|
val.value[idx] += props.step;
|
|
}
|
|
cross(val);
|
|
}
|
|
}
|
|
}
|
|
|
|
function cross(val: any) {
|
|
if (val.value[0] > val.value[1]) {
|
|
let tmp = val.value[0];
|
|
val.value[0] = val.value[1];
|
|
val.value[1] = tmp;
|
|
currbtn = currbtn === 0 ? 1 : 0;
|
|
}
|
|
}
|
|
// 断点
|
|
const dots = makeDots(props);
|
|
|
|
const focusDot = (item: number) => {
|
|
let currbtn = moveNeighbors(item, rv);
|
|
rv.value[currbtn] = item;
|
|
emit("link-val-hook", rv.value);
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div
|
|
ref="tracker"
|
|
@mousedown.stop="handle_mousedown"
|
|
class="layui-slider-srange"
|
|
:class="[disabled ? 'layui-slider-disabled' : '']"
|
|
>
|
|
<lay-tooltip :content="'' + rv[0]" :is-can-hide="tooptipHide">
|
|
<div
|
|
:style="{ left: rv[0] + '%' }"
|
|
class="layui-slider-btn-v"
|
|
:class="[props.disabled ? 'layui-slider-disabled disable-btn' : '']"
|
|
></div>
|
|
</lay-tooltip>
|
|
|
|
<lay-tooltip :content="'' + rv[1]" :is-can-hide="tooptipHide">
|
|
<div
|
|
:style="{ left: rv[1] + '%' }"
|
|
class="layui-slider-btn-v"
|
|
:class="[props.disabled ? 'layui-slider-disabled disable-btn' : '']"
|
|
></div>
|
|
</lay-tooltip>
|
|
<div class="layui-slider-line-v"></div>
|
|
<div
|
|
:style="{
|
|
width: rv[1] - rv[0] + '%',
|
|
left: rv[0] + '%',
|
|
}"
|
|
class="layui-slider-rate-v"
|
|
:class="[props.disabled ? 'layui-slider-disabled disable-line' : '']"
|
|
></div>
|
|
<div
|
|
v-show="showDots"
|
|
@click="focusDot(item)"
|
|
class="layui-slider-dots"
|
|
v-for="(item, index) in dots"
|
|
:key="index"
|
|
:style="{ left: item + '%' }"
|
|
></div>
|
|
</div>
|
|
</template>
|