504 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Vue
		
	
	
			
		
		
	
	
			504 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Vue
		
	
	
<template>
 | 
						|
  <s-layout title="用户信息" class="set-userinfo-wrap">
 | 
						|
    <uni-forms
 | 
						|
      :model="state.model"
 | 
						|
      :rules="state.rules"
 | 
						|
      labelPosition="left"
 | 
						|
      border
 | 
						|
      class="form-box"
 | 
						|
    >
 | 
						|
      <view class="ss-flex ss-row-center ss-col-center ss-p-t-60 ss-p-b-0 bg-white">
 | 
						|
        <view class="header-box-content">
 | 
						|
          <su-image
 | 
						|
            class="content-img"
 | 
						|
            isPreview
 | 
						|
            :current="0"
 | 
						|
            :src="sheep.$url.cdn(state.model.avatar)"
 | 
						|
            :height="160"
 | 
						|
            :width="160"
 | 
						|
            :radius="80"
 | 
						|
            mode="scaleToFill"
 | 
						|
          ></su-image>
 | 
						|
          <view class="avatar-action">
 | 
						|
            <!-- #ifdef MP -->
 | 
						|
            <button
 | 
						|
              class="ss-reset-button avatar-action-btn"
 | 
						|
              open-type="chooseAvatar"
 | 
						|
              @chooseavatar="onChooseAvatar"
 | 
						|
              >修改</button
 | 
						|
            >
 | 
						|
            <!-- #endif -->
 | 
						|
            <!-- #ifndef MP -->
 | 
						|
            <button class="ss-reset-button avatar-action-btn" @tap="onChangeAvatar">修改</button>
 | 
						|
            <!-- #endif -->
 | 
						|
          </view>
 | 
						|
        </view>
 | 
						|
      </view>
 | 
						|
 | 
						|
      <view class="bg-white ss-p-x-30">
 | 
						|
        <uni-forms-item name="username" label="用户名" @tap="onChangeUsername" class="label-box">
 | 
						|
          <uni-easyinput
 | 
						|
            v-model="userInfo.username"
 | 
						|
            disabled
 | 
						|
            :inputBorder="false"
 | 
						|
            :styles="{ disableColor: '#fff' }"
 | 
						|
            placeholder="设置用户名"
 | 
						|
            :clearable="false"
 | 
						|
            :placeholderStyle="placeholderStyle"
 | 
						|
          >
 | 
						|
            <template v-slot:right>
 | 
						|
              <su-radio class="ss-flex" v-if="userInfo.verification?.username" :modelValue="true" />
 | 
						|
              <button v-else class="ss-reset-button">
 | 
						|
                <text class="_icon-forward" style="color: #bbbbbb; font-size: 26rpx"></text>
 | 
						|
              </button>
 | 
						|
            </template>
 | 
						|
          </uni-easyinput>
 | 
						|
        </uni-forms-item>
 | 
						|
 | 
						|
        <uni-forms-item name="nickname" label="昵称">
 | 
						|
          <uni-easyinput
 | 
						|
            v-model="state.model.nickname"
 | 
						|
            type="nickname"
 | 
						|
            placeholder="设置昵称"
 | 
						|
            :inputBorder="false"
 | 
						|
            :placeholderStyle="placeholderStyle"
 | 
						|
          />
 | 
						|
        </uni-forms-item>
 | 
						|
 | 
						|
        <uni-forms-item name="gender" label="性别">
 | 
						|
          <view class="ss-flex ss-col-center ss-h-100">
 | 
						|
            <radio-group @change="onChangeGender" class="ss-flex ss-col-center">
 | 
						|
              <label class="radio" v-for="item in genderRadioMap" :key="item.value">
 | 
						|
                <view class="ss-flex ss-col-center ss-m-r-32">
 | 
						|
                  <radio
 | 
						|
                    :value="item.value"
 | 
						|
                    color="var(--ui-BG-Main)"
 | 
						|
                    style="transform: scale(0.8)"
 | 
						|
                    :checked="item.value == state.model.gender"
 | 
						|
                  />
 | 
						|
                  <view class="gender-name">{{ item.name }}</view>
 | 
						|
                </view>
 | 
						|
              </label>
 | 
						|
            </radio-group>
 | 
						|
          </view>
 | 
						|
        </uni-forms-item>
 | 
						|
 | 
						|
        <uni-forms-item name="mobile" label="手机号" @tap="onChangeMobile">
 | 
						|
          <uni-easyinput
 | 
						|
            v-model="userInfo.mobile"
 | 
						|
            placeholder="请绑定手机号"
 | 
						|
            :inputBorder="false"
 | 
						|
            disabled
 | 
						|
            :styles="{ disableColor: '#fff' }"
 | 
						|
            :placeholderStyle="placeholderStyle"
 | 
						|
            :clearable="false"
 | 
						|
          >
 | 
						|
            <template v-slot:right>
 | 
						|
              <view class="ss-flex ss-col-center">
 | 
						|
                <su-radio v-if="userInfo.verification?.mobile" :modelValue="true" />
 | 
						|
                <button v-else class="ss-reset-button ss-flex ss-col-center ss-row-center">
 | 
						|
                  <text class="_icon-forward" style="color: #bbbbbb; font-size: 26rpx"></text>
 | 
						|
                </button>
 | 
						|
              </view>
 | 
						|
            </template>
 | 
						|
          </uni-easyinput>
 | 
						|
        </uni-forms-item>
 | 
						|
 | 
						|
        <uni-forms-item name="password" label="登录密码" @tap="onSetPassword">
 | 
						|
          <uni-easyinput
 | 
						|
            v-model="userInfo.password"
 | 
						|
            :placeholder="userInfo.verification?.password ? '修改登录密码' : '点击设置登录密码'"
 | 
						|
            :inputBorder="false"
 | 
						|
            :styles="{ disableColor: '#fff' }"
 | 
						|
            disabled
 | 
						|
            placeholderStyle="color:#BBBBBB;font-size:28rpx;line-height:normal"
 | 
						|
            :clearable="false"
 | 
						|
          >
 | 
						|
            <template v-slot:right>
 | 
						|
              <view class="ss-flex ss-col-center">
 | 
						|
                <su-radio
 | 
						|
                  class="ss-flex"
 | 
						|
                  v-if="userInfo.verification?.password"
 | 
						|
                  :modelValue="true"
 | 
						|
                />
 | 
						|
 | 
						|
                <button v-else class="ss-reset-button ss-flex ss-col-center ss-row-center">
 | 
						|
                  <text class="_icon-forward" style="color: #bbbbbb; font-size: 26rpx"></text>
 | 
						|
                </button>
 | 
						|
              </view>
 | 
						|
            </template>
 | 
						|
          </uni-easyinput>
 | 
						|
        </uni-forms-item>
 | 
						|
      </view>
 | 
						|
 | 
						|
      <view class="bg-white ss-m-t-14">
 | 
						|
        <uni-list>
 | 
						|
          <uni-list-item
 | 
						|
            clickable
 | 
						|
            @tap="sheep.$router.go('/pages/user/address/list')"
 | 
						|
            title="地址管理"
 | 
						|
            showArrow
 | 
						|
            :border="false"
 | 
						|
            class="list-border"
 | 
						|
          ></uni-list-item>
 | 
						|
          <uni-list-item
 | 
						|
            clickable
 | 
						|
            @tap="sheep.$router.go('/pages/user/invoice/list')"
 | 
						|
            title="发票管理"
 | 
						|
            showArrow
 | 
						|
            :border="false"
 | 
						|
            class="list-border"
 | 
						|
          ></uni-list-item>
 | 
						|
        </uni-list>
 | 
						|
      </view>
 | 
						|
    </uni-forms>
 | 
						|
 | 
						|
    <view v-if="sheep.$platform.name !== 'H5'">
 | 
						|
      <view class="title-box ss-p-l-30">第三方账号绑定</view>
 | 
						|
 | 
						|
      <view class="account-list ss-flex ss-row-between">
 | 
						|
        <view v-if="'WechatOfficialAccount' === sheep.$platform.name" class="ss-flex ss-col-center">
 | 
						|
          <image
 | 
						|
            class="list-img"
 | 
						|
            :src="sheep.$url.static('/static/img/shop/platform/WechatOfficialAccount.png')"
 | 
						|
          />
 | 
						|
          <text class="list-name">微信公众号</text>
 | 
						|
        </view>
 | 
						|
        <view v-if="'WechatMiniProgram' === sheep.$platform.name" class="ss-flex ss-col-center">
 | 
						|
          <image
 | 
						|
            class="list-img"
 | 
						|
            :src="sheep.$url.static('/static/img/shop/platform/WechatMiniProgram.png')"
 | 
						|
          />
 | 
						|
          <text class="list-name">微信小程序</text>
 | 
						|
        </view>
 | 
						|
        <view v-if="'App' === sheep.$platform.name" class="ss-flex ss-col-center">
 | 
						|
          <image
 | 
						|
            class="list-img"
 | 
						|
            :src="sheep.$url.static('/static/img/shop/platform/wechat.png')"
 | 
						|
          />
 | 
						|
          <text class="list-name">微信开放平台</text>
 | 
						|
        </view>
 | 
						|
        <view class="ss-flex ss-col-center">
 | 
						|
          <view class="info ss-flex ss-col-center" v-if="state.thirdOauthInfo">
 | 
						|
            <image
 | 
						|
              class="avatar ss-m-r-20"
 | 
						|
              :src="sheep.$url.cdn(state.thirdOauthInfo.avatar)"
 | 
						|
            ></image>
 | 
						|
            <text class="name">{{ state.thirdOauthInfo.nickname }}</text>
 | 
						|
          </view>
 | 
						|
          <view class="bind-box ss-m-l-20">
 | 
						|
            <button
 | 
						|
              v-if="state.thirdOauthInfo"
 | 
						|
              class="ss-reset-button relieve-btn"
 | 
						|
              @tap="unBindThirdOauth"
 | 
						|
            >
 | 
						|
              解绑
 | 
						|
            </button>
 | 
						|
            <button v-else class="ss-reset-button bind-btn" @tap="bindThirdOauth">绑定</button>
 | 
						|
          </view>
 | 
						|
        </view>
 | 
						|
      </view>
 | 
						|
    </view>
 | 
						|
 | 
						|
    <su-fixed bottom placeholder bg="none">
 | 
						|
      <view class="footer-box ss-p-20">
 | 
						|
        <button class="ss-rest-button logout-btn ui-Shadow-Main" @tap="onSubmit">保存</button>
 | 
						|
      </view>
 | 
						|
    </su-fixed>
 | 
						|
  </s-layout>
 | 
						|
</template>
 | 
						|
 | 
						|
<script setup>
 | 
						|
  import { computed, ref, reactive, onBeforeMount, unref } from 'vue';
 | 
						|
  import { mobile, password, username } from '@/sheep/validate/form';
 | 
						|
  import sheep from '@/sheep';
 | 
						|
  import { clone } from 'lodash';
 | 
						|
  import { showAuthModal } from '@/sheep/hooks/useModal';
 | 
						|
 | 
						|
  const state = reactive({
 | 
						|
    model: {},
 | 
						|
    rules: {},
 | 
						|
    thirdOauthInfo: null,
 | 
						|
  });
 | 
						|
 | 
						|
  const placeholderStyle = 'color:#BBBBBB;font-size:28rpx;line-height:normal';
 | 
						|
 | 
						|
  const genderRadioMap = [
 | 
						|
    {
 | 
						|
      name: '男',
 | 
						|
      value: '1',
 | 
						|
    },
 | 
						|
    {
 | 
						|
      name: '女',
 | 
						|
      value: '2',
 | 
						|
    },
 | 
						|
    {
 | 
						|
      name: '未知',
 | 
						|
      value: '0',
 | 
						|
    },
 | 
						|
  ];
 | 
						|
 | 
						|
  const userInfo = computed(() => sheep.$store('user').userInfo);
 | 
						|
 | 
						|
  // 选择性别
 | 
						|
  function onChangeGender(e) {
 | 
						|
    state.model.gender = e.detail.value;
 | 
						|
  }
 | 
						|
  // 修改用户名
 | 
						|
  const onChangeUsername = () => {
 | 
						|
    !state.model.verification?.username && showAuthModal('changeUsername');
 | 
						|
  };
 | 
						|
 | 
						|
  // 修改手机号
 | 
						|
  const onChangeMobile = () => {
 | 
						|
    showAuthModal('changeMobile');
 | 
						|
  };
 | 
						|
 | 
						|
  function onChooseAvatar(e) {
 | 
						|
    const tempUrl = e.detail.avatarUrl || '';
 | 
						|
    uploadAvatar(tempUrl);
 | 
						|
  }
 | 
						|
 | 
						|
  //修改头像
 | 
						|
  function onChangeAvatar() {
 | 
						|
    uni.chooseImage({
 | 
						|
      success: async (chooseImageRes) => {
 | 
						|
        const tempUrl = chooseImageRes.tempFilePaths[0];
 | 
						|
        uploadAvatar(tempUrl);
 | 
						|
      },
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  async function uploadAvatar(tempUrl) {
 | 
						|
    if (!tempUrl) return;
 | 
						|
    let { path } = await sheep.$api.app.upload(tempUrl, 'ugc');
 | 
						|
    state.model.avatar = path;
 | 
						|
  }
 | 
						|
 | 
						|
  // 修改/设置密码
 | 
						|
  function onSetPassword() {
 | 
						|
    if (state.model.verification.password) {
 | 
						|
      showAuthModal('changePassword');
 | 
						|
    } else {
 | 
						|
      showAuthModal('resetPassword');
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // 绑定第三方账号
 | 
						|
  async function bindThirdOauth() {
 | 
						|
    let result = await sheep.$platform.useProvider('wechat').bind();
 | 
						|
    if (result) {
 | 
						|
      getUserInfo();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // 解绑第三方账号
 | 
						|
  function unBindThirdOauth() {
 | 
						|
    uni.showModal({
 | 
						|
      title: '解绑提醒',
 | 
						|
      content: '解绑后您将无法通过微信登录此账号',
 | 
						|
      cancelText: '再想想',
 | 
						|
      confirmText: '确定',
 | 
						|
      success: async function (res) {
 | 
						|
        if (res.confirm) {
 | 
						|
          const result = await sheep.$platform.useProvider('wechat').unbind();
 | 
						|
          if (result) {
 | 
						|
            getUserInfo();
 | 
						|
          }
 | 
						|
        }
 | 
						|
      },
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  // 保存信息
 | 
						|
  async function onSubmit() {
 | 
						|
    const { error, data } = await sheep.$api.user.update({
 | 
						|
      avatar: state.model.avatar,
 | 
						|
      nickname: state.model.nickname,
 | 
						|
      gender: state.model.gender,
 | 
						|
    });
 | 
						|
    if (error === 0) {
 | 
						|
      getUserInfo();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  const getUserInfo = async () => {
 | 
						|
    const userInfo = await sheep.$store('user').getInfo();
 | 
						|
    state.model = clone(userInfo);
 | 
						|
 | 
						|
    if (sheep.$platform.name !== 'H5') {
 | 
						|
      let { data, error } = await sheep.$api.user.thirdOauthInfo();
 | 
						|
      if (error === 0) {
 | 
						|
        state.thirdOauthInfo = data;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  };
 | 
						|
 | 
						|
  onBeforeMount(async () => {
 | 
						|
    getUserInfo();
 | 
						|
  });
 | 
						|
</script>
 | 
						|
 | 
						|
<style lang="scss" scoped>
 | 
						|
  :deep() {
 | 
						|
    .uni-file-picker {
 | 
						|
      border-radius: 50%;
 | 
						|
    }
 | 
						|
 | 
						|
    .uni-file-picker__container {
 | 
						|
      margin: -14rpx -12rpx;
 | 
						|
    }
 | 
						|
 | 
						|
    .file-picker__progress {
 | 
						|
      height: 0 !important;
 | 
						|
    }
 | 
						|
 | 
						|
    .uni-list-item__content-title {
 | 
						|
      font-size: 28rpx !important;
 | 
						|
      color: #333333 !important;
 | 
						|
      line-height: normal !important;
 | 
						|
    }
 | 
						|
 | 
						|
    .uni-icons {
 | 
						|
      font-size: 40rpx !important;
 | 
						|
    }
 | 
						|
 | 
						|
    .is-disabled {
 | 
						|
      color: #333333;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  :deep(.disabled) {
 | 
						|
    opacity: 1;
 | 
						|
  }
 | 
						|
 | 
						|
  .gender-name {
 | 
						|
    font-size: 28rpx;
 | 
						|
    font-weight: 500;
 | 
						|
    line-height: normal;
 | 
						|
    color: #333333;
 | 
						|
  }
 | 
						|
 | 
						|
  .title-box {
 | 
						|
    font-size: 28rpx;
 | 
						|
    font-weight: 500;
 | 
						|
    color: #666666;
 | 
						|
    line-height: 100rpx;
 | 
						|
  }
 | 
						|
 | 
						|
  .logout-btn {
 | 
						|
    width: 710rpx;
 | 
						|
    height: 80rpx;
 | 
						|
    background: linear-gradient(90deg, var(--ui-BG-Main), var(--ui-BG-Main-gradient));
 | 
						|
    border-radius: 40rpx;
 | 
						|
    font-size: 30rpx;
 | 
						|
    font-weight: 500;
 | 
						|
    color: $white;
 | 
						|
  }
 | 
						|
 | 
						|
  .radio-dark {
 | 
						|
    filter: grayscale(100%);
 | 
						|
    filter: gray;
 | 
						|
    opacity: 0.4;
 | 
						|
  }
 | 
						|
 | 
						|
  .content-img {
 | 
						|
    border-radius: 50%;
 | 
						|
  }
 | 
						|
  .header-box-content {
 | 
						|
    position: relative;
 | 
						|
    width: 160rpx;
 | 
						|
    height: 160rpx;
 | 
						|
    overflow: hidden;
 | 
						|
    border-radius: 50%;
 | 
						|
  }
 | 
						|
  .avatar-action {
 | 
						|
    position: absolute;
 | 
						|
    left: 50%;
 | 
						|
    transform: translateX(-50%);
 | 
						|
    bottom: 0;
 | 
						|
    z-index: 1;
 | 
						|
    width: 160rpx;
 | 
						|
    height: 46rpx;
 | 
						|
    background: rgba(#000000, 0.3);
 | 
						|
 | 
						|
    .avatar-action-btn {
 | 
						|
      width: 160rpx;
 | 
						|
      height: 46rpx;
 | 
						|
      font-weight: 500;
 | 
						|
      font-size: 24rpx;
 | 
						|
      color: #ffffff;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // 绑定项
 | 
						|
  .account-list {
 | 
						|
    background-color: $white;
 | 
						|
    height: 100rpx;
 | 
						|
    padding: 0 20rpx;
 | 
						|
 | 
						|
    .list-img {
 | 
						|
      width: 40rpx;
 | 
						|
      height: 40rpx;
 | 
						|
      margin-right: 10rpx;
 | 
						|
    }
 | 
						|
 | 
						|
    .list-name {
 | 
						|
      font-size: 28rpx;
 | 
						|
      color: #333333;
 | 
						|
    }
 | 
						|
 | 
						|
    .info {
 | 
						|
      .avatar {
 | 
						|
        width: 38rpx;
 | 
						|
        height: 38rpx;
 | 
						|
        border-radius: 50%;
 | 
						|
        overflow: hidden;
 | 
						|
      }
 | 
						|
 | 
						|
      .name {
 | 
						|
        font-size: 28rpx;
 | 
						|
        font-weight: 400;
 | 
						|
        color: $dark-9;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    .bind-box {
 | 
						|
      width: 100rpx;
 | 
						|
      height: 50rpx;
 | 
						|
      line-height: normal;
 | 
						|
      display: flex;
 | 
						|
      justify-content: center;
 | 
						|
      align-items: center;
 | 
						|
      font-size: 24rpx;
 | 
						|
 | 
						|
      .bind-btn {
 | 
						|
        width: 100%;
 | 
						|
        height: 100%;
 | 
						|
        border-radius: 25rpx;
 | 
						|
        background: #f4f4f4;
 | 
						|
        color: #999999;
 | 
						|
      }
 | 
						|
      .relieve-btn {
 | 
						|
        width: 100%;
 | 
						|
        height: 100%;
 | 
						|
        border-radius: 25rpx;
 | 
						|
        background: var(--ui-BG-Main-opacity-1);
 | 
						|
        color: var(--ui-BG-Main);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  .list-border {
 | 
						|
    font-size: 28rpx;
 | 
						|
    font-weight: 400;
 | 
						|
    color: #333333;
 | 
						|
    border-bottom: 2rpx solid #eeeeee;
 | 
						|
  }
 | 
						|
 | 
						|
  image {
 | 
						|
    width: 100%;
 | 
						|
    height: 100%;
 | 
						|
  }
 | 
						|
</style>
 |