withdraw.vue 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
  1. <!-- 提现 -->
  2. <template>
  3. <s-layout title="提现">
  4. <view class="bg-white ss-modal-box ss-flex-col">
  5. <!-- 提现方式 -->
  6. <view class="modal-content">
  7. <view class="out-title ss-p-l-30 ss-m-y-30">选择提现方式</view>
  8. <radio-group @change="onTapOut">
  9. <label class="out-type-item" v-for="item in state.outMethods" :key="item.title">
  10. <view class="out-item ss-flex ss-col-center ss-row-between ss-p-x-30 border-bottom"
  11. :class="{ 'disabled-out-item': item.disabled }">
  12. <view class="ss-flex ss-col-center">
  13. <view class="check-box ss-flex ss-col-center ss-p-l-10">
  14. <radio
  15. :value="item.value"
  16. color="var(--ui-BG-Main)"
  17. style="transform: scale(0.8)"
  18. :disabled="item.disabled"
  19. :checked="state.payment === item.value"
  20. />
  21. </view>
  22. <text class="out-title">{{ item.title }}</text>
  23. </view>
  24. <text style="float: right;" v-if="item.account">{{item.account}}</text>
  25. <text style="float: right;" v-else @click="handleBind(item.value)">暂未绑定,点击绑定&nbsp;></text>
  26. </view>
  27. </label>
  28. </radio-group>
  29. </view>
  30. <!-- 提现金额 -->
  31. <view class="modal-content ">
  32. <view class="out-title ss-p-l-30 ss-m-y-30">提现金额</view>
  33. <view class="ss-flex ss-row-left ss-col-center input-money ss-m-y-30" >
  34. <input v-model.number="state.outMoney" class="uni-input " type="number"
  35. placeholder="请输入金额" oninput="handleInput" />
  36. </view>
  37. <view class="ss-flex ss-row-center ss-col-center ss-m-t-30">
  38. 您当前可兑换金额:¥<text class="text-red">{{canUseMoney}}</text>
  39. <button class="ss-m-l-10 all-btn " @click="useAllPonints">全部</button>
  40. </view>
  41. </view>
  42. <!-- 工具 -->
  43. <view class="modal-footer ss-flex ss-row-center ss-col-center ss-m-t-80 ss-m-b-40 ss-flex-5">
  44. <button class="ss-reset-button save-btn" @tap="submit" :disabled="state.disabled" :class="{ 'disabled-btn': state.disabled }"
  45. >
  46. 确定
  47. </button>
  48. </view>
  49. </view>
  50. </s-layout>
  51. </template>
  52. <script setup>
  53. import {
  54. computed,
  55. reactive,
  56. watchEffect,
  57. nextTick,
  58. onUnmounted
  59. } from 'vue';
  60. import {
  61. onLoad
  62. } from '@dcloudio/uni-app';
  63. import sheep from '@/sheep';
  64. import { clone } from 'lodash';
  65. import {
  66. fen2yuan,
  67. points2point
  68. } from '@/sheep/hooks/useGoods';
  69. import md5 from 'blueimp-md5';
  70. import PayWalletApi from '@/sheep/api/pay/wallet';
  71. import { showAuthModal, showShareModal } from '@/sheep/hooks/useModal';
  72. const userWallet = computed(() => sheep.$store('user').userWallet);
  73. const userInfo = computed(() => sheep.$store('user').userInfo);
  74. const canUseMoney = computed(() => points2point(userWallet.value.integralDO.currentQuota));
  75. const alipayAccount = (account) => {
  76. // let account = state.model.alipayAccount;
  77. if (!account) {
  78. return false
  79. }
  80. // 手机号脱敏
  81. if (/^\d{11}$/.test(account)) { // 检查是否是11位数字的手机号
  82. return `${account.substring(0, 3)}****${account.substring(7)}`;
  83. } else if (/^\w+([.-]?\w+)*@\w+([.-]?\w+)*(\.\w{2,3})+$/.test(account)) {
  84. const atIndex = account.indexOf('@');
  85. // 邮箱用户名长度小于等于3位时,不脱敏
  86. if (atIndex <= 3) {
  87. return account;
  88. }
  89. const username = account.substring(0, Math.ceil(atIndex / 2)); // 取邮箱用户名的一半
  90. const domain = account.substring(atIndex); // 邮箱域名部分
  91. return `${username}***${domain}`;
  92. }
  93. }
  94. const bankAccount = (account) => {
  95. if (!account) {
  96. return false
  97. }
  98. if (account.length === 8) {
  99. return account.substring(0, 2) + '********' + account.substr(-2);
  100. } else {
  101. return account.substring(0, 4) + '******' + account.substr(-4);
  102. }
  103. }
  104. const state = reactive({
  105. model: {},
  106. orderType: 'goods', // 订单类型; goods - 商品订单, recharge - 充值订单
  107. outMent: '',
  108. outMoney:undefined,
  109. disabled:true,
  110. outMethods: [
  111. // {
  112. // title: "提现到微信",
  113. // value: '1'
  114. // },
  115. {
  116. title: "提现到支付宝",
  117. value: '2',
  118. account:''
  119. },
  120. {
  121. title: "提现到银行卡",
  122. value: '3',
  123. account:''
  124. }
  125. ]
  126. });
  127. const handleBind = async (type) => {
  128. // console.log(type)
  129. if(type === '2'){
  130. showAuthModal('alipayAccount');
  131. }else if(type === '3'){
  132. showAuthModal('bankAccount');
  133. }
  134. }
  135. const submit = async () => {
  136. if (state.outMent === '') {
  137. sheep.$helper.toast('请选择提现方式');
  138. return;
  139. }
  140. if (!state.outMoney) {
  141. sheep.$helper.toast('请输入提现金额');
  142. return;
  143. }
  144. if (state.outMent === '2' && !userInfo.value.alipayAccount){
  145. // 没绑定支付宝
  146. uni.showModal({
  147. title: '提示',
  148. content: '未绑定支付宝账号',
  149. confirmText:'去绑定',
  150. success: async function(res) {
  151. if (!res.confirm) {
  152. return;
  153. }
  154. showAuthModal('alipayAccount');
  155. },
  156. });
  157. return;
  158. }
  159. if (state.outMent === '3' && !userInfo.value.bankAccount){
  160. // 没绑定银行卡
  161. uni.showModal({
  162. title: '提示',
  163. content: '未绑定银行卡',
  164. confirmText:'去绑定',
  165. success: async function(res) {
  166. if (!res.confirm) {
  167. return;
  168. }
  169. showAuthModal('bankAccount');
  170. },
  171. });
  172. return;
  173. }
  174. let {
  175. code,
  176. data
  177. } = await PayWalletApi.createWithdrawal({
  178. amount:state.outMoney,
  179. withdrawalType:state.outMent
  180. });
  181. if(code === 0){
  182. uni.showToast({
  183. icon: 'success',
  184. title: "申请成功",
  185. });
  186. sheep.$router.go('/pages/user/wallet/withdrawalLog')
  187. uni.$emit('createWithDrawComplete');
  188. }
  189. };
  190. // 切换提现方式
  191. function onTapOut(e) {
  192. state.outMent = e.detail.value;
  193. }
  194. // 提现全部积分
  195. async function useAllPonints(){
  196. const {code,data} = await PayWalletApi.getDuserInfo();
  197. const userCanUsePoints = parseFloat(points2point(data.integralDO.currentQuota));
  198. state.outMoney = parseInt(userCanUsePoints);
  199. state.disable = false;
  200. }
  201. watchEffect(() => {
  202. // 提现金额不能大于可用积分
  203. if (state.outMoney > canUseMoney.value) {
  204. // 使用 nextTick 确保 DOM 更新
  205. nextTick(() => {
  206. state.outMoney = canUseMoney.value;
  207. });
  208. }
  209. // 如果计算出来的当前可以使用的最大积分等于小于0 则不给输入
  210. if(canUseMoney.value == 0 || canUseMoney.value < 0){
  211. state.disabled = true
  212. }
  213. if(canUseMoney.value > 0){
  214. state.disabled = false
  215. }
  216. })
  217. // 获得用户信息
  218. const getUserInfo = async () => {
  219. // 个人信息
  220. const userInfo = await sheep.$store('user').getInfo();
  221. state.model = clone(userInfo);
  222. state.outMethods.forEach(item=>{
  223. if(item.value === '2'){
  224. item.account = alipayAccount(state.model.alipayAccount);
  225. }else if(item.value === '3'){
  226. item.account = bankAccount(state.model.bankAccount);
  227. }
  228. // console.log(item)
  229. })
  230. };
  231. onLoad(async(options) => {
  232. await getUserInfo();
  233. // refresh()
  234. uni.$on('alipayAccountChangeComplete', getUserInfo);
  235. uni.$on('bankAccountChangeComplete', getUserInfo);
  236. });
  237. </script>
  238. <style lang="scss" scoped>
  239. .all-btn{
  240. height: 60rpx;
  241. line-height: 60rpx;
  242. min-width: 80rpx;
  243. padding: 0 30rpx;
  244. border-radius: 30rpx;
  245. font-size: 26rpx;
  246. margin-right: 10rpx;
  247. border: 2rpx solid var(--ui-BG-Main);
  248. color: var(--ui-BG-Main);
  249. }
  250. .out-icon {
  251. width: 36rpx;
  252. height: 36rpx;
  253. margin-right: 26rpx;
  254. }
  255. .ss-modal-box {
  256. height: calc(100vh - 88rpx);
  257. // max-height: 1000rpx;
  258. .input-money {
  259. width:90% ;
  260. padding:0 10rpx;
  261. // text-indent: 20rpx;
  262. height: 80rpx;
  263. border: 1px solid #bbbbbb;
  264. border-radius: 10rpx;
  265. margin: 15rpx auto;
  266. font-size: 28rpx;
  267. input{
  268. width: 100%;
  269. height: 100%;
  270. font-size: 28rpx;
  271. }
  272. }
  273. .modal-header {
  274. position: relative;
  275. padding: 60rpx 20rpx 40rpx;
  276. .money-text {
  277. color: $red;
  278. font-size: 46rpx;
  279. font-weight: bold;
  280. font-family: OPPOSANS;
  281. &::before {
  282. content: '¥';
  283. font-size: 30rpx;
  284. }
  285. }
  286. .time-text {
  287. font-size: 26rpx;
  288. color: $gray-b;
  289. }
  290. .close-icon {
  291. position: absolute;
  292. top: 10rpx;
  293. right: 20rpx;
  294. font-size: 46rpx;
  295. opacity: 0.2;
  296. }
  297. }
  298. .modal-content {
  299. overflow-y: auto;
  300. .out-title {
  301. font-size: 26rpx;
  302. font-weight: 500;
  303. color: #333333;
  304. }
  305. .out-tip {
  306. font-size: 26rpx;
  307. color: #bbbbbb;
  308. }
  309. .out-item {
  310. height: 86rpx;
  311. }
  312. .disabled-out-item {
  313. .out-title {
  314. color: #999999;
  315. }
  316. }
  317. .userInfo-money {
  318. font-size: 26rpx;
  319. color: #bbbbbb;
  320. line-height: normal;
  321. }
  322. }
  323. .save-btn {
  324. width: 710rpx;
  325. height: 80rpx;
  326. border-radius: 40rpx;
  327. background: linear-gradient(90deg, var(--ui-BG-Main), var(--ui-BG-Main-gradient));
  328. color: $white;
  329. }
  330. .disabled-btn {
  331. background: #e5e5e5;
  332. color: #999999;
  333. }
  334. .past-due-btn {
  335. width: 710rpx;
  336. height: 80rpx;
  337. border-radius: 40rpx;
  338. background-color: #999;
  339. color: #fff;
  340. }
  341. }
  342. </style>