liu-indexed-list.vue 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395
  1. <template>
  2. <view class="liu-list">
  3. <scroll-view
  4. class="liu-scroll-left"
  5. scroll-y="true"
  6. :scroll-with-animation="true"
  7. :scroll-into-view="scrollIntoView"
  8. >
  9. <!-- <view class="liu-search" id="TOP">
  10. <image class="liu-search-img" src="../../static/search.png"></image>
  11. <input class="liu-input" @input="search" v-model="searchStr" placeholder="请输入搜索信息" maxlength="50"
  12. placeholder-class="liu-placeholder" />
  13. </view> -->
  14. <view
  15. class="left-list"
  16. v-for="(item, index) of scrollLeftObj"
  17. :key="index"
  18. :id="index != '#' ? index : 'BOTTOM'"
  19. >
  20. <view class="left-item-title" v-if="item && item.length">{{ index }}</view>
  21. <view
  22. class="left-item-card"
  23. v-for="(mess, inx) in item"
  24. @click.stop="chooseItem(mess)"
  25. :key="inx"
  26. >
  27. <image
  28. :style="'border-radius:50%'"
  29. class="left-item-card-img img-info"
  30. :src="
  31. mess.avatar
  32. ? mess.avatar
  33. : 'https://web.sdk.qcloud.com/component/TUIKit/assets/avatar_21.png'
  34. "
  35. v-if="mess.avatar"
  36. @click.stop="preview(mess.avatar)"
  37. ></image>
  38. <view :style="'border-radius:' + radius" class="left-item-card-img" v-else>
  39. <!-- {{ (mess.nick && mess.nick.slice(0, 1)) || "" }} -->
  40. <image
  41. :style="'border-radius:50%'"
  42. class="left-item-card-img img-info"
  43. src="https://bucket.sxdirectpurchase.com/wx/static/img/ImAvatar.png"
  44. @click.stop="preview(mess.avatar)"
  45. ></image>
  46. </view>
  47. <view
  48. class="left-item-card-info"
  49. :style="inx < item.length - 1 ? 'border-bottom: solid #F4F4F4 1rpx;' : ''"
  50. >
  51. <view class="left-item-card-name">{{ mess.nick || mess.userID }}</view>
  52. <view class="left-item-card-phone" v-if="mess[phoneKey]">{{
  53. mess[phoneKey]
  54. }}</view>
  55. </view>
  56. </view>
  57. </view>
  58. <view class="no-data" v-if="!hasData">
  59. <image class="no-data-img" src="../../static/noData.png"></image>
  60. <view class="no-data-name">暂无数据</view>
  61. </view>
  62. </scroll-view>
  63. <view class="liu-scroll-right" v-if="hasData">
  64. <image
  65. class="liu-scroll-right-top"
  66. src="../../static/top.png"
  67. @click.stop="scrollIntoView = 'TOP'"
  68. ></image>
  69. <view
  70. :class="{
  71. 'liu-scroll-right-name': true,
  72. 'liu-scroll-right-select': item == scrollIntoView,
  73. }"
  74. v-for="(item, index) in scrollRightList"
  75. :key="index"
  76. @click.stop="chooseType(item)"
  77. >{{ item }}
  78. </view>
  79. </view>
  80. </view>
  81. </template>
  82. <script>
  83. import { pinyinUtil } from "./pinyinUtil.js";
  84. export default {
  85. props: {
  86. //数据源
  87. dataList: {
  88. type: Array,
  89. required: true,
  90. default: () => {
  91. return [];
  92. },
  93. },
  94. //显示的主键key值
  95. idKey: {
  96. type: String,
  97. default: "id",
  98. },
  99. //显示的名字key值
  100. nameKey: {
  101. type: String,
  102. default: "name",
  103. },
  104. //显示的电话key值
  105. phoneKey: {
  106. type: String,
  107. default: "phone",
  108. },
  109. //显示的头像key值
  110. imgKey: {
  111. type: String,
  112. default: "img",
  113. },
  114. //头像圆角(rpx、px、%)
  115. radius: {
  116. type: String,
  117. default: "6rpx",
  118. },
  119. },
  120. data() {
  121. return {
  122. searchStr: "",
  123. scrollIntoView: "",
  124. scrollLeftObj: {},
  125. oldObj: {},
  126. scrollRightList: [],
  127. hasData: true,
  128. };
  129. },
  130. watch: {
  131. dataList: {
  132. immediate: true,
  133. deep: true,
  134. handler(newList) {
  135. if (newList && newList.length) this.cleanData(newList);
  136. },
  137. },
  138. },
  139. methods: {
  140. search() {
  141. if (this.searchStr) {
  142. let has = false;
  143. this.scrollLeftObj = JSON.parse(JSON.stringify(this.oldObj));
  144. for (let i in this.scrollLeftObj) {
  145. this.scrollLeftObj[i] = this.scrollLeftObj[i].filter((item) => {
  146. return (
  147. item[this.nameKey].indexOf(this.searchStr) != -1 ||
  148. item[this.phoneKey].indexOf(this.searchStr) != -1
  149. );
  150. });
  151. if (this.scrollLeftObj[i].length) has = true;
  152. }
  153. if (has) this.hasData = true;
  154. else this.hasData = false;
  155. } else {
  156. this.hasData = true;
  157. this.scrollLeftObj = JSON.parse(JSON.stringify(this.oldObj));
  158. }
  159. },
  160. cleanData(list) {
  161. this.scrollRightList = this.getLetter();
  162. let newList = [];
  163. list.forEach((res) => {
  164. let initial = pinyinUtil.getFirstLetter(res.nick.trim());
  165. let firsfirs = initial ? initial.substring(0, 1) : "";
  166. if (!newList[firsfirs]) newList[firsfirs] = [];
  167. newList[firsfirs].push(res);
  168. console.log(newList[firsfirs]);
  169. });
  170. this.scrollRightList.forEach((t) => {
  171. if (newList[t]) {
  172. this.$set(this.scrollLeftObj, t, newList[t]);
  173. } else {
  174. this.$set(this.scrollLeftObj, t, []);
  175. }
  176. });
  177. let surplusList = [];
  178. for (var i in newList) {
  179. let han = this.scrollRightList.find((v) => {
  180. return v == i;
  181. });
  182. if (!han) surplusList.push(newList[i]);
  183. }
  184. surplusList.forEach((item) => {
  185. this.scrollLeftObj["#"] = this.scrollLeftObj["#"].concat(item);
  186. });
  187. this.oldObj = JSON.parse(JSON.stringify(this.scrollLeftObj));
  188. },
  189. getLetter() {
  190. let list = [];
  191. for (var i = 0; i < 26; i++) {
  192. list.push(String.fromCharCode(65 + i));
  193. }
  194. list.push("#");
  195. return list;
  196. },
  197. chooseType(item) {
  198. if (item == "#") item = "BOTTOM";
  199. this.scrollIntoView = item;
  200. },
  201. preview(img) {
  202. uni.previewImage({
  203. current: 0,
  204. urls: [img],
  205. });
  206. },
  207. chooseItem(item) {
  208. this.$emit("click", item);
  209. },
  210. },
  211. };
  212. </script>
  213. <style>
  214. /deep/ ::-webkit-scrollbar {
  215. width: 0;
  216. height: 0;
  217. color: transparent;
  218. display: none;
  219. }
  220. </style>
  221. <style lang="scss" scoped>
  222. .liu-list {
  223. width: 100%;
  224. max-height: 100vh;
  225. background-color: #f4f4f4;
  226. box-sizing: border-box;
  227. padding-top: 1px;
  228. .liu-scroll-left {
  229. height: 100%;
  230. .liu-search {
  231. width: 100%;
  232. height: 106rpx;
  233. background-color: #ffffff;
  234. display: flex;
  235. align-items: center;
  236. justify-content: center;
  237. position: relative;
  238. .liu-search-img {
  239. width: 32rpx;
  240. height: 32rpx;
  241. position: absolute;
  242. left: 64rpx;
  243. }
  244. .liu-input {
  245. width: calc(100% - 64rpx);
  246. height: 72rpx;
  247. background: #eeeeee;
  248. border-radius: 36rpx;
  249. padding: 0 32rpx 0 80rpx;
  250. box-sizing: border-box;
  251. color: #333333;
  252. }
  253. .liu-placeholder {
  254. color: #777777;
  255. }
  256. }
  257. .left-list {
  258. height: auto;
  259. .left-item-title {
  260. width: calc(100% - 24rpx);
  261. height: 60rpx;
  262. padding-left: 24rpx;
  263. text-align: left;
  264. line-height: 60rpx;
  265. font-size: 30rpx;
  266. color: #666666;
  267. }
  268. .left-item-card {
  269. width: 100%;
  270. height: 112rpx;
  271. background-color: #ffffff;
  272. box-sizing: border-box;
  273. padding-left: 24rpx;
  274. display: flex;
  275. align-items: center;
  276. justify-content: flex-start;
  277. .left-item-card-img {
  278. width: 80rpx;
  279. min-width: 80rpx;
  280. height: 80rpx;
  281. // background-color: #cfcfcf;
  282. display: flex;
  283. align-items: center;
  284. justify-content: center;
  285. font-size: 36rpx;
  286. font-weight: bold;
  287. color: #ffffff;
  288. }
  289. .img-info {
  290. background: none;
  291. border: solid #f0f0f0 1rpx;
  292. }
  293. .left-item-card-info {
  294. width: 100%;
  295. margin-left: 32rpx;
  296. height: 100%;
  297. display: flex;
  298. align-items: flex-start;
  299. justify-content: center;
  300. flex-direction: column;
  301. .left-item-card-name {
  302. font-size: 30rpx;
  303. line-height: 30rpx;
  304. color: #333333;
  305. }
  306. .left-item-card-phone {
  307. margin-top: 14rpx;
  308. font-size: 28rpx;
  309. line-height: 28rpx;
  310. color: #999999;
  311. }
  312. }
  313. }
  314. }
  315. .no-data {
  316. width: 100%;
  317. display: flex;
  318. align-items: center;
  319. justify-items: center;
  320. flex-direction: column;
  321. margin-top: 25%;
  322. .no-data-img {
  323. width: 200rpx;
  324. height: 200rpx;
  325. }
  326. .no-data-name {
  327. margin-top: 20rpx;
  328. font-size: 28rpx;
  329. color: #666666;
  330. }
  331. }
  332. }
  333. .liu-scroll-right {
  334. position: fixed;
  335. right: 0rpx;
  336. top: 50%;
  337. transform: translateY(-47%);
  338. z-index: 999 !important;
  339. display: flex;
  340. align-items: center;
  341. justify-content: center;
  342. flex-direction: column;
  343. .liu-scroll-right-top {
  344. width: 32rpx;
  345. height: 32rpx;
  346. margin-right: 14rpx;
  347. z-index: 999 !important;
  348. }
  349. .liu-scroll-right-name {
  350. width: 32rpx;
  351. padding-right: 14rpx;
  352. height: 28rpx;
  353. font-size: 22rpx;
  354. color: #333333;
  355. line-height: 22rpx;
  356. margin-top: 8rpx;
  357. display: flex;
  358. align-items: center;
  359. justify-content: center;
  360. }
  361. .liu-scroll-right-select {
  362. padding: 0;
  363. margin-right: 14rpx;
  364. width: 28rpx;
  365. height: 28rpx;
  366. border-radius: 50%;
  367. background: #2991ff;
  368. color: #ffffff;
  369. }
  370. }
  371. }
  372. </style>