index.vue 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229
  1. <script setup lang="ts">
  2. import { onMounted, ref } from 'vue'
  3. const props = defineProps({
  4. centerName: {
  5. type: String,
  6. default: '中心点',
  7. },
  8. topName: {
  9. type: String,
  10. default: '',
  11. },
  12. moveName: {
  13. type: String,
  14. default: '3333',
  15. },
  16. })
  17. const emits = defineEmits(['curValue'])
  18. // 定义角度值
  19. const angle = ref(0)
  20. const angleDiff = ref(0)
  21. // 获取滑块元素
  22. const sliderElement = ref<HTMLDivElement | null>(null)
  23. // 鼠标按下事件
  24. function onMouseDown(event: MouseEvent) {
  25. startDrag(event)
  26. }
  27. // 触摸开始事件
  28. function onTouchStart(event: TouchEvent) {
  29. startDrag(event.touches[0])
  30. }
  31. // 开始拖动
  32. function startDrag(event: MouseEvent | Touch) {
  33. document.addEventListener('mousemove', onMouseMove)
  34. document.addEventListener('mouseup', onMouseUp)
  35. document.addEventListener('touchmove', onTouchMove)
  36. document.addEventListener('touchend', onMouseUp)
  37. // 计算初始角度
  38. const rect = sliderElement.value!.getBoundingClientRect()
  39. const x = event.clientX - (rect.left + rect.width / 2)
  40. const y = event.clientY - (rect.top + rect.height / 2)
  41. angle.value = calculateAngle(x, y) + 90
  42. angleDiff.value = calculateAngleDiff(angle.value)
  43. emits('curValue', angleDiff.value)
  44. }
  45. // 鼠标移动事件
  46. function onMouseMove(event: MouseEvent) {
  47. updateAngle(event)
  48. }
  49. // 触摸移动事件
  50. function onTouchMove(event: TouchEvent) {
  51. updateAngle(event.touches[0])
  52. }
  53. // 更新角度
  54. function updateAngle(event: MouseEvent | Touch) {
  55. const rect = sliderElement.value!.getBoundingClientRect()
  56. const x = event.clientX - (rect.left + rect.width / 2)
  57. const y = event.clientY - (rect.top + rect.height / 2)
  58. angle.value = calculateAngle(x, y) + 90
  59. angleDiff.value = calculateAngleDiff(angle.value)
  60. emits('curValue', angleDiff.value)
  61. }
  62. // 鼠标抬起事件
  63. function onMouseUp() {
  64. document.removeEventListener('mousemove', onMouseMove)
  65. document.removeEventListener('mouseup', onMouseUp)
  66. document.removeEventListener('touchmove', onTouchMove)
  67. document.removeEventListener('touchend', onMouseUp)
  68. }
  69. // 计算角度
  70. function calculateAngle(x: number, y: number): number {
  71. const rad = Math.atan2(y, x)
  72. const deg = (rad * 180) / Math.PI
  73. return deg >= 0 ? deg : 360 + deg
  74. }
  75. // 计算角度差
  76. function calculateAngleDiff(currentAngle: number): number {
  77. let diff = currentAngle
  78. if (diff > 360 && diff <= 450) {
  79. diff -= 360
  80. }
  81. else if (diff >= 180 && diff <= 360) {
  82. diff = 360 - diff
  83. }
  84. return diff
  85. }
  86. onMounted(() => {
  87. if (sliderElement.value) {
  88. // 初始化角度
  89. angle.value = calculateAngle(0, 0) // 假设初始位置在正上方
  90. angleDiff.value = calculateAngleDiff(angle.value)
  91. }
  92. const touchArea = document.getElementById('touchArea')
  93. // 监听触摸事件
  94. touchArea!.addEventListener('touchstart', (event) => {
  95. // 记录开始触摸的位置
  96. const xStart = event.touches[0].pageX
  97. touchArea!.addEventListener('touchmove', (event) => {
  98. // 计算移动的距离
  99. const xMove = event.touches[0].pageX - xStart
  100. // 如果移动的距离大于0,则为左滑,小于0则为右滑
  101. if (Math.abs(xMove) > 0) {
  102. // 阻止页面滚动
  103. event.preventDefault()
  104. // 执行滑动处理
  105. // console.log(`Swipe detected, distance: ${xMove}`)
  106. }
  107. })
  108. }, false)
  109. })
  110. </script>
  111. <template>
  112. <div class="slider-container">
  113. <div
  114. id="touchArea"
  115. ref="sliderElement"
  116. class="circle-slider"
  117. @mousedown="onMouseDown"
  118. @mousemove="onMouseMove"
  119. @mouseup="onMouseUp"
  120. @touchstart="onTouchStart"
  121. @touchmove="onTouchMove"
  122. @touchend="onMouseUp"
  123. >
  124. <div class="center-point" />
  125. <div class="absolute-center z-399 w-[100px] text-center text-[16px] text-[#3A3A3A]">
  126. {{ props.centerName }}
  127. </div>
  128. <div class="target-point flex-column flex-center text-[16px] text-[#3A3A3A]">
  129. <van-icon name="play" class="rotate-90" />
  130. <span>{{ props.topName }}</span>
  131. </div>
  132. <div class="move-point" :style="{ transform: `translateX(-50%) rotate(${angle}deg)` }">
  133. <span class="absolute left-26px block w-[180px] text-[16px] text-[#F87171]">{{ props.moveName }}</span>
  134. </div>
  135. <img
  136. src="@/assets/images/task/spatialOrientationAbility/icon_pointer.png"
  137. class="absolute left-[50%] top-[38px] z-19 h-[150px] w-[20px] translate-x-[-50%]"
  138. alt=""
  139. >
  140. <img
  141. src="@/assets/images/task/spatialOrientationAbility/icon_pointer_red.png"
  142. class="red-line absolute left-[50%] top-[40px] z-19 h-[130px] w-[4px] rounded-[2px]"
  143. :style="{ transform: `translateX(-50%) rotate(${angle}deg)` }"
  144. alt=""
  145. >
  146. <span class="absolute top-[295px] block text-[15px] text-[#3A3A3A]">{{ angleDiff.toFixed(5) }}°</span>
  147. </div>
  148. </div>
  149. </template>
  150. <style scoped>
  151. .slider-container {
  152. display: flex;
  153. justify-content: center;
  154. align-items: center;
  155. position: relative;
  156. }
  157. .circle-slider {
  158. position: relative;
  159. width: 280px; /* 直径 */
  160. height: 280px; /* 直径 */
  161. border-radius: 50%;
  162. display: flex;
  163. justify-content: center;
  164. align-items: center;
  165. background-image: url('@/assets/images/task/spatialOrientationAbility/bg_circle.png');
  166. background-size: cover;
  167. background-repeat: no-repeat;
  168. background-position: center; /* 可选,让图片居中对齐 */
  169. }
  170. .center-point {
  171. position: absolute;
  172. top: 50%;
  173. left: 50%;
  174. transform: translate(-50%, -50%);
  175. width: 48px;
  176. height: 48px;
  177. border-radius: 50%;
  178. z-index: 99;
  179. background-image: url('@/assets/images/task/spatialOrientationAbility/icon_center.png');
  180. background-size: cover;
  181. background-repeat: no-repeat;
  182. background-position: center; /* 可选,让图片居中对齐 */
  183. }
  184. .target-point {
  185. position: absolute;
  186. top: -40px;
  187. left: 50%;
  188. transform: translateX(-50%);
  189. width: 160px;
  190. height: 40px;
  191. }
  192. .move-point {
  193. position: absolute;
  194. top: 20px;
  195. left: 50%;
  196. transform-origin: center 550%;
  197. width: 22px;
  198. height: 22px;
  199. border-radius: 50%;
  200. background-color: #f3f3f3 !important;
  201. box-shadow: 0 0 4px 0 #000 !important;
  202. cursor: move;
  203. }
  204. .red-line {
  205. transform-origin: 50% 76%;
  206. }
  207. </style>