tiefen_space_app/components/MySlider/index.jsx

343 lines
9.9 KiB
React
Raw Normal View History

2024-08-30 13:55:31 +08:00
import { View, Text, PanResponder, Dimensions } from "react-native";
2024-08-16 23:55:29 +08:00
import React, { useState, useEffect, useRef } from "react";
import { useTailwind } from "tailwind-rn";
export default function MySlider({
2024-08-21 17:54:04 +08:00
lower_bound,
upper_bound,
2024-08-16 23:55:29 +08:00
step,
2024-08-21 17:54:04 +08:00
unit,
2024-08-30 13:55:31 +08:00
unitSite = "right",
2024-08-23 19:03:05 +08:00
width = Dimensions.get("window").width - 44,
2024-08-21 17:54:04 +08:00
onChange,
2024-08-23 21:38:23 +08:00
handleFunc = () => {},
2024-08-16 23:55:29 +08:00
disabled = false,
2024-08-21 17:54:04 +08:00
thumbSize = 25,
2024-08-23 14:06:27 +08:00
stepValues = [],
hasInfinity,
2024-09-04 12:23:56 +08:00
leftValue = lower_bound,
rightValue = stepValues[stepValues.length - 1],
2024-08-16 23:55:29 +08:00
maximumTrackTintColor = "#dcdbdb",
minimumTrackTintColor = "#577BFF",
processHeight = 7,
}) {
2024-08-21 17:54:04 +08:00
// const leftProcess = useRef(0);
// const rightProcess = useRef(0);
const [leftProcess, setLeftProcess] = useState(0);
const [rightProcess, setRightProcess] = useState(0);
2024-08-23 14:06:27 +08:00
const [stepNum, setStepNum] = useState(stepValues.length - 1);
2024-08-16 23:55:29 +08:00
const [processWidth, setProcessWidth] = useState(330);
const tailwind = useTailwind();
const leftWatcher = useRef(null);
const rightWatcher = useRef(null);
2024-08-23 19:03:05 +08:00
const currentLeftValue = useRef(leftValue);
const currentRightValue = useRef(rightValue);
2024-08-16 23:55:29 +08:00
useEffect(() => {
leftWatcher.current = PanResponder.create({
// 建立监视器
onStartShouldSetPanResponder: () => true,
2024-08-23 21:38:23 +08:00
onPanResponderGrant: _onPanResponderGrantLeft, // 按下
2024-08-16 23:55:29 +08:00
onPanResponderMove: _onPanResponderMoveLeft, // 移动
2024-08-23 21:38:23 +08:00
onPanResponderEnd: _onPanResponderEndLeft, // 结束
2024-08-16 23:55:29 +08:00
});
rightWatcher.current = PanResponder.create({
// 建立监视器
onStartShouldSetPanResponder: () => true,
2024-08-23 21:38:23 +08:00
onPanResponderGrant: _onPanResponderGrantRight, // 按下
2024-08-16 23:55:29 +08:00
onPanResponderMove: _onPanResponderMoveRight, // 移动
2024-08-23 21:38:23 +08:00
onPanResponderEnd: _onPanResponderEndRight, // 结束
2024-08-16 23:55:29 +08:00
});
2024-08-21 17:54:04 +08:00
setProcessWidth(width - thumbSize * 2);
2024-08-16 23:55:29 +08:00
}, []);
2024-08-21 17:54:04 +08:00
useEffect(() => {
2024-08-23 14:06:27 +08:00
// if(itemKey)
2024-08-23 19:03:05 +08:00
const currentLeftProcess =
(stepValues.indexOf(leftValue) * step - lower_bound) /
(upper_bound - lower_bound);
2024-08-23 14:06:27 +08:00
const currentRightProcess =
(stepValues.indexOf(rightValue) * step - lower_bound) /
(upper_bound - lower_bound);
setLeftProcess(currentLeftProcess);
setRightProcess(currentRightProcess);
2024-08-23 19:03:05 +08:00
currentLeftValue.current = leftValue;
currentRightValue.current = rightValue;
2024-08-23 14:06:27 +08:00
}, [leftValue, rightValue]);
2024-08-16 23:55:29 +08:00
// 左侧滑块事件
2024-09-04 12:23:56 +08:00
const _onPanResponderGrantLeft = () => {
2024-08-23 21:38:23 +08:00
handleFunc && handleFunc();
};
2024-08-16 23:55:29 +08:00
2024-09-04 12:23:56 +08:00
const _onPanResponderEndLeft = () => {
2024-08-23 21:38:23 +08:00
handleFunc && handleFunc();
};
2024-08-16 23:55:29 +08:00
const _onPanResponderMoveLeft = (e, gestureState) => {
const process =
2024-08-23 21:38:23 +08:00
(gestureState.x0 - thumbSize * 2 + gestureState.dx) / processWidth;
2024-08-16 23:55:29 +08:00
_changeProcess(process, "left");
};
// 右侧滑块事件
2024-09-04 12:23:56 +08:00
const _onPanResponderGrantRight = () => {
2024-08-23 21:38:23 +08:00
handleFunc && handleFunc();
};
2024-09-04 12:23:56 +08:00
const _onPanResponderEndRight = () => {
2024-08-23 21:38:23 +08:00
handleFunc && handleFunc();
};
2024-08-16 23:55:29 +08:00
const _onPanResponderMoveRight = (e, gestureState) => {
const process =
2024-08-23 21:02:09 +08:00
(gestureState.x0 - thumbSize / 2 + gestureState.dx) / processWidth;
2024-08-16 23:55:29 +08:00
_changeProcess(process, "right");
};
const _changeProcess = (changeProcess, direction) => {
// 判断滑动开关
if (disabled) return;
2024-08-23 20:52:07 +08:00
// 按步长比例变化刻度
changeProcess =
changeProcess > 1 ? 1 : changeProcess < 0 ? 0 : changeProcess;
const v = changeProcess * (upper_bound - lower_bound);
const newValue = Math.round(v / step) * step;
const newProcess = Math.round(newValue) / (upper_bound - lower_bound);
setStepNum(stepNum + 1);
if (process !== newProcess) {
if (direction == "left") {
2024-08-23 21:02:09 +08:00
const currentRightProcess =
(stepValues.indexOf(currentRightValue.current) * step - lower_bound) /
(upper_bound - lower_bound);
2024-08-23 20:52:07 +08:00
if (newProcess < currentRightProcess) {
2024-08-23 21:38:23 +08:00
const newLower =
stepValues[
Math.round(
(newProcess * (upper_bound - lower_bound) + lower_bound) / step
)
];
if (newLower <= currentRightValue.current - 3) {
setLeftProcess(newProcess);
setStepNum((old) => old + 1);
onChange({
lower_bound: newLower,
upper_bound: currentRightValue.current,
});
}
2024-08-23 20:52:07 +08:00
}
} else {
const currentLeftProcess =
(stepValues.indexOf(currentLeftValue.current) * step - lower_bound) /
2024-08-23 19:03:05 +08:00
(upper_bound - lower_bound);
2024-08-23 20:52:07 +08:00
if (newProcess > currentLeftProcess) {
2024-08-23 21:38:23 +08:00
const newUpper =
stepValues[
Math.round(
(newProcess * (upper_bound - lower_bound) + lower_bound) / step
)
];
if (newUpper >= currentLeftValue.current + 3) {
setRightProcess(newProcess);
setStepNum((old) => old - 1);
onChange({
lower_bound: currentLeftValue.current,
upper_bound: newUpper,
});
}
2024-08-16 23:55:29 +08:00
}
}
}
};
return (
<View
style={{
2024-08-21 17:54:04 +08:00
height: "upper_bound-content",
2024-08-16 23:55:29 +08:00
width,
// flex: 1,
flexDirection: "row",
justifyContent: "center",
alignItems: "center",
2024-08-21 17:54:04 +08:00
marginBottom: 36,
2024-08-23 20:52:07 +08:00
marginTop: 24,
2024-08-16 23:55:29 +08:00
}}
>
<View style={{ width: "100%", position: "relative" }}>
2024-08-23 19:03:05 +08:00
{/* 格子步数 */}
{hasInfinity && (
<View
style={{
...tailwind("flex flex-row justify-between items-center"),
position: "absolute",
top: -23,
flex: 1,
width: "100%",
padding: 17,
paddingRight: 20,
marginLeft: -12,
}}
>
{stepValues.map((item, index) => (
<View
key={index}
style={{
backgroundColor:
item >= leftValue && item <= rightValue
? minimumTrackTintColor
: maximumTrackTintColor,
width: thumbSize / 2,
height: thumbSize / 2,
borderRadius: 50,
}}
></View>
))}
</View>
)}
2024-08-16 23:55:29 +08:00
<View
style={{
backgroundColor: maximumTrackTintColor,
height: processHeight,
width: processWidth,
position: "absolute",
2024-08-21 17:54:04 +08:00
marginTop: -processHeight / 2,
2024-08-16 23:55:29 +08:00
left: thumbSize / 2,
zIndex: 1,
2024-08-21 17:54:04 +08:00
borderRadius: 50,
2024-08-16 23:55:29 +08:00
}}
/>
<View
style={{
backgroundColor: minimumTrackTintColor,
2024-08-21 17:54:04 +08:00
width: (rightProcess - leftProcess) * processWidth,
2024-08-16 23:55:29 +08:00
height: processHeight,
2024-08-21 17:54:04 +08:00
marginLeft: thumbSize / 2 + leftProcess * processWidth,
2024-08-16 23:55:29 +08:00
position: "absolute",
2024-08-21 17:54:04 +08:00
marginTop: -processHeight / 2,
2024-08-16 23:55:29 +08:00
left: 0,
zIndex: 10,
2024-08-21 17:54:04 +08:00
borderRadius: 50,
2024-08-16 23:55:29 +08:00
}}
/>
</View>
2024-08-23 19:03:05 +08:00
{/* 左侧控件 */}
2024-08-16 23:55:29 +08:00
<View
transition={1000}
cachePolicy="disk"
style={{
position: "absolute",
2024-08-23 21:38:23 +08:00
left: leftProcess * processWidth,
2024-08-16 23:55:29 +08:00
zIndex: 10,
2024-08-23 21:38:23 +08:00
width: thumbSize,
height: thumbSize,
borderRadius: 50,
backgroundColor: "#ff75c8",
borderColor: "#ffffff",
borderWidth: 2,
2024-08-16 23:55:29 +08:00
}}
{...leftWatcher.current?.panHandlers}
2024-08-23 21:38:23 +08:00
></View>
2024-08-23 19:03:05 +08:00
{/* 右侧控件 */}
2024-08-16 23:55:29 +08:00
<View
transition={1000}
cachePolicy="disk"
style={{
position: "absolute",
2024-08-23 21:38:23 +08:00
left: rightProcess * processWidth,
2024-08-16 23:55:29 +08:00
zIndex: 10,
2024-08-23 21:38:23 +08:00
width: thumbSize,
height: thumbSize,
backgroundColor: "#fff",
borderRadius: 50,
backgroundColor: "#ff75c8",
borderColor: "#ffffff",
borderWidth: 2,
2024-08-16 23:55:29 +08:00
}}
2024-08-23 21:02:09 +08:00
{...rightWatcher.current?.panHandlers}
2024-08-23 21:38:23 +08:00
></View>
2024-08-23 19:03:05 +08:00
2024-08-21 17:54:04 +08:00
<View
style={{
...tailwind("flex flex-row justify-between items-center"),
position: "absolute",
2024-08-23 20:52:07 +08:00
top: -48,
right: 24,
2024-08-21 17:54:04 +08:00
}}
>
2024-08-30 13:55:31 +08:00
{unitSite == "left" && (
<Text
style={{
...tailwind("text-white text-lg"),
maxWidth: thumbSize * 2,
color: "#ff75c8",
}}
>
{unit}
</Text>
)}
2024-08-21 17:54:04 +08:00
<Text
style={{
...tailwind("text-white text-lg"),
color: "#ff75c8",
}}
>
2024-08-23 21:02:09 +08:00
{hasInfinity && leftProcess == 1 ? ">" + upper_bound : leftValue}
2024-08-21 17:54:04 +08:00
</Text>
<Text
style={{
...tailwind("text-lg"),
marginHorizontal: 4,
color: "#ff75c8",
}}
>
-
</Text>
<Text
style={{
...tailwind("text-white text-lg"),
color: "#ff75c8",
}}
>
2024-08-30 13:55:31 +08:00
{hasInfinity && rightProcess >= 1 ? ">" + upper_bound : rightValue}
2024-08-21 17:54:04 +08:00
</Text>
2024-08-30 13:55:31 +08:00
{unitSite == "right" && (
<Text
style={{
...tailwind("text-white text-lg"),
maxWidth: thumbSize * 2,
color: "#ff75c8",
}}
>
{unit}
</Text>
)}
2024-08-21 17:54:04 +08:00
</View>
2024-08-30 13:55:31 +08:00
<View
2024-08-21 17:54:04 +08:00
style={{
2024-08-30 13:55:31 +08:00
...tailwind(
"absolute text-white text-sm flex flex-row justify-between"
),
width: width - 22,
2024-08-21 17:54:04 +08:00
left: 0,
top: 20,
color: "#ffffff80",
}}
>
2024-08-30 13:55:31 +08:00
<Text
style={{
...tailwind(" text-white text-sm"),
minWidth: thumbSize,
maxWidth: thumbSize * 2,
color: "#ffffff80",
}}
>
{lower_bound}
</Text>
<Text
style={{
...tailwind(" text-white text-sm"),
minWidth: thumbSize,
color: "#ffffff80",
}}
>
{hasInfinity ? ">" + upper_bound : upper_bound}
</Text>
</View>
2024-08-16 23:55:29 +08:00
</View>
);
}