You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

229 lines
5.8 KiB

6 months ago
  1. <template>
  2. <view class="hx-jump-ball">
  3. <view class="ballBox" :animation="ballBoxAnimationData" :style="{'z-index':index}">
  4. <view class="ballOuter"
  5. :animation="ballOuterAnimationData"
  6. :style="{width:ballWidth*2 + 'upx',height:ballHeight*2 + 'upx','background-color':backgroundColor,'background-image': backgroundImage ?`url(${backgroundImage})`: ''}">
  7. </view>
  8. </view>
  9. </view>
  10. </template>
  11. <script>
  12. export default {
  13. name: "hx-jump-ball",
  14. data() {
  15. return {
  16. flag: false,
  17. ballBoxAnimation: null,
  18. ballOuterAnimation: null,
  19. ballBoxAnimationData: {},
  20. ballOuterAnimationData: {},
  21. };
  22. },
  23. props: {
  24. //小球宽度
  25. ballWidth: {
  26. type: Number,
  27. default: 15
  28. },
  29. //小球高度
  30. ballHeight: {
  31. type: Number,
  32. default: 15
  33. },
  34. //小球颜色
  35. backgroundColor: {
  36. type: String,
  37. default: "reg"
  38. },
  39. //图片
  40. backgroundImage: {
  41. type: String,
  42. default: ""
  43. },
  44. //小球的堆叠顺序
  45. index: {
  46. type: Number,
  47. default: 0
  48. },
  49. //开始动画
  50. start: {
  51. type: Number,
  52. default: 1
  53. },
  54. //html元素class名称,[起点元素,终点元素]
  55. element:{
  56. type: Array,
  57. default(){
  58. return []
  59. }
  60. },
  61. //下落速度 ms毫秒
  62. speed:{
  63. type: Number,
  64. default: 800
  65. },
  66. //贝塞尔曲线
  67. bezier:{
  68. type: String,
  69. default: "cubic-bezier(.6,-0.63,.94,.71)"
  70. }
  71. },
  72. watch:{
  73. start(val,oldVal) {
  74. var that = this;
  75. if(!that.element){
  76. return;
  77. }
  78. if(this.flag){
  79. return;
  80. }
  81. that.flag = !that.flag;
  82. that.getElementCoordinate(that.element[0],that.element[1]);
  83. }
  84. },
  85. created() {
  86. this.ballBoxAnimation = uni.createAnimation({
  87. duration: 0,
  88. timingFunction: this.bezier,
  89. delay: 0
  90. });
  91. this.ballOuterAnimation = uni.createAnimation({
  92. duration: 0,
  93. timingFunction: "linear",
  94. delay: 0
  95. });
  96. this.setEnd();
  97. //初始化位置
  98. },
  99. methods:{
  100. //获取元素坐标
  101. getElementCoordinate(startElement,endElement){
  102. let that = this;
  103. const query = uni.createSelectorQuery();
  104. let nodesRef = query.select(startElement);
  105. nodesRef.fields({
  106. id: true,
  107. rect: true,
  108. size: true
  109. }, res => {
  110. if(!res){
  111. that.flag = !that.flag;
  112. that.$emit("msg",{code: 100, error: '未获取到起始元素位置'})
  113. return ;
  114. }
  115. const SLeft = res.left + ((res.width + that.ballWidth ) / 2 - that.ballWidth);
  116. const STop = res.bottom - ((res.height - that.ballHeight ) / 2 + that.ballHeight);
  117. let nodesRef2 = uni.createSelectorQuery().select(endElement);
  118. nodesRef2.fields({
  119. id: true,
  120. rect: true,
  121. size: true
  122. }, res2 => {
  123. if(!res2){
  124. that.$emit("msg",{code: 101, error: '未获取到结束元素位置'})
  125. return ;
  126. }
  127. //计算出元素的中心坐标
  128. let ELeft = res2.left + ((res2.width + that.ballWidth ) / 2 - that.ballWidth);
  129. let ETop = res2.bottom - ((res2.height - that.ballHeight ) / 2 + that.ballHeight);
  130. //初始化位置
  131. that.setStart(SLeft,STop,ELeft,ETop);
  132. that.startAnimation(SLeft,STop,ELeft,ETop);
  133. }).exec()
  134. }).exec()
  135. },
  136. //开始动画
  137. startAnimation(SLeft,STop,ELeft,ETop){
  138. let that = this;
  139. let jumpDistance = SLeft - ELeft;
  140. setTimeout(function() {
  141. that.setEnd();
  142. that.flag = !that.flag;
  143. that.$emit("msg",{code:0,status:true});
  144. }, 100 + that.speed);
  145. // 暂时注释掉,待uniapp修复bug后再调整
  146. //根坐标
  147. // this.ballBoxAnimation.translate3d(ELeft,STop,0).step({duration: 800});
  148. // this.ballBoxAnimation.translate3d(ELeft,ETop,0).step({duration: 800});
  149. // this.ballBoxAnimationData = this.ballBoxAnimation.export();
  150. // console.log('根坐标执行玩');
  151. // //相对根的坐标
  152. // this.ballOuterAnimation.translate3d(jumpDistance,0,0).step({duration: 800});
  153. // this.ballOuterAnimation.translate3d(0,0,0).step({duration: 800});
  154. // this.ballOuterAnimationData = this.ballOuterAnimation.export();
  155. // console.log('相对根的坐标');
  156. // setTimeout(function() {
  157. // console.log("动画完成");
  158. // that.flag = !that.flag;
  159. // }, 800);
  160. //因为uniapp step()有bug,所以必须要延时执行
  161. setTimeout(function() {
  162. //根坐标
  163. that.ballBoxAnimation.opacity(1).translate3d(ELeft,ETop,0).step({duration: that.speed});
  164. that.ballBoxAnimationData = that.ballBoxAnimation.export();
  165. //相对根的坐标
  166. that.ballOuterAnimation.opacity(1).translate3d(0,0,0).step({duration: that.speed});
  167. that.ballOuterAnimationData = that.ballOuterAnimation.export();
  168. }, 50);
  169. },
  170. //动画开始前初始化小球位置并显示小球
  171. setStart(SLeft,STop,ELeft,ETop){
  172. this.ballBoxAnimation.translate3d(ELeft,STop,0).opacity(1).step({duration: 0});
  173. this.ballBoxAnimationData = this.ballBoxAnimation.export();
  174. this.ballOuterAnimation.translate3d(SLeft - ELeft,0,0).opacity(1).step({duration: 0});
  175. this.ballOuterAnimationData = this.ballOuterAnimation.export();
  176. },
  177. //隐藏小球
  178. setEnd(){
  179. this.ballBoxAnimation.opacity(0).step({duration: 0});
  180. this.ballBoxAnimationData = this.ballBoxAnimation.export();
  181. this.ballOuterAnimation.opacity(0).step({duration: 0});
  182. this.ballOuterAnimationData = this.ballOuterAnimation.export();
  183. }
  184. }
  185. }
  186. </script>
  187. <style>
  188. .ballBox{
  189. position: fixed;
  190. left: 0;
  191. top: 0;
  192. z-index: 9;
  193. /* 用颜色来演示用原理 */
  194. /*background-color: #4CD964*/;
  195. height:30rpx;
  196. width:30rpx;
  197. }
  198. .ballOuter {
  199. background:red;
  200. height:100%;
  201. width:100%;
  202. border-radius: 50%;
  203. background-size: 100% 100%;
  204. background-position: center;
  205. }
  206. </style>