withdraw.vue 8.9 KB

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