drag-button.vue 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172
  1. <template>
  2. <view>
  3. <view id="_drag_button" class="drag" :style="{top:top+'px',left:left+'px',opacity:firstIn?1:0}"
  4. @touchstart="touchstart" @touchmove.stop.prevent="touchmove" @touchend="touchend"
  5. @click.stop.prevent="click" :class="{transition: isDock && !isMove }">
  6. <button class="btn" style="border: none;padding: 0;margin: 0;">
  7. <image class="img" src="@/static/icon/cart.png" mode="widthFix"></image>
  8. <text>购物车</text>
  9. </button>
  10. </view>
  11. </view>
  12. </template>
  13. <script>
  14. export default {
  15. name: 'drag-button',
  16. props: {
  17. isDock: {
  18. type: Boolean,
  19. default: false
  20. },
  21. customBar: {
  22. type: Boolean,
  23. default: false
  24. }
  25. },
  26. data() {
  27. return {
  28. top: 0,
  29. left: 0,
  30. width: 0,
  31. height: 0,
  32. offsetWidth: 0,
  33. offsetHeight: 0,
  34. windowWidth: 0,
  35. windowHeight: 0,
  36. isMove: true,
  37. edge: 10,
  38. firstIn: false,
  39. customBarHeight: this.CustomBar
  40. }
  41. },
  42. methods: {
  43. init() {
  44. // 获取手机信息配置接口
  45. const sys = uni.getSystemInfoSync();
  46. // 屏幕的宽高
  47. this.windowWidth = sys.windowWidth;
  48. this.windowHeight = sys.windowHeight;
  49. if (sys.windowTop) {
  50. this.windowHeight += sys.windowTop;
  51. }
  52. // 获取元素
  53. const query = uni.createSelectorQuery().in(this);
  54. query.select('#_drag_button').boundingClientRect(data => {
  55. this.width = data.width;
  56. this.height = data.height;
  57. this.offsetWidth = data.width / 2;
  58. this.offsetHeight = data.height / 2;
  59. // this.left = this.windowWidth - this.width - this.edge;
  60. // this.top = this.windowHeight - this.height - this.edge;
  61. // 若storage无值,设置初始值
  62. if(!uni.getStorageSync('top')) {
  63. uni.setStorageSync("top", this.windowHeight - this.height - this.edge - 100 - (this.customBar ? this.customBarHeight : 0));
  64. }
  65. if(!uni.getStorageSync('left')) {
  66. uni.setStorageSync("left", this.windowWidth - this.width - this.edge);
  67. }
  68. this.left = uni.getStorageSync('left');
  69. this.top = uni.getStorageSync('top') - (this.customBar ? this.customBarHeight : 0);
  70. this.$nextTick(() => {
  71. this.firstIn = true;
  72. })
  73. }).exec();
  74. },
  75. click() {
  76. uni.switchTab({
  77. url:'/pages/cart/index'
  78. })
  79. },
  80. touchstart(e) {
  81. },
  82. touchmove(e) {
  83. // 单指触摸
  84. if (e.touches.length !== 1) {
  85. return false;
  86. }
  87. this.isMove = true;
  88. this.left = e.touches[0].clientX - this.offsetWidth;
  89. let clientY = e.touches[0].clientY - this.offsetHeight;
  90. let edgeBottom = this.windowHeight - this.height - this.edge;
  91. // 上下触及边界
  92. if (clientY < this.edge) {
  93. this.top = this.edge;
  94. } else if (clientY > edgeBottom) {
  95. this.top = edgeBottom;
  96. } else {
  97. this.top = clientY
  98. }
  99. uni.setStorageSync("top", this.top + (this.customBar ? this.customBarHeight : 0));
  100. },
  101. touchend(e) {
  102. if (this.isDock) {
  103. let edgeRigth = this.windowWidth - this.width - this.edge;
  104. if (this.left < this.windowWidth / 2 - this.offsetWidth) {
  105. this.left = this.edge;
  106. } else {
  107. this.left = edgeRigth;
  108. }
  109. }
  110. uni.setStorageSync("left", this.left);
  111. this.isMove = false;
  112. },
  113. }
  114. }
  115. </script>
  116. <style lang="scss">
  117. .drag {
  118. display: flex;
  119. justify-content: center;
  120. align-items: center;
  121. width: 100rpx;
  122. height: 100rpx;
  123. border-radius: 50%;
  124. font-size: $uni-font-size-sm;
  125. position: fixed;
  126. z-index: 999999;
  127. background: #FFA162;
  128. border: 4rpx solid #FE781F;
  129. &.transition {
  130. transition: left .3s ease, top .3s ease;
  131. }
  132. }
  133. .btn {
  134. background-color: transparent;
  135. width: 100rpx;
  136. height: 100rpx;
  137. border-radius: 50%;
  138. z-index: 9999;
  139. display: flex;
  140. flex-direction: column;
  141. align-items: center;
  142. justify-content: center;
  143. .img {
  144. width: 36rpx;
  145. display: block;
  146. }
  147. text {
  148. font-size: 22rpx;
  149. line-height: 24rpx;
  150. color: #FFFFFF;
  151. margin-top: 8rpx;
  152. }
  153. }
  154. button::after {
  155. border: none;
  156. }
  157. </style>