BlogPage.vue 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221
  1. <template>
  2. <q-page>
  3. <div
  4. class="row justify-center text-h4 text-weight-bold text-blue-6 q-mt-md"
  5. >
  6. {{ $t("blogs_of_the_week") }}
  7. </div>
  8. <div class="q-pa-md">
  9. <q-carousel
  10. animated
  11. v-model="slide"
  12. navigation
  13. infinite
  14. height="540px"
  15. :autoplay="autoplay"
  16. arrows
  17. transition-prev="slide-right"
  18. transition-next="slide-left"
  19. @mouseenter="autoplay = false"
  20. @mouseleave="autoplay = 2000"
  21. >
  22. <q-carousel-slide
  23. v-for="(item, index) in sliders"
  24. :key="index"
  25. :name="index"
  26. :img-src="item.imageUrl"
  27. class="cursor-pointer"
  28. @click="gotoUrl(item.linkToBlog)"
  29. >
  30. <div class="q-mt-xl q-ml-xl absolute-left custom-caption">
  31. <div style="height: 100px"></div>
  32. <div class="text-h1 text-white text-weight-bolder">
  33. {{ item.title }}
  34. </div>
  35. <div class="text-h5 text-white text-weight-medium">
  36. {{ item.subTitle }}
  37. </div>
  38. </div>
  39. </q-carousel-slide>
  40. </q-carousel>
  41. </div>
  42. <div class="row justify-center q-pa-md">
  43. <q-btn
  44. rounded
  45. color="primary"
  46. size="xl"
  47. @click="goPage('/blog-create')"
  48. :label="$t('write_your_blog')"
  49. />
  50. </div>
  51. <div class="row q-pa-md">
  52. <div class="q-gutter-y-md col-12 col-md-8">
  53. <q-tabs v-model="tab" dense align="justify" class="text-primary">
  54. <q-tab :ripple="false" name="lb" :label="$t('latest_blogs')" />
  55. <q-tab
  56. :ripple="false"
  57. name="mvm"
  58. :label="$t('most_viewed_in_month')"
  59. />
  60. <q-tab
  61. :ripple="false"
  62. name="bfl"
  63. :label="$t('blogers_m_following')"
  64. />
  65. </q-tabs>
  66. </div>
  67. </div>
  68. <div class="row justify-left">
  69. <div
  70. class="col-12 col-sm-4 col-md-2"
  71. v-for="(item, index) in globalUnifiedItemList"
  72. v-bind:key="index"
  73. >
  74. <div class="q-pa-md">
  75. <q-card class="cursor-pointer" flat bordered style="height: 530px">
  76. <q-img
  77. @click="goPage('/blog-detail/' + item.value.blogId)"
  78. :src="item.value.headImageUrl"
  79. style="height: 300px"
  80. placeholder-src="https://photoprism.hichinatravel.com/api/v1/t/2bfc32550ae040956f7e861566d26c487c0143e7/32mcf2k4/tile_224"
  81. />
  82. <q-card-section style="max-height: 200px; overflow: hidden">
  83. <div class="text-overline text-orange-9">
  84. {{ item.value.createdTime }}
  85. </div>
  86. <div class="text-h5 q-mt-sm q-mb-xs">
  87. <a :href="'./blog-detail/' + item.value.blogId">{{
  88. item.value.title
  89. }}</a>
  90. </div>
  91. <div class="text-caption text-grey">
  92. {{ item.value.content }}
  93. </div>
  94. </q-card-section>
  95. </q-card>
  96. </div>
  97. </div>
  98. </div>
  99. <div class="row">
  100. <div class="col-12">
  101. <p class="text-center" style="background-color: #b4b4b4">
  102. Scroll to load more
  103. </p>
  104. </div>
  105. </div>
  106. </q-page>
  107. </template>
  108. <script>
  109. import { ref, onMounted } from "vue";
  110. import { api } from "boot/axios";
  111. import { debounce } from "lodash";
  112. export default {
  113. name: "BlogPage",
  114. setup() {
  115. const unifiedItemList = ref([]);
  116. const blogPageSizePage = ref(20);
  117. const currentPage = ref(1);
  118. const bloglist = ref([]);
  119. const sliders = ref([]);
  120. const totalBlogCount = ref(0);
  121. const globalUnifiedItemList = ref([]);
  122. function loadBlogList() {
  123. unifiedItemList.value = [];
  124. var params = {};
  125. params.pageSize = blogPageSizePage.value;
  126. params.query = "";
  127. params.page = currentPage.value;
  128. api
  129. .get("/api/public/blog/list", { params: params })
  130. .then(function (response) {
  131. bloglist.value = response.data.data.data;
  132. totalBlogCount.value = response.data.data.total;
  133. for (var index in bloglist.value) {
  134. var obj = {};
  135. // obj.type=parseInt(Math.random() * 2)==0?'blog':'scaleblog';
  136. obj.type = "blog";
  137. obj.value = bloglist.value[index];
  138. unifiedItemList.value.push(obj);
  139. }
  140. globalUnifiedItemList.value = globalUnifiedItemList.value.concat(
  141. unifiedItemList.value
  142. );
  143. console.log("globalUnifiedItemList");
  144. console.log(globalUnifiedItemList.value);
  145. })
  146. .catch(function (error) {
  147. console.log(error);
  148. });
  149. }
  150. function loadSliders() {
  151. api
  152. .get("/api/public/pagecontent/bloghomesliders")
  153. .then(function (response) {
  154. console.log("blog sliders:");
  155. sliders.value = response.data.data;
  156. console.log(sliders.value);
  157. loadBlogList();
  158. })
  159. .catch(function (error) {
  160. console.log(error);
  161. });
  162. }
  163. function loadMore() {
  164. currentPage.value += 1;
  165. var maxPage = totalBlogCount.value / blogPageSizePage.value;
  166. if (currentPage.value <= maxPage + 1) {
  167. loadBlogList();
  168. }
  169. }
  170. function getNextBatch() {
  171. window.onscroll = debounce(function () {
  172. let bottomOfWindow =
  173. document.documentElement.scrollTop + window.innerHeight + 100 >
  174. document.documentElement.scrollHeight;
  175. if (bottomOfWindow) {
  176. loadMore();
  177. }
  178. }, 500);
  179. }
  180. function logPv() {
  181. api
  182. .post("/api/public/pagestats/pv/blog")
  183. .then((res) => {
  184. console.log("log pv:");
  185. console.log(res.data);
  186. })
  187. .catch((err) => {
  188. console.error("Error:", err);
  189. });
  190. }
  191. onMounted(() => {
  192. logPv();
  193. loadSliders();
  194. getNextBatch();
  195. });
  196. return {
  197. slide: ref(1),
  198. autoplay: ref(true),
  199. sliders,
  200. tab: ref("lb"),
  201. globalUnifiedItemList,
  202. };
  203. },
  204. };
  205. </script>
  206. <style lang="sass" scoped>
  207. .custom-caption
  208. text-align: left
  209. padding: 12px
  210. color: white
  211. </style>