info.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450
  1. <!-- 个人信息 -->
  2. <template>
  3. <view class="container">
  4. <view class="u-flex u-row-between u-p-x-30 info-head u-p-y-20">
  5. <view class="info-title">基本信息</view>
  6. <button class="u-reset-button save-btn" :class="{ 'dis-btn': editInfoDisabled }" :disabled="editInfoDisabled" @tap="editUserInfo">保存</button>
  7. </view>
  8. <view class="user-list u-flex u-row-between">
  9. <text class="list-name">头像</text>
  10. <view class="u-flex" @tap="onChooseImg">
  11. <image class="avatar" :src="userData.avatar" mode=""></image>
  12. <text class="u-iconfont uicon-arrow-right u-m-l-20" style="color:#999;"></text>
  13. </view>
  14. </view>
  15. <view class="user-list u-flex u-row-between">
  16. <text class="list-name">昵称</text>
  17. <view class="u-flex">
  18. <input
  19. class="list-val"
  20. v-model="userData.nickname"
  21. @input="onChangeNickName"
  22. @blur="onChangeNickName"
  23. @confirm="onChangeNickName"
  24. placeholder="请输入昵称~"
  25. maxlength="50"
  26. placeholder-style="color:#c8c9cc;fontSize:28rpx"
  27. />
  28. <text class="u-iconfont uicon-arrow-right u-m-l-20" style="color:#999;"></text>
  29. </view>
  30. </view>
  31. <view class="user-list u-flex u-row-between u-m-b-10" @tap="showCalendar = true">
  32. <text class="list-name">生日</text>
  33. <view class="u-flex">
  34. <text class="list-val">{{ userData.birthday || '请选择生日~' }}</text>
  35. <text class="u-iconfont uicon-arrow-right u-m-l-20" style="color:#999;"></text>
  36. </view>
  37. </view>
  38. <view class="user-list u-flex u-row-between" @tap="$Router.push('/pages/user/address/list')">
  39. <text class="list-name">地址管理</text>
  40. <view class="u-flex">
  41. <text class="list-val"></text>
  42. <text class="u-iconfont uicon-arrow-right u-m-l-20" style="color:#999;"></text>
  43. </view>
  44. </view>
  45. <view class="user-list u-flex u-row-between" @tap="changePwd">
  46. <text class="list-name">修改密码</text>
  47. <view class="u-flex">
  48. <text class="list-val"></text>
  49. <text class="u-iconfont uicon-arrow-right u-m-l-20" style="color:#999;"></text>
  50. </view>
  51. </view>
  52. <view class="user-list u-flex u-row-between" @tap="bindMobile">
  53. <text class="list-name">手机号</text>
  54. <view class="u-flex">
  55. <text class="list-val">{{ userInfo.mobile || '暂未绑定手机号' }}</text>
  56. <text v-if="userInfo.id && !userInfo.verification.mobile" class="u-iconfont uicon-arrow-right u-m-l-20" style="color:#999;"></text>
  57. <text v-else class="u-iconfont uicon-checkmark-circle-fill u-m-l-20" style="color:#09BB07;"></text>
  58. </view>
  59. </view>
  60. <!-- 第三方账号 -->
  61. <view class="other-account" v-if="platform !== 'H5'">
  62. <view class="head-title u-p-x-30">第三方账号绑定</view>
  63. <!-- #ifdef MP-WEIXIN -->
  64. <!-- 微信小程序 -->
  65. <view v-if="platform === 'wxMiniProgram'" class="account-list u-flex u-row-between">
  66. <view class=" u-flex u-col-center">
  67. <image class="list-img" :src="$IMG_URL + '/imgs/user/bind_wxmp.png'" mode=""></image>
  68. <text class="list-name">微信小程序</text>
  69. </view>
  70. <view class="u-flex u-col-center">
  71. <view class="info u-flex u-col-center">
  72. <open-data class="avatar u-m-r-20" type="userAvatarUrl"></open-data>
  73. <open-data class="name" type="userNickName"></open-data>
  74. </view>
  75. <view class="bind-box u-m-l-20">
  76. <button class="u-reset-button bind-btn" v-if="userInfo.id && !userInfo.verification.wxMiniProgram" @tap="bindThirdOauth('wxMiniProgram', 'Wechat')">
  77. 绑定
  78. </button>
  79. <button class="u-reset-button relieve-btn" v-else @tap="unbindThirdOauth('wxMiniProgram', 'Wechat')">解绑</button>
  80. </view>
  81. </view>
  82. </view>
  83. <!-- #endif -->
  84. <!-- #ifdef H5 -->
  85. <!-- 微信公众号 -->
  86. <view v-if="platform === 'wxOfficialAccount'" class="account-list u-flex u-row-between">
  87. <view class=" u-flex u-col-center">
  88. <image class="list-img" :src="$IMG_URL + '/imgs/user/bind_wxh5.png'" mode=""></image>
  89. <text class="list-name">微信公众号</text>
  90. </view>
  91. <view class="u-flex u-col-center">
  92. <view class="info u-flex u-col-center" v-if="thirdOauthInfo">
  93. <image class="avatar u-m-r-20" :src="thirdOauthInfo.headimgurl" mode=""></image>
  94. <view class="name">{{ thirdOauthInfo.nickname }}</view>
  95. </view>
  96. <view class="bind-box u-m-l-20">
  97. <button class="u-reset-button bind-btn" v-if="userInfo.id && !userInfo.verification.wxOfficialAccount" @tap="bindThirdOauth('wxOfficialAccount', 'Wechat')">
  98. 绑定
  99. </button>
  100. <button class="u-reset-button relieve-btn" v-else @tap="unbindThirdOauth('wxOfficialAccount', 'Wechat')">解绑</button>
  101. </view>
  102. </view>
  103. </view>
  104. <!-- #endif -->
  105. <!-- #ifdef APP-PLUS -->
  106. <!-- App -->
  107. <view v-if="platform === 'App'" class="account-list u-flex u-row-between">
  108. <view class=" u-flex u-col-center">
  109. <image class="list-img" :src="$IMG_URL + '/imgs/user/bind_wx.png'" mode=""></image>
  110. <text class="list-name">微信</text>
  111. </view>
  112. <view class="u-flex u-col-center">
  113. <view class="info u-flex u-col-center" v-if="thirdOauthInfo">
  114. <image class="avatar u-m-r-20" :src="thirdOauthInfo.headimgurl" mode=""></image>
  115. <view class="name">{{ thirdOauthInfo.nickname }}</view>
  116. </view>
  117. <view class="bind-box u-m-l-20">
  118. <button class="u-reset-button bind-btn" v-if="userInfo.id && !userInfo.verification.wxOpenPlatform" @tap="bindThirdOauth('App', 'Wechat')">绑定</button>
  119. <button class="u-reset-button relieve-btn" v-else @tap="unbindThirdOauth('App', 'Wechat')">解绑</button>
  120. </view>
  121. </view>
  122. </view>
  123. <!-- #endif -->
  124. </view>
  125. <view class="btn-box u-flex u-row-center u-col-center">
  126. <button hover-class="hover-logout-btn" class="u-reset-button logout-btn" @tap="showModal = true">退出登录</button>
  127. </view>
  128. <!-- 登录提示 -->
  129. <shopro-auth-modal v-if="authType"></shopro-auth-modal>
  130. <!-- 日历 -->
  131. <u-picker
  132. @confirm="changeCalendar"
  133. mode="time"
  134. confirm-color="#e9b461"
  135. v-model="showCalendar"
  136. :params="{ year: true, month: true, day: true, hour: false, minute: false, second: false }"
  137. ></u-picker>
  138. <!-- modal -->
  139. <u-modal
  140. ref="uModal"
  141. v-model="showModal"
  142. :show-cancel-button="true"
  143. confirm-color="#F54141"
  144. cancel-color="#666666"
  145. @confirm="confirmLogOut"
  146. content="确定退出登录?"
  147. ></u-modal>
  148. </view>
  149. </template>
  150. <script>
  151. import { mapMutations, mapActions, mapState, mapGetters } from 'vuex';
  152. import Auth from '@/shopro/permission/index.js';
  153. import wechat from '@/shopro/wechat/wechat';
  154. export default {
  155. components: {},
  156. data() {
  157. return {
  158. platform: this.$platform.get(),
  159. userData: {},
  160. showCalendar: false,
  161. showModal: false,
  162. editInfoDisabled: true,
  163. thirdOauthInfo: null //第三方信息
  164. };
  165. },
  166. computed: {
  167. ...mapGetters(['userInfo', 'authType'])
  168. },
  169. onLoad() {
  170. this.userData = JSON.parse(JSON.stringify(this.userInfo));
  171. this.getThirdOauthInfo();
  172. },
  173. methods: {
  174. ...mapActions(['getUserInfo', 'showAuthModal', 'logout']),
  175. // 选择日期
  176. changeCalendar(e) {
  177. this.userData.birthday = `${e.year}-${e.month}-${e.day}`;
  178. this.editInfoDisabled = this.userData.birthday == this.userInfo.birthday;
  179. },
  180. // 修改密码
  181. changePwd() {
  182. this.showAuthModal('changePwd');
  183. },
  184. // 修改昵称
  185. onChangeNickName() {
  186. this.editInfoDisabled = this.userData.nickname == this.userInfo.nickname;
  187. },
  188. // 修改手机号
  189. bindMobile() {
  190. !this.userInfo.verification.mobile && this.showAuthModal('bindMobile');
  191. },
  192. // 更换头像
  193. async onChooseImg() {
  194. let authState = 0;
  195. authState += await new Auth('writePhotosAlbum').check();
  196. if (authState < 1) return;
  197. authState += await new Auth('camera').check();
  198. if (authState < 2) return;
  199. const chooseImageRes = await this.chooseImage(1);
  200. const uploadImageRes = await this.uploadImage(chooseImageRes[0]);
  201. this.userData.avatar = uploadImageRes.fullurl;
  202. this.userData.fileUrl = uploadImageRes.url;
  203. this.editInfoDisabled = this.userData.avatar == this.userInfo.avatar;
  204. },
  205. // 第三方绑定信息
  206. getThirdOauthInfo() {
  207. let that = this;
  208. that.$http('user.thirdOauthInfo').then(res => {
  209. if (res.code === 1) {
  210. that.thirdOauthInfo = res.data;
  211. }
  212. });
  213. },
  214. // 解绑
  215. unbindThirdOauth(platform, provider) {
  216. let that = this;
  217. that.$http('user.unbindThirdOauth', {
  218. platform,
  219. provider
  220. }).then(res => {
  221. if (res.code === 1) {
  222. uni.showToast({
  223. title: res.msg
  224. });
  225. that.thirdOauthInfo = null;
  226. that.getUserInfo();
  227. }
  228. });
  229. },
  230. // 绑定
  231. async bindThirdOauth(platform, provider) {
  232. let that = this;
  233. switch (platform) {
  234. case 'wxOfficialAccount':
  235. wechat.bind();
  236. break;
  237. case 'wxMiniProgram':
  238. await wechat.bind();
  239. that.getUserInfo();
  240. break;
  241. case 'App':
  242. if (provider === 'Wechat') {
  243. await wechat.bind();
  244. }
  245. that.getUserInfo();
  246. that.getThirdOauthInfo();
  247. break;
  248. }
  249. },
  250. // 退出登录
  251. confirmLogOut() {
  252. this.logout();
  253. this.$Router.back();
  254. },
  255. // 修改信息
  256. editUserInfo() {
  257. let that = this;
  258. if (!that.userData.nickname) {
  259. this.$u.toast('昵称不能为空');
  260. return false;
  261. }
  262. that.$http(
  263. 'user.profile',
  264. {
  265. nickname: that.userData.nickname,
  266. birthday: that.userData.birthday,
  267. avatar: that.userData.fileUrl
  268. },
  269. '保存中...'
  270. ).then(res => {
  271. if (res.code === 1) {
  272. that.getUserInfo();
  273. that.editInfoDisabled = true;
  274. that.$u.toast('保存成功');
  275. }
  276. });
  277. },
  278. // 图片处理-选择图片
  279. async chooseImage(count = 1) {
  280. return new Promise((resolve, reject) => {
  281. uni.chooseImage({
  282. count: count, //默认9
  283. sizeType: ['original', 'compressed'], //可以指定是原图还是压缩图,默认二者都有
  284. sourceType: ['album'], //从相册选择
  285. success: res => {
  286. resolve(res.tempFilePaths);
  287. }
  288. });
  289. }).catch(e => {
  290. console.log(e);
  291. });
  292. },
  293. // 图片处理-上传图片
  294. async uploadImage(url) {
  295. const that = this;
  296. return new Promise((resolve, reject) => {
  297. that.$u.toast('上传中...');
  298. uni.uploadFile({
  299. url: that.$API_URL + 'index/upload',
  300. filePath: url,
  301. name: 'file',
  302. success: res => {
  303. res = JSON.parse(res.data);
  304. if (res.code === 1) {
  305. that.$u.toast('上传成功');
  306. resolve(res.data);
  307. } else {
  308. that.$u.toast('上传失败');
  309. }
  310. },
  311. complete: e => {
  312. uni.hideLoading();
  313. }
  314. });
  315. }).catch(e => {
  316. console.log(e);
  317. });
  318. }
  319. }
  320. };
  321. </script>
  322. <style lang="scss" scoped>
  323. // 基本信息
  324. .info-head {
  325. .info-title {
  326. line-height: 60rpx;
  327. font-size: 28rpx;
  328. font-weight: 600;
  329. color: #333333;
  330. }
  331. .save-btn {
  332. color: #fff;
  333. background: linear-gradient(90deg, rgba(233, 180, 97, 1), rgba(238, 204, 137, 1));
  334. width: 100rpx;
  335. border-radius: 30rpx;
  336. height: 60rpx;
  337. line-height: 60rpx;
  338. font-size: 28rpx;
  339. }
  340. .dis-btn {
  341. opacity: 0.5;
  342. }
  343. }
  344. // 三方
  345. .other-account {
  346. .head-title {
  347. line-height: 82rpx;
  348. font-size: 28rpx;
  349. font-weight: 600;
  350. color: #333333;
  351. }
  352. .list-img {
  353. width: 40rpx;
  354. height: 40rpx;
  355. margin-right: 10rpx;
  356. }
  357. .account-list {
  358. background-color: #fff;
  359. height: 100rpx;
  360. padding: 0 20rpx;
  361. .info {
  362. .avatar {
  363. width: 38rpx;
  364. height: 38rpx;
  365. border-radius: 50%;
  366. overflow: hidden;
  367. }
  368. .name {
  369. font-size: 28rpx;
  370. font-weight: 400;
  371. color: #999999;
  372. }
  373. }
  374. .bind-box {
  375. .bind-btn {
  376. padding: 0 20rpx;
  377. line-height: 60rpx;
  378. background: #f5f6f8;
  379. border-radius: 20rpx;
  380. color: #999;
  381. }
  382. .relieve-btn {
  383. border-radius: 20rpx;
  384. padding: 0 20rpx;
  385. line-height: 60rpx;
  386. background: rgba(#ff0000, 0.05);
  387. color: #ff0000;
  388. }
  389. }
  390. }
  391. }
  392. // 个人信息项
  393. .user-list {
  394. background: #fff;
  395. height: 100rpx;
  396. border-bottom: 1rpx solid #f6f6f6;
  397. padding: 0 20rpx;
  398. .list-name {
  399. font-size: 28rpx;
  400. }
  401. .avatar {
  402. width: 67rpx;
  403. height: 67rpx;
  404. border-radius: 50%;
  405. }
  406. .list-val {
  407. color: #999;
  408. font-size: 28rpx;
  409. text-align: right;
  410. }
  411. }
  412. .btn-box {
  413. margin-top: 200rpx;
  414. .logout-btn {
  415. width: 710rpx;
  416. height: 80rpx;
  417. line-height: 80rpx;
  418. border: 1rpx solid #f54141;
  419. border-radius: 40rpx;
  420. font-size: 30rpx;
  421. font-weight: 500;
  422. color: #f54141;
  423. }
  424. .hover-logout-btn {
  425. width: 710rpx;
  426. height: 80rpx;
  427. background: #f54141;
  428. border: 1rpx solid #f54141;
  429. border-radius: 40rpx;
  430. font-size: 30rpx;
  431. font-weight: 500;
  432. color: #ffffff;
  433. }
  434. }
  435. </style>