index.vue 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. <script setup name="IndexSection">
  2. // import { type } from "os";
  3. import SvgIcon from "~/components/SvgIcon/index.vue";
  4. import * as formatTime from "~/utils/formatTime";
  5. const handleTimeAgo = formatTime.timeAgo;
  6. const Props = defineProps({
  7. type: {
  8. type: String,
  9. validator: (key) => ["article", "common"].includes(key),
  10. default: "common",
  11. },
  12. // 没有图片的就是 `article` 类型
  13. image: String,
  14. title: String,
  15. timeAgo: String,
  16. descs: String,
  17. topicName: String,
  18. author: String,
  19. authorAvatar: {
  20. type: String,
  21. default: "", // TODO: 理应存在默认头像
  22. },
  23. view_count: String, //浏览量
  24. comment_count: String, //评论数
  25. like_count: String, //点赞数
  26. collect_count: String, //收藏数
  27. share_count: String, //分享数
  28. });
  29. const emit = defineEmits(["todetails"]);
  30. const todetails = () => {
  31. emit("update");
  32. };
  33. const shareLink = () => {};
  34. </script>
  35. <template>
  36. <div class="index-section-container" @click="todetails">
  37. <div v-if="type === 'common'" class="imgbox">
  38. <img :src="image" style="width: 279px; height: 155px;object-fit: cover;" alt="" />
  39. </div>
  40. <div v-else class="article-wrapper">
  41. <div class="title ellipsis-two">{{ title }}</div>
  42. <p class="desc">{{ descs }}</p>
  43. </div>
  44. <div class="content">
  45. <div v-if="type === 'common'" class="title ellipsis-two">
  46. {{ title }}
  47. </div>
  48. <!-- TODO: 话题是否必填 v-show="topicName" -->
  49. <div class="flag" :style="{ opacity: topicName ? '1' : '0' }">
  50. #{{ topicName }}
  51. </div>
  52. <div class="moment-user flex-row flex-aic flex-jc-sb">
  53. <div class="left flex-row flex-aic">
  54. <div class="avatar">
  55. <img :src="authorAvatar" :alt="author" />
  56. </div>
  57. <div class="nickname ellipsis">{{ author }}</div>
  58. </div>
  59. <div class="ago">{{ handleTimeAgo(timeAgo) }}</div>
  60. </div>
  61. <div class="footer">
  62. <ul class="flex-row flex-aic flex-jc-sb">
  63. <li class="active">
  64. <SvgIcon name="like" :size="16" :rgap="2" color="#00b0b0" />
  65. <span>{{ view_count || 0 }}</span>
  66. </li>
  67. <li class="active">
  68. <SvgIcon name="collect" :size="16" :rgap="2" color="#00b0b0" />
  69. <span>{{ comment_count || 0 }}</span>
  70. </li>
  71. <li>
  72. <SvgIcon name="message" :size="16" :rgap="2" color="#999999" />
  73. <span>{{ collect_count || 0 }}</span>
  74. </li>
  75. <li>
  76. <SvgIcon name="forward" :size="16" :rgap="2" color="#999999" />
  77. <span>{{ share_count || 0 }}</span>
  78. </li>
  79. </ul>
  80. </div>
  81. </div>
  82. </div>
  83. </template>
  84. <style lang="scss" scoped>
  85. .index-section {
  86. &-container {
  87. width: 275px;
  88. height: 337px;
  89. background-color: #fff;
  90. border-radius: 6px;
  91. overflow: hidden;
  92. .imgbox {
  93. width: 279px;
  94. height: 155px;
  95. background: #eee;
  96. border-radius: 6px 6px 0px 0px;
  97. }
  98. .article-wrapper {
  99. padding: 14px 10px 0;
  100. height: 201px;
  101. /* height: 187px; */
  102. .title {
  103. margin-bottom: 16px;
  104. }
  105. .desc {
  106. margin: 0;
  107. padding: 0;
  108. font-size: 14px;
  109. height: calc(100% - 60px);
  110. line-height: 1.44;
  111. overflow: hidden;
  112. }
  113. }
  114. .title {
  115. width: calc(279px - 20px);
  116. height: 44px;
  117. font-size: 16px;
  118. font-family: PingFangSC, PingFang SC;
  119. font-weight: 600;
  120. color: #333333;
  121. line-height: 22px;
  122. margin-bottom: 16px;
  123. }
  124. .content {
  125. padding: 16px 10px;
  126. .flag {
  127. height: 30px;
  128. line-height: 30px;
  129. font-size: 13px;
  130. font-weight: 400;
  131. color: #ff6700;
  132. }
  133. .moment-user {
  134. height: 30px;
  135. .left {
  136. width: 0;
  137. flex: 1;
  138. }
  139. .avatar {
  140. width: 20px;
  141. height: 20px;
  142. border-radius: 50%;
  143. overflow: hidden;
  144. }
  145. .nickname {
  146. flex: 1;
  147. padding: 0 10px 0 8px;
  148. font-size: 12px;
  149. font-weight: 400;
  150. color: #444444;
  151. }
  152. .ago {
  153. height: 17px;
  154. font-size: 12px;
  155. font-family: PingFangSC, PingFang SC;
  156. font-weight: 400;
  157. color: #888888;
  158. line-height: 17px;
  159. }
  160. }
  161. .footer {
  162. height: 30px;
  163. font-size: 14px;
  164. font-weight: 400;
  165. color: #999999;
  166. line-height: 30px;
  167. ul {
  168. padding: 0;
  169. margin: 0;
  170. list-style: none;
  171. li {
  172. cursor: pointer;
  173. display: flex;
  174. align-items: center;
  175. justify-content: center;
  176. &.active {
  177. color: #00b0b0;
  178. }
  179. }
  180. }
  181. }
  182. }
  183. }
  184. }
  185. </style>