userInfo.vue 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. <template>
  2. <!-- 个人信息页面 -->
  3. <view class="content">
  4. <view class="user-info">
  5. <!-- 头像 -->
  6. <view class="user-item">
  7. <view class="user-item-label">头像</view>
  8. <view class="user-item-value">
  9. <!-- #ifdef MP-WEIXIN -->
  10. <button class="avatar-btn" open-type="chooseAvatar" @chooseavatar="onChooseAvatar">
  11. <view class="avatar-container">
  12. <image
  13. :src="formData.avatar || getFileUrl('img/personal/userDemo.png')"
  14. :mode="formData.avatar ? 'aspectFill' : 'widthFix'"
  15. class="avatar-img"
  16. />
  17. </view>
  18. </button>
  19. <!-- #endif -->
  20. <!-- #ifndef MP-WEIXIN -->
  21. <view class="avatar-container" @click="handleAvatarClick">
  22. <image
  23. :src="formData.avatar || getFileUrl('img/personal/userDemo.png')"
  24. :mode="formData.avatar ? 'aspectFill' : 'widthFix'"
  25. class="avatar-img"
  26. />
  27. </view>
  28. <!-- #endif -->
  29. </view>
  30. </view>
  31. <!-- 昵称:微信端 type="nickname" 可一键填入微信昵称 -->
  32. <view class="user-item">
  33. <view class="user-item-label">昵称</view>
  34. <view class="user-item-value">
  35. <input
  36. v-model="formData.nickname"
  37. type="nickname"
  38. class="user-input"
  39. placeholder="请输入昵称,可点击使用微信昵称"
  40. placeholder-style="color: #999;"
  41. />
  42. </view>
  43. </view>
  44. <!-- 性别 -->
  45. <view class="user-item">
  46. <view class="user-item-label">性别</view>
  47. <view class="user-item-value">
  48. <view class="gender-group">
  49. <view
  50. v-for="item in genderOptions"
  51. :key="item.value"
  52. class="gender-item"
  53. :class="{ active: formData.gender === item.value }"
  54. hover-class="none"
  55. @click="formData.gender = item.value"
  56. >
  57. <image
  58. :src="getFileUrl(formData.gender === item.value ? 'img/personal/sele2.png' : 'img/personal/sele1.png')"
  59. mode="widthFix"
  60. class="gender-img"
  61. />
  62. <text class="gender-text">{{ item.label }}</text>
  63. </view>
  64. </view>
  65. </view>
  66. </view>
  67. <!-- 生日 -->
  68. <view class="user-item">
  69. <view class="user-item-label">生日</view>
  70. <view class="user-item-value">
  71. <picker mode="date" :value="formData.birthday" @change="handleBirthdayChange"
  72. class="picker-wrapper">
  73. <view class="picker-content">
  74. <text :class="['picker-text', { placeholder: !formData.birthday }]">
  75. {{ formData.birthday || '请选择生日' }}
  76. </text>
  77. <view class="arrow-icon"></view>
  78. </view>
  79. </picker>
  80. </view>
  81. </view>
  82. <!-- 手机号码 -->
  83. <view class="user-item">
  84. <view class="user-item-label">手机号码</view>
  85. <view class="user-item-value">
  86. <view class="phone-wrapper">
  87. <text class="phone-number">{{ formData.phone || '1510987760' }}</text>
  88. <text class="change-btn" @click="handleChangePhone">更换</text>
  89. </view>
  90. </view>
  91. </view>
  92. </view>
  93. </view>
  94. </template>
  95. <script setup>
  96. import { ref, onMounted } from 'vue';
  97. import { useUserStore } from '@/store/user.js';
  98. import { getFileUrl } from '@/utils/file.js';
  99. import { go } from '@/utils/utils.js';
  100. const userStore = useUserStore();
  101. // 性别选项
  102. const genderOptions = [
  103. { value: 'male', label: '男' },
  104. { value: 'female', label: '女' }
  105. ];
  106. // 表单数据
  107. const formData = ref({
  108. avatar: '',
  109. nickname: '',
  110. gender: 'male', // 默认选中男
  111. birthday: '',
  112. phone: '1510987760'
  113. });
  114. // 初始化用户信息
  115. onMounted(() => {
  116. const userInfo = userStore.getUserInfo;
  117. if (userInfo) {
  118. formData.value = {
  119. avatar: userInfo.avatar || '',
  120. nickname: userInfo.nickname || '',
  121. gender: userInfo.gender || 'male',
  122. birthday: userInfo.birthday || '',
  123. phone: userInfo.phone || '1510987760'
  124. };
  125. }
  126. });
  127. // 处理头像点击
  128. const handleAvatarClick = () => {
  129. uni.showActionSheet({
  130. itemList: ['微信头像', '从相册选择', '拍照'],
  131. success: (res) => {
  132. if (res.tapIndex === 0) {
  133. // 选择微信头像
  134. handleWechatAvatar();
  135. } else if (res.tapIndex === 1) {
  136. // 选择相册
  137. handleChooseAlbum();
  138. } else if (res.tapIndex === 2) {
  139. // 选择拍照
  140. handleTakePhoto();
  141. }
  142. },
  143. fail: (err) => {
  144. console.log('取消选择', err);
  145. }
  146. });
  147. };
  148. // 微信小程序:通过「头像昵称填写」能力选择头像(button open-type="chooseAvatar" 触发)
  149. // 注意:e.detail.avatarUrl 为临时路径,需上传到服务器后保存永久链接
  150. const onChooseAvatar = (e) => {
  151. if (e.detail && e.detail.avatarUrl) {
  152. formData.value.avatar = e.detail.avatarUrl;
  153. uni.showToast({ title: '头像已选择', icon: 'success' });
  154. }
  155. };
  156. // 获取微信头像(仅非 MP-WEIXIN 时从 actionSheet 进入;MP-WEIXIN 已改用模板内 button chooseAvatar)
  157. const handleWechatAvatar = () => {
  158. // #ifdef MP-WEIXIN
  159. // 微信已废弃 getUserProfile,请直接点击页面头像区域,使用「选择头像」按钮
  160. uni.showToast({
  161. title: '请点击上方头像区域选择',
  162. icon: 'none'
  163. });
  164. // #endif
  165. // #ifndef MP-WEIXIN
  166. uni.showToast({
  167. title: '当前环境不支持',
  168. icon: 'none'
  169. });
  170. // #endif
  171. };
  172. // 选择相册
  173. const handleChooseAlbum = () => {
  174. uni.chooseImage({
  175. count: 1,
  176. sizeType: ['original', 'compressed'],
  177. sourceType: ['album'],
  178. success: (res) => {
  179. const tempFilePath = res.tempFilePaths[0];
  180. // 这里可以调用上传接口
  181. // 暂时直接使用本地路径
  182. formData.value.avatar = tempFilePath;
  183. },
  184. fail: (err) => {
  185. console.log('选择相册失败', err);
  186. }
  187. });
  188. };
  189. // 拍照
  190. const handleTakePhoto = () => {
  191. uni.chooseImage({
  192. count: 1,
  193. sizeType: ['original', 'compressed'],
  194. sourceType: ['camera'],
  195. success: (res) => {
  196. const tempFilePath = res.tempFilePaths[0];
  197. // 这里可以调用上传接口
  198. // 暂时直接使用本地路径
  199. formData.value.avatar = tempFilePath;
  200. },
  201. fail: (err) => {
  202. console.log('拍照失败', err);
  203. }
  204. });
  205. };
  206. // 处理生日选择
  207. const handleBirthdayChange = (e) => {
  208. formData.value.birthday = e.detail.value;
  209. };
  210. // 处理更换手机号
  211. const handleChangePhone = () => {
  212. go('/pages/personal/setPhone');
  213. };
  214. </script>
  215. <style scoped lang="scss">
  216. .content {
  217. width: 100%;
  218. min-height: 100vh;
  219. background-color: #F5F5F5;
  220. box-sizing: border-box;
  221. padding: 20rpx;
  222. }
  223. .user-info {
  224. width: 100%;
  225. border-radius: 23rpx;
  226. background-color: #fff;
  227. box-sizing: border-box;
  228. padding: 0 30rpx;
  229. .user-item {
  230. display: flex;
  231. align-items: center;
  232. justify-content: space-between;
  233. padding: 30rpx 0;
  234. border-bottom: 1rpx solid #F5F5F5;
  235. &:last-child {
  236. border-bottom: none;
  237. }
  238. .user-item-label {
  239. font-size: 28rpx;
  240. color: #333;
  241. font-weight: 400;
  242. }
  243. .user-item-value {
  244. display: flex;
  245. align-items: center;
  246. justify-content: flex-end;
  247. flex: 1;
  248. }
  249. }
  250. }
  251. // 头像样式
  252. .avatar-btn {
  253. padding: 0;
  254. margin: 0;
  255. background: transparent;
  256. border: none;
  257. line-height: 1;
  258. &::after {
  259. border: none;
  260. }
  261. }
  262. .avatar-container {
  263. width: 76rpx;
  264. height: 76rpx;
  265. border-radius: 8rpx;
  266. display: flex;
  267. align-items: center;
  268. justify-content: center;
  269. position: relative;
  270. overflow: hidden;
  271. box-sizing: border-box;
  272. .avatar-img {
  273. width: 100%;
  274. height: 100%;
  275. border-radius: 50%;
  276. }
  277. }
  278. // 输入框样式
  279. .user-input {
  280. flex: 1;
  281. text-align: right;
  282. font-size: 28rpx;
  283. color: #333;
  284. }
  285. // 性别选择样式
  286. .gender-group {
  287. display: flex;
  288. align-items: center;
  289. gap: 40rpx;
  290. .gender-item {
  291. display: flex;
  292. align-items: center;
  293. gap: 12rpx;
  294. cursor: pointer;
  295. .gender-text {
  296. font-size: 28rpx;
  297. color: #333;
  298. }
  299. &.active {
  300. .gender-text {
  301. color: #FF6B35;
  302. }
  303. }
  304. }
  305. }
  306. // 日期选择器样式
  307. .picker-wrapper {
  308. width: 100%;
  309. display: flex;
  310. justify-content: flex-end;
  311. }
  312. .picker-content {
  313. display: flex;
  314. align-items: center;
  315. gap: 12rpx;
  316. .picker-text {
  317. font-size: 28rpx;
  318. color: #333;
  319. &.placeholder {
  320. color: #999;
  321. }
  322. }
  323. .arrow-icon {
  324. width: 12rpx;
  325. height: 12rpx;
  326. border-right: 2rpx solid #999;
  327. border-top: 2rpx solid #999;
  328. transform: rotate(45deg);
  329. margin-right: 4rpx;
  330. }
  331. }
  332. // 手机号样式
  333. .phone-wrapper {
  334. display: flex;
  335. align-items: center;
  336. gap: 20rpx;
  337. .phone-number {
  338. font-size: 28rpx;
  339. color: #333;
  340. }
  341. .change-btn {
  342. font-size: 28rpx;
  343. color: #F47D1F;
  344. cursor: pointer;
  345. }
  346. }
  347. .gender-img {
  348. width: 26rpx;
  349. height: 26rpx;
  350. }
  351. </style>