SeekBar.js 6.7 KB

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