IndexPage.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  1. <template>
  2. <q-page>
  3. <div class="q-pa-md">
  4. <q-carousel
  5. animated
  6. v-model="slide"
  7. navigation
  8. infinite
  9. height="540px"
  10. :autoplay="2000"
  11. arrows
  12. transition-prev="slide-right"
  13. transition-next="slide-left"
  14. @mouseenter="autoplay = false"
  15. @mouseleave="autoplay = true"
  16. >
  17. <q-carousel-slide
  18. v-for="(item, index) in sliders"
  19. :key="index"
  20. :name="index"
  21. :img-src="item.imageUrl"
  22. class="cursor-pointer"
  23. @click="gotoUrl(item.linkToBlog)"
  24. >
  25. <div class="q-mt-xl q-ml-xl absolute-left custom-caption">
  26. <div style="height: 100px"></div>
  27. <div class="text-h1 text-weight-bolder">{{ item.title }}</div>
  28. <div class="text-h5 text-weight-medium">{{ item.subTitle }}</div>
  29. </div>
  30. </q-carousel-slide>
  31. </q-carousel>
  32. </div>
  33. <div class="q-pa-md" style="height: 40px"></div>
  34. <div class="row">
  35. <div class="col-6 col-md-2 text-blue-6 text-h4 q-pl-md">
  36. {{ $t("destinations") }}
  37. </div>
  38. <div class="col-md-8"></div>
  39. <div
  40. @click="goPage('/destination')"
  41. class="col-6 col-md-2 text-blue-6 cursor-pointer"
  42. >
  43. {{ $t("more_destinations") }} >
  44. </div>
  45. </div>
  46. <div class="row">
  47. <div class="text-h3 q-pl-md q-mt-md">
  48. {{ $t("places_in_china") + "!" }}
  49. </div>
  50. </div>
  51. <div class="row justify-center">
  52. <div
  53. class="col-12 col-sm-4 col-md-2"
  54. v-for="item in randDestinations"
  55. :key="item.blogId"
  56. >
  57. <div class="q-pa-md">
  58. <q-card
  59. @click="goPage('/destination-detail/' + item.destinationId)"
  60. class="rand-destination-card cursor-pointer rounded-borders"
  61. @mouseenter="hoverFlag = true"
  62. @mouseleave="hoverFlag = false"
  63. >
  64. <q-img
  65. class="rounded-borders"
  66. style="height: 100%"
  67. :src="normalizeMultiImageUrl(item.destinationProfileImage)"
  68. >
  69. <div class="absolute-bottom" v-if="hoverFlag">
  70. <div class="text-h6">{{ item.destinationName }}</div>
  71. <div class="text-subtitle2">
  72. {{ item.parentDestinationName }}
  73. </div>
  74. </div>
  75. </q-img>
  76. </q-card>
  77. </div>
  78. </div>
  79. </div>
  80. <div class="q-pa-md">
  81. <div class="row">
  82. <div
  83. v-if="$q.screen.gt.xs"
  84. @click="goPage('/contact')"
  85. class="cursor-pointer col-12 q-mt-md q-pt-md rounded-borders text-white text-weight-bold text-h5 text-no-wrap text-center"
  86. style="
  87. background-color: #2a82e4;
  88. height: 80px;
  89. border-radius: 20px;
  90. border: 1px solid black;
  91. "
  92. >
  93. {{ $t("tailor_made_trip") }}
  94. </div>
  95. <div
  96. v-if="!$q.screen.gt.xs"
  97. @click="goPage('/contact')"
  98. class="cursor-pointer col-12 q-mt-md q-pt-md rounded-borders text-white text-weight-bold text-subtitle1 text-no-wrap text-center"
  99. style="
  100. background-color: #2a82e4;
  101. height: 80px;
  102. border-radius: 20px;
  103. border: 1px solid black;
  104. "
  105. >
  106. Want a Tailor Made Trip to China? Click Here for Help
  107. </div>
  108. </div>
  109. </div>
  110. <div class="q-pa-md">
  111. <div class="row">
  112. <div
  113. class="col-12 cursor-pointer"
  114. style="height: 160px"
  115. @click="goPage('/contact')"
  116. >
  117. <q-img :src="homePostImageUrl" fit="fill" style="height: 100%">
  118. </q-img>
  119. </div>
  120. </div>
  121. </div>
  122. <div class="row">
  123. <div class="text-h5 text-weight-bold text-blue-6 q-pl-md q-mt-md">
  124. {{ $t("people_are_traveling") + "!" }}
  125. </div>
  126. </div>
  127. <div class="row">
  128. <div class="text-h3 text-weight-medium q-pl-md q-mt-md">
  129. {{ $t("whats_in_china") + "!" }}
  130. </div>
  131. </div>
  132. <div class="row justify-left">
  133. <div
  134. class="col-12 col-sm-4 col-md-2"
  135. v-for="(item, index) in globalUnifiedItemList"
  136. v-bind:key="index"
  137. >
  138. <div class="q-pa-md">
  139. <q-card
  140. v-if="item.type === 'blog'"
  141. class="cursor-pointer"
  142. flat
  143. bordered
  144. style="height: 530px"
  145. >
  146. <q-img
  147. @click="goPage('/blog-detail/' + item.value.blogId)"
  148. :src="item.value.headImageUrl"
  149. placeholder-src="https://photoprism.hichinatravel.com/api/v1/t/2bfc32550ae040956f7e861566d26c487c0143e7/32mcf2k4/tile_224"
  150. style="height: 300px"
  151. />
  152. <q-card-section style="max-height: 200px; overflow: hidden">
  153. <div class="text-overline text-orange-9">
  154. {{ item.value.createdTime }}
  155. </div>
  156. <div class="text-h5 q-mt-sm q-mb-xs">
  157. <a :href="'./blog-detail/' + item.value.blogId">{{
  158. item.value.title
  159. }}</a>
  160. </div>
  161. <div class="text-caption text-grey">
  162. <div v-if="!showFullContent">
  163. {{ item.value.content.slice(0, 200) }}
  164. <!-- Display the first 200 characters -->
  165. >
  166. </div>
  167. <div v-else>
  168. {{ item.value.content }}
  169. </div>
  170. </div>
  171. </q-card-section>
  172. </q-card>
  173. <q-card
  174. v-if="item.type === 'product'"
  175. class="cursor-pointer"
  176. flat
  177. bordered
  178. style="height: 400px"
  179. >
  180. <q-img
  181. @click="goPage('/product-detail/' + item.value.skuGroupId)"
  182. :src="item.value.imageUrl"
  183. placeholder-src="https://photoprism.hichinatravel.com/api/v1/t/2bfc32550ae040956f7e861566d26c487c0143e7/32mcf2k4/tile_224"
  184. style="height: 220px"
  185. />
  186. <q-card-section>
  187. <div class="text-subtitle1 text-pink">
  188. Start from ¥{{ item.value.minPrice }}
  189. </div>
  190. <div class="text-h5 q-mt-sm q-mb-xs">
  191. {{ item.value.skuGroupName }}
  192. </div>
  193. </q-card-section>
  194. </q-card>
  195. </div>
  196. </div>
  197. </div>
  198. <div class="row">
  199. <div class="col-12">
  200. <p class="text-center" style="background-color: #b4b4b4">
  201. Scroll to load more
  202. </p>
  203. </div>
  204. </div>
  205. </q-page>
  206. </template>
  207. <script>
  208. import { ref, onMounted } from "vue";
  209. import { defineComponent } from "vue";
  210. import { debounce } from "lodash";
  211. import { api } from "boot/axios";
  212. export default defineComponent({
  213. name: "IndexPage",
  214. setup() {
  215. const sliders = ref([]);
  216. const randDestinations = ref([]);
  217. const hoverFlag = ref(false);
  218. const homePostLink = ref("");
  219. const homePostImageUrl = ref("");
  220. const currentPage = ref(1);
  221. const totalBlogCount = ref(-1);
  222. const blogPageSize = ref(30);
  223. const productPageSizePage = ref(5);
  224. const loading = ref(false);
  225. const bloglist = ref([]);
  226. const productlist = ref([]);
  227. const globalUnifiedItemList = ref([]);
  228. const unifiedItemList = ref([]);
  229. // function goToBlog(url) {
  230. // window.location.href = url;
  231. // }
  232. // function normalizeMultiImageUrl(input) {
  233. // if (input.indexOf(",") > -1) {
  234. // return input.split(",").shift();
  235. // } else if (input.indexOf(";") > -1) {
  236. // return input.split(";").shift();
  237. // }
  238. // return input;
  239. // }
  240. function loadRand6Destinations() {
  241. api
  242. .get("/api/public/destination/rand6")
  243. .then((response) => {
  244. randDestinations.value = response.data.data;
  245. })
  246. .catch((e) => {
  247. alert(e);
  248. });
  249. }
  250. function loadHomeSliders() {
  251. api
  252. .get("/api/public/pagecontent/homesliders")
  253. .then((response) => {
  254. sliders.value = response.data.data;
  255. loadBlogList();
  256. })
  257. .catch((e) => {
  258. console.log(e);
  259. });
  260. }
  261. function loadHomePost() {
  262. api
  263. .get("/api/public/pagecontent/homepost")
  264. .then((response) => {
  265. homePostLink.value = response.data.data.postLink;
  266. homePostImageUrl.value = response.data.data.postImageUrl;
  267. })
  268. .catch((e) => {
  269. console.log(e);
  270. });
  271. }
  272. function loadProducts() {
  273. var params = {};
  274. params.pageSize = productPageSizePage.value;
  275. params.query = "";
  276. params.page = currentPage.value;
  277. params.productTypeId = "";
  278. api
  279. .get("/api/public/productsku/productskugrouplist", { params: params })
  280. .then(function (response) {
  281. productlist.value = response.data.data.data;
  282. for (var index in productlist.value) {
  283. var obj = {};
  284. obj.type = "product";
  285. obj.value = productlist.value[index];
  286. unifiedItemList.value.push(obj);
  287. }
  288. globalUnifiedItemList.value = globalUnifiedItemList.value.concat(
  289. unifiedItemList.value
  290. );
  291. loading.value = false;
  292. })
  293. .catch(function (error) {
  294. console.log(error);
  295. });
  296. }
  297. function loadBlogList() {
  298. loading.value = true;
  299. unifiedItemList.value = [];
  300. var params = {};
  301. params.pageSize = blogPageSize.value;
  302. params.query = "";
  303. console.log("loading page: " + currentPage.value);
  304. params.page = currentPage.value;
  305. api
  306. .get("/api/public/blog/list", { params: params })
  307. .then(function (response) {
  308. if (totalBlogCount.value == -1) {
  309. totalBlogCount.value = response.data.data.total;
  310. }
  311. bloglist.value = response.data.data.data;
  312. for (var index in bloglist.value) {
  313. var obj = {};
  314. // obj.type=parseInt(Math.random() * 2)==0?'blog':'scaleblog';
  315. obj.type = "blog";
  316. obj.value = bloglist.value[index];
  317. unifiedItemList.value.push(obj);
  318. }
  319. loadProducts();
  320. })
  321. .catch(function (error) {
  322. console.log(error);
  323. });
  324. }
  325. function loadMore() {
  326. currentPage.value += 1;
  327. var maxPage = totalBlogCount.value / blogPageSize.value;
  328. if (currentPage.value <= maxPage + 1) {
  329. loadBlogList();
  330. }
  331. }
  332. function getNextBatch() {
  333. window.onscroll = debounce(function () {
  334. let bottomOfWindow =
  335. document.documentElement.scrollTop + window.innerHeight + 100 >
  336. document.documentElement.scrollHeight;
  337. if (bottomOfWindow) {
  338. loadMore();
  339. }
  340. }, 500);
  341. }
  342. function logPv() {
  343. api
  344. .post("/api/public/pagestats/pv/home")
  345. .then((res) => {
  346. console.log("log pv:");
  347. console.log(res.data);
  348. })
  349. .catch((err) => {
  350. console.error("Error:", err);
  351. });
  352. }
  353. onMounted(() => {
  354. logPv();
  355. // we call "next()" method of our component
  356. loadHomeSliders();
  357. loadRand6Destinations();
  358. loadHomePost();
  359. getNextBatch();
  360. });
  361. return {
  362. slide: ref(0),
  363. autoplay: ref(true),
  364. sliders,
  365. randDestinations,
  366. hoverFlag,
  367. homePostLink,
  368. homePostImageUrl,
  369. globalUnifiedItemList,
  370. };
  371. },
  372. });
  373. </script>
  374. <style lang="sass" scoped>
  375. .custom-caption
  376. text-align: left
  377. padding: 12px
  378. color: white
  379. .rand-destination-card
  380. width: 100%
  381. max-width: 550px
  382. height: 300px
  383. </style>