SeekBar.js 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. import React, { Component } from "react";
  2. import {
  3. Platform,
  4. StyleSheet,
  5. Text,
  6. View,
  7. Image,
  8. Button,
  9. PanResponder,
  10. findNodeHandle,
  11. UIManager,
  12. DeviceEventEmitter
  13. } from "react-native";
  14. import BasePage from "../BasePage";
  15. const instructions = Platform.select({
  16. ios: "Press Cmd+R to reload,\n" + "Cmd+D or shake for dev menu",
  17. android:
  18. "Double tap R on your keyboard to reload,\n" +
  19. "Shake or press menu button for dev menu"
  20. });
  21. type Props = {};
  22. export default class SeekBar extends BasePage {
  23. constructor(props) {
  24. super(props);
  25. this.pressStatus = false;
  26. }
  27. config = {
  28. changeX: 0,
  29. changeY: 0,
  30. xDiff: 0,
  31. yDiff: 0
  32. };
  33. state = {
  34. max: this.props.max,
  35. view_width: 0,
  36. icon_transform: [{ scale: 1 }],
  37. left: 0,
  38. top: 0,
  39. touch_down: false
  40. };
  41. render() {
  42. return (
  43. <View
  44. style={{
  45. height: 80,
  46. width: "100%",
  47. backgroundColor: "black",
  48. alignItems: "center",
  49. justifyContent: "center"
  50. }}
  51. >
  52. <View
  53. //只有红线黄圆重叠处,点红线外圆点外部分无法选中
  54. style={{
  55. height: 5,
  56. width: "90%",
  57. backgroundColor: "red"
  58. }}
  59. ref={c => {
  60. this.progressBar = c;
  61. }}
  62. onLayout={() => {
  63. const handle = findNodeHandle(this.progressBar);
  64. setTimeout(() => {
  65. UIManager.measure(handle, (x, y, width, height, pageX, pageY) => {
  66. // console.warn(x, y, width, height, pageX, pageY);
  67. this.setState({
  68. view_width: width
  69. });
  70. });
  71. }, 1000);
  72. }}
  73. >
  74. <Image
  75. {...this._panResponder.panHandlers}
  76. style={{
  77. height: 20,
  78. width: 20,
  79. borderRadius: 10,
  80. backgroundColor: "yellow",
  81. transform: this.state.icon_transform,
  82. left: this.state.left,
  83. top: -8
  84. }}
  85. />
  86. </View>
  87. </View>
  88. );
  89. }
  90. componentWillMount() {
  91. this._panResponder = PanResponder.create({
  92. onStartShouldSetPanResponder: (evt, gestureState) => true,
  93. onStartShouldSetPanResponderCapture: (evt, gestureState) => true,
  94. onMoveShouldSetPanResponder: (evt, gestureState) => true,
  95. onMoveShouldSetPanResponderCapture: (evt, gestureState) => true,
  96. onPanResponderGrant: (evt, gestureState) => {
  97. this.pressStatus = true;
  98. this.config.changeY = evt.nativeEvent.pageY;
  99. this.config.changeX = evt.nativeEvent.pageX;
  100. },
  101. onPanResponderStart: (evt, gestureState) => {
  102. this.pressStatus = true;
  103. console.log("onPanResponderStart");
  104. this.setState({
  105. icon_transform: [{ scale: 1.5 }],
  106. touch_down: true
  107. });
  108. },
  109. onPanResponderMove: (evt, gestureState) => {
  110. this.config.yDiff = evt.nativeEvent.pageY - this.config.changeY;
  111. this.config.xDiff = evt.nativeEvent.pageX - this.config.changeX;
  112. this.state.left = this.state.left + this.config.xDiff;
  113. this.state.top = this.state.top + this.config.yDiff;
  114. this.config.changeY = evt.nativeEvent.pageY;
  115. this.config.changeX = evt.nativeEvent.pageX;
  116. if (
  117. this.state.left < 0 ||
  118. this.state.left > this.state.view_width - 20
  119. ) {
  120. if (this.state.left < 0) {
  121. this.state.left = 0;
  122. } else if (this.state.left > this.state.view_width - 20) {
  123. this.state.left = this.state.view_width - 20;
  124. }
  125. this.setState({
  126. left: this.state.left,
  127. top: this.state.top
  128. });
  129. } else {
  130. this.setState({
  131. left: this.state.left,
  132. top: this.state.top
  133. });
  134. console.log("onPanResponderMove:" + this.state.left);
  135. }
  136. },
  137. onPanResponderEnd: (evt, gestureState) => {
  138. this.pressStatus = true;
  139. console.log("onPanResponderEnd");
  140. },
  141. onPanResponderTerminationRequest: (evt, gestureState) => true,
  142. onPanResponderRelease: (evt, gestureState) => {
  143. if (this.pressStatus) {
  144. this.props.onPress && this.props.onPress();
  145. this.setState({
  146. icon_transform: [{ scale: 1 }],
  147. touch_down: false
  148. });
  149. }
  150. this.pressStatus = false;
  151. var sss = this.state.left / (this.state.view_width - 20);
  152. // alert(sss + "%");
  153. this.props.touchUpCallBack(this.state.max * sss);
  154. },
  155. onPanResponderTerminate: (evt, gestureState) => {
  156. // 另一个组件已经成为了新的响应者,所以当前手势将被取消。
  157. },
  158. onShouldBlockNativeResponder: (evt, gestureState) => {
  159. // 返回一个布尔值,决定当前组件是否应该阻止原生组件成为JS响应者
  160. // 默认返回true。目前暂时只支持android。
  161. //基于业务交互场景,如果这里使用js事件处理,会导致容器不能左右滑动。所以设置成false.
  162. return false;
  163. }
  164. });
  165. }
  166. getTouchDown() {
  167. return this.state.touch_down;
  168. }
  169. setProgress(progress) {
  170. if (this.state.touch_down) {
  171. } else {
  172. var aa = progress / this.state.max;
  173. console.warn(
  174. "progress:" +
  175. progress +
  176. "--max:" +
  177. this.state.max +
  178. " progress/max:" +
  179. aa +
  180. " moveLeft:" +
  181. (this.state.view_width - 20) * aa
  182. );
  183. var aaa = (this.state.view_width - 20) * aa;
  184. this.setState({
  185. left: (this.state.view_width - 20) * aa,
  186. icon_transform: [{ scale: 1 }]
  187. });
  188. }
  189. }
  190. }
  191. const styles = StyleSheet.create({});
  192. /**
  193. <SeekBar
  194. ref={view => (this.seekbar = view)}
  195. max={100}
  196. //必须带此方法,作为滑动之后抬起的回调
  197. touchUpCallBack={this.touch_up_callback.bind(this)}
  198. progress={this.state.curprogress}
  199. />
  200. touch_up_callback(progress) {
  201. //抬起之后,获取算出来的progress
  202. this.setState({
  203. curprogress: progress
  204. });
  205. }
  206. setProgressIcon() {
  207. //轮询设置值,超过max则停止
  208. setTimeout(() => {
  209. if (!this.seekbar.getTouchDown()) {
  210. this.setState({
  211. curprogress: this.state.curprogress + 1
  212. });
  213. }
  214. if (this.state.curprogress >= 100) {
  215. alert("100了");
  216. } else {
  217. this.setProgressIcon();
  218. }
  219. this.seekbar.setProgress(this.state.curprogress);
  220. }, 1000);
  221. }
  222. */