tiefen_space_app/components/MySlider/index.jsx

292 lines
8.2 KiB
React
Raw Normal View History

2024-08-16 23:55:29 +08:00
import {
View,
Text,
TouchableOpacity,
Modal,
PanResponder,
Dimensions,
} from "react-native";
import React, { useState, useEffect, useRef } from "react";
import { useTailwind } from "tailwind-rn";
import { Image } from "expo-image";
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,
lable,
2024-08-16 23:55:29 +08:00
height = 60,
2024-08-21 17:54:04 +08:00
width = Dimensions.get("window").width - 48,
onChange,
2024-08-16 23:55:29 +08:00
onAfterChange = () => {},
defaultValue = 0,
disabled = false,
2024-08-21 17:54:04 +08:00
thumbSize = 25,
leftValue = 0,
rightValue = 0,
2024-08-16 23:55:29 +08:00
thumbImage = null,
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-16 23:55:29 +08:00
const [processWidth, setProcessWidth] = useState(330);
const tailwind = useTailwind();
const leftWatcher = useRef(null);
const rightWatcher = useRef(null);
useEffect(() => {
leftWatcher.current = PanResponder.create({
// 建立监视器
onStartShouldSetPanResponder: () => true,
onPanResponderGrant: _onPanResponderGrantLeft, // 按下
onPanResponderMove: _onPanResponderMoveLeft, // 移动
onPanResponderEnd: _onPanResponderEndLeft, // 结束
});
rightWatcher.current = PanResponder.create({
// 建立监视器
onStartShouldSetPanResponder: () => true,
onPanResponderGrant: _onPanResponderGrantRight, // 按下
onPanResponderMove: _onPanResponderMoveRight, // 移动
onPanResponderEnd: _onPanResponderEndRight, // 结束
});
2024-08-21 17:54:04 +08:00
const currentLeftProcess = 0;
const currentRightProcess = 1;
// leftProcess.current = currentLeftProcess;
setLeftProcess(currentLeftProcess);
// rightProcess.current = currentRightProcess;
setRightProcess(currentRightProcess);
2024-08-16 23:55:29 +08:00
// setRightProcess(rightProcess);
2024-08-21 17:54:04 +08:00
setProcessWidth(width - thumbSize * 2);
console.log("rightProcess", currentRightProcess);
2024-08-16 23:55:29 +08:00
}, []);
2024-08-21 17:54:04 +08:00
useEffect(() => {
onChange({
lower_bound: Math.floor(
leftProcess * (upper_bound - lower_bound) + lower_bound
),
upper_bound: Math.floor(
rightProcess * (upper_bound - lower_bound) + lower_bound
),
});
}, [leftProcess, rightProcess]);
2024-08-16 23:55:29 +08:00
// 左侧滑块事件
const _onPanResponderGrantLeft = (e, gestureState) => {
2024-08-21 17:54:04 +08:00
const process = (gestureState.x0 - 40 - thumbSize / 2) / processWidth;
2024-08-16 23:55:29 +08:00
_changeProcess(process, "left");
};
const _onPanResponderEndLeft = (e, gestureState) => {
if (onAfterChange) {
onAfterChange(gestureState.x0);
}
};
const _onPanResponderMoveLeft = (e, gestureState) => {
const process =
2024-08-21 17:54:04 +08:00
(gestureState.x0 - 40 - thumbSize / 2 + gestureState.dx) / processWidth;
2024-08-16 23:55:29 +08:00
_changeProcess(process, "left");
2024-08-21 17:54:04 +08:00
console.log("newValue_left", gestureState.x0, gestureState.dx, process);
2024-08-16 23:55:29 +08:00
};
// 右侧滑块事件
const _onPanResponderGrantRight = (e, gestureState) => {
2024-08-21 17:54:04 +08:00
const process = (gestureState.x0 - 40 - thumbSize / 2) / processWidth;
2024-08-16 23:55:29 +08:00
_changeProcess(process, "right");
};
const _onPanResponderEndRight = (e, gestureState) => {
if (onAfterChange) {
onAfterChange(gestureState.x0);
}
};
const _onPanResponderMoveRight = (e, gestureState) => {
const process =
2024-08-21 17:54:04 +08:00
(gestureState.x0 - 40 - thumbSize / 2 + gestureState.dx) / processWidth;
2024-08-16 23:55:29 +08:00
_changeProcess(process, "right");
// console.log("process", processWidth);
};
const _changeProcess = (changeProcess, direction) => {
// 判断滑动开关
if (disabled) return;
if (changeProcess >= 0 && changeProcess <= 1) {
// 按步长比例变化刻度
2024-08-21 17:54:04 +08:00
const v = changeProcess * (upper_bound - lower_bound);
2024-08-16 23:55:29 +08:00
const newValue = Math.round(v / step) * step;
2024-08-21 17:54:04 +08:00
const newProcess = newValue / (upper_bound - lower_bound);
2024-08-16 23:55:29 +08:00
if (process !== newProcess) {
if (direction == "left") {
2024-08-21 17:54:04 +08:00
if (newProcess < rightProcess) {
setLeftProcess(newProcess);
}
2024-08-16 23:55:29 +08:00
} else {
2024-08-21 17:54:04 +08:00
if (newProcess > leftProcess) {
console.log("newValue", newProcess, rightProcess);
// rightProcess.current = newProcess;
setRightProcess(newProcess);
2024-08-16 23:55:29 +08:00
}
}
}
}
};
return (
<View
// style={[
// styles.container,
// {
// height,
// width,
// },
// ]}
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,
marginTop: 12,
2024-08-16 23:55:29 +08:00
}}
>
<View style={{ width: "100%", position: "relative" }}>
<View
style={{
backgroundColor: maximumTrackTintColor,
// flex: 1,
height: processHeight,
width: processWidth,
// marginRight: thumbSize / 2,
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>
<View
transition={1000}
cachePolicy="disk"
// style={tailwind("w-full h-full rounded-lg overflow-hidden")}
style={{
width: thumbSize,
height: thumbSize,
borderRadius: 50,
position: "absolute",
2024-08-21 17:54:04 +08:00
left: leftProcess * processWidth,
marginTop: -processHeight / 2,
2024-08-16 23:55:29 +08:00
zIndex: 10,
2024-08-21 17:54:04 +08:00
backgroundColor: "#ff75c8",
borderColor: "#ffffff",
borderWidth: 2,
2024-08-16 23:55:29 +08:00
}}
{...leftWatcher.current?.panHandlers}
/>
<View
transition={1000}
cachePolicy="disk"
// style={tailwind("w-full h-full rounded-lg overflow-hidden")}
style={{
width: thumbSize,
height: thumbSize,
backgroundColor: "#fff",
borderRadius: 50,
position: "absolute",
2024-08-21 17:54:04 +08:00
left: rightProcess * processWidth,
2024-08-16 23:55:29 +08:00
zIndex: 10,
2024-08-21 17:54:04 +08:00
backgroundColor: "#ff75c8",
borderColor: "#ffffff",
borderWidth: 2,
2024-08-16 23:55:29 +08:00
}}
{...rightWatcher.current?.panHandlers}
2024-08-21 17:54:04 +08:00
></View>
<View
style={{
...tailwind("flex flex-row justify-between items-center"),
position: "absolute",
top: -46,
right: 12,
}}
>
<Text
style={{
...tailwind("text-white text-lg"),
maxWidth: thumbSize * 2,
color: "#ff75c8",
}}
>
{Math.floor(leftProcess * (upper_bound - lower_bound) + lower_bound)}
</Text>
<Text
style={{
...tailwind("text-lg"),
marginHorizontal: 4,
color: "#ff75c8",
}}
>
-
</Text>
<Text
style={{
...tailwind("text-white text-lg"),
maxWidth: thumbSize * 2,
color: "#ff75c8",
}}
>
{Math.floor(rightProcess * (upper_bound - lower_bound) + lower_bound)}
</Text>
<Text
style={{
...tailwind("text-white text-lg"),
maxWidth: thumbSize * 2,
color: "#ff75c8",
}}
>
{unit}
</Text>
</View>
<Text
style={{
...tailwind("absolute text-white text-sm"),
left: 0,
top: 20,
minWidth: thumbSize,
maxWidth: thumbSize * 2,
color: "#ffffff80",
}}
>
{lower_bound}
</Text>
<Text
style={{
...tailwind("absolute text-white text-sm"),
left: processWidth,
top: 20,
minWidth: thumbSize,
maxWidth: thumbSize * 2,
color: "#ffffff80",
}}
>
{upper_bound}
</Text>
2024-08-16 23:55:29 +08:00
</View>
);
}