xuan-switch.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. <template>
  2. <view class="switch-container" :style="[{ background: bj_color }]">
  3. <view class="switch_view">
  4. <view
  5. class="switch-item"
  6. :class="{ checked_switch: isSwitch }"
  7. :style="isSwitch ? `color:${checked_color}` : ''"
  8. @click.prevent.stop="changeSwitch(true)"
  9. :animation="animationData2"
  10. >
  11. {{ switchList[0] }}
  12. </view>
  13. <view
  14. class="switch-item"
  15. :class="{ checked_switch: !isSwitch }"
  16. :style="!isSwitch ? `color:${checked_color}` : ''"
  17. @click.prevent.stop="changeSwitch(false)"
  18. :animation="animationData3"
  19. >
  20. {{ switchList[1] }}
  21. </view>
  22. </view>
  23. <view class="disabled" v-if="disabled"></view>
  24. <view
  25. class="position_view"
  26. :animation="animationData1"
  27. :style="[{ background: checked_bj_color }]"
  28. ></view>
  29. </view>
  30. </template>
  31. <script>
  32. export default {
  33. props: {
  34. switchList: {
  35. type: Array,
  36. default: () => {
  37. return ["开", "关"];
  38. },
  39. },
  40. defaultSwitch: {
  41. type: Boolean,
  42. default: true,
  43. },
  44. isShowModal: {
  45. //改变开关时,是否弹框提醒
  46. type: Boolean,
  47. default: false,
  48. },
  49. disabled: {
  50. type: Boolean,
  51. default: false,
  52. },
  53. bj_color: {
  54. type: String,
  55. default: "#fff",
  56. },
  57. // checked_bj_color:{
  58. // type:String,
  59. // default:'rgb(0, 188, 38)',
  60. // },
  61. changeBgColor: {
  62. type: Boolean,
  63. default: true,
  64. },
  65. checked_color: {
  66. type: String,
  67. default: "#fff",
  68. },
  69. },
  70. data() {
  71. return {
  72. isSwitch: true,
  73. initAnimation: {},
  74. animationData1: {},
  75. animationData2: {},
  76. animationData3: {},
  77. checked_bj_color: "00bc26", //#f44336红 #00bc26绿
  78. };
  79. },
  80. created() {
  81. this.initAnimation = uni.createAnimation({
  82. duration: 500,
  83. timingFunction: "ease",
  84. });
  85. this.isSwitch = this.defaultSwitch;
  86. this.changeAnimation();
  87. },
  88. methods: {
  89. changeSwitch(isSwitch) {
  90. if (isSwitch == this.isSwitch || this.disabled) {
  91. return;
  92. }
  93. if (this.isShowModal) {
  94. let index = isSwitch ? 0 : 1;
  95. let text = this.switchList[index];
  96. uni.showModal({
  97. title: "提示",
  98. content: `您确定要将其调整为${text}吗?`,
  99. success: (res) => {
  100. if (res.confirm) {
  101. this.isSwitch = isSwitch;
  102. this.changeAnimation();
  103. this.callParentEvent(isSwitch);
  104. }
  105. },
  106. });
  107. } else {
  108. this.isSwitch = isSwitch;
  109. this.changeAnimation();
  110. this.callParentEvent(isSwitch);
  111. }
  112. },
  113. changeAnimation() {
  114. if (this.isSwitch) {
  115. this.animationData1 = this.initAnimation
  116. .left(0)
  117. .width("60%")
  118. .step()
  119. .export();
  120. this.animationData2 = this.initAnimation
  121. .width("60%")
  122. .step()
  123. .export();
  124. this.animationData3 = this.initAnimation
  125. .width("40%")
  126. .step()
  127. .export();
  128. } else {
  129. this.animationData1 = this.initAnimation
  130. .left("40%")
  131. .width("60%")
  132. .step()
  133. .export();
  134. this.animationData2 = this.initAnimation
  135. .width("40%")
  136. .step()
  137. .export();
  138. this.animationData3 = this.initAnimation
  139. .width("60%")
  140. .step()
  141. .export();
  142. }
  143. },
  144. callParentEvent() {
  145. this.$emit("change", this.isSwitch);
  146. },
  147. },
  148. watch: {
  149. isSwitch(v) {
  150. if (this.changeBgColor) {
  151. if (v) {
  152. this.checked_bj_color = "#00bc26";
  153. } else {
  154. this.checked_bj_color = "#f44336";
  155. }
  156. }
  157. },
  158. },
  159. };
  160. </script>
  161. <style lang="scss" scoped>
  162. .switch-container {
  163. display: flex;
  164. flex-direction: row;
  165. width: 160upx;
  166. height: 60upx;
  167. border-radius: 100upx;
  168. border: 1upx solid #ccc;
  169. position: relative;
  170. .switch_view {
  171. position: absolute;
  172. top: 0;
  173. left: 0;
  174. width: 100%;
  175. height: 100%;
  176. z-index: 1;
  177. display: flex;
  178. border-radius: 100upx;
  179. .switch-item {
  180. color: #666;
  181. font-size: 24upx;
  182. height: 100%;
  183. width: 40%;
  184. border-radius: 100upx;
  185. display: flex;
  186. justify-content: center;
  187. align-items: center;
  188. }
  189. }
  190. .position_view {
  191. position: absolute;
  192. top: 0;
  193. left: 0;
  194. width: 60%;
  195. height: 100%;
  196. border-radius: 100upx;
  197. background: rgb(0, 188, 38);
  198. transition: all 0.3s;
  199. }
  200. .disabled {
  201. position: absolute;
  202. top: 0;
  203. left: 0;
  204. width: 100%;
  205. height: 100%;
  206. z-index: 99;
  207. background: #fff;
  208. opacity: 0.6;
  209. border-radius: 100upx;
  210. }
  211. }
  212. </style>