DestinationDetailPage.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. <template>
  2. <q-page>
  3. <div class="row" style="height: 110px; background-color: #e5f2fa">
  4. <div class="col-12 col-sm-10 column" style="margin-left: 20px">
  5. <div class="q-pt-md q-pl-md text-black text-weight-bold">
  6. Home > Destination > {{ destinationName }}
  7. </div>
  8. <div class="col text-weight-bold q-pl-md text-h3">
  9. {{ destinationName }}
  10. </div>
  11. </div>
  12. </div>
  13. <div class="row">
  14. <!-- Left Column for Image -->
  15. <div
  16. class="col-12 col-sm-5 row q-px-md q-py-lg d-flex justify-center align-center"
  17. style="height: 450px; background-color: white"
  18. >
  19. <q-img
  20. :src="destinationProfileImage"
  21. style="width: 100%; max-width: 650px; height: 100%"
  22. fit="cover"
  23. >
  24. </q-img>
  25. </div>
  26. <!-- Right Column for Description, Guidebook, and Download Button -->
  27. <div class="col-12 col-sm-7 row q-px-md q-py-lg" style="height: 450px">
  28. <div
  29. class="column col-12 d-flex flex-column justify-between"
  30. style="background-color: #eff6fd; height: 100%"
  31. >
  32. <div class="q-pa-md text-subtitle1">
  33. <div v-html="description"></div>
  34. </div>
  35. <div class="q-pa-md text-h4" style="margin-top: auto">
  36. Guidebook of {{ destinationName }}
  37. <q-btn class="primary" color="primary" @click="goDownload">
  38. <q-icon left size="3em" name="download" />
  39. <div>Download</div>
  40. </q-btn>
  41. </div>
  42. </div>
  43. </div>
  44. </div>
  45. <div
  46. class="row flex flex-left text-h4 text-weight-bold"
  47. style="
  48. background-color: #eff6fd;
  49. border-top: 1px solid gray;
  50. height: 55px;
  51. margin-left: 30px;
  52. "
  53. >
  54. Places to go in {{ destinationName }}
  55. </div>
  56. <div
  57. v-for="item in childDestinations"
  58. v-bind:key="item.destinationId"
  59. class="row q-pa-md"
  60. style="width: 90%; min-height: 200px; margin: 0 auto"
  61. >
  62. <div class="col-3 q-pa-md" style="border-right: 1px solid gray">
  63. <q-img
  64. class="cursor-pointer"
  65. @click="goPage('/destination-detail/' + item.destinationId)"
  66. fit="fill"
  67. :src="normalizeMultiImageUrl(item.destinationProfileImage)"
  68. ></q-img>
  69. </div>
  70. <div class="col-9 column q-pa-md">
  71. <div class="text-h4 text-weight-bold" style="background-color: #f3f6f8">
  72. <a :href="'./destination-detail/' + item.destinationId">{{
  73. item.destinationName
  74. }}</a>
  75. </div>
  76. <div class="q-pt-md">
  77. {{ removeHtmlTag(item.description) }}
  78. </div>
  79. </div>
  80. </div>
  81. </q-page>
  82. </template>
  83. <script>
  84. import { ref, onMounted, getCurrentInstance } from "vue";
  85. import { api } from "boot/axios";
  86. import { useRoute } from "vue-router";
  87. import { useSeoMeta } from "unhead";
  88. import { useQuasar } from "quasar";
  89. export default {
  90. name: "DestinationDetailPage",
  91. setup() {
  92. const instance = getCurrentInstance();
  93. const app = getCurrentInstance().appContext.app;
  94. const gp = app.config.globalProperties;
  95. const $q = useQuasar();
  96. const route = useRoute();
  97. const description = ref("");
  98. const destinationProfileImage = ref("");
  99. const downloadUrl = ref("");
  100. const destinationName = ref("");
  101. const childDestinations = ref([]);
  102. const relevantToursProduct = ref([]);
  103. function goDownload() {
  104. if (downloadUrl.value == null || downloadUrl.value == "") {
  105. gp.$generalNotify(
  106. $q,
  107. false,
  108. "There is no guidebook for this destination"
  109. );
  110. return;
  111. } else {
  112. window.location.href = downloadUrl.value;
  113. }
  114. }
  115. function logPv() {
  116. api
  117. .post(
  118. "/api/public/pagestats/pv/destination-detail-" +
  119. route.params.destinationId
  120. )
  121. .then((res) => {
  122. console.log("log pv:");
  123. console.log(res.data);
  124. })
  125. .catch((err) => {
  126. console.error("Error:", err);
  127. });
  128. }
  129. function logView() {
  130. api
  131. .post(
  132. "/api/public/pagestats/view-destination/" + route.params.destinationId
  133. )
  134. .then((res) => {
  135. console.log("view cnt of this destination:");
  136. console.log(res.data);
  137. })
  138. .catch((err) => {
  139. console.error("Error:", err);
  140. });
  141. }
  142. function loadRelatedTours() {
  143. api
  144. .get(
  145. "/api/public/destination/relavanttourproduct/" +
  146. route.params.destinationId
  147. )
  148. .then((response) => {
  149. if (response.data.ok === true) {
  150. relevantToursProduct.value = response.data.data;
  151. } else {
  152. relevantToursProduct.value = [];
  153. }
  154. })
  155. .catch((e) => {
  156. console.log("get relevant tours product err");
  157. console.log(e);
  158. });
  159. }
  160. function loadChildrenDestination() {
  161. api
  162. .get("/api/public/destination/children/" + route.params.destinationId)
  163. .then((response) => {
  164. childDestinations.value = response.data.data;
  165. })
  166. .catch((e) => {
  167. console.log("get destination detail err");
  168. console.log(e);
  169. });
  170. }
  171. function loadDestinations() {
  172. api
  173. .get("/api/public/destination/" + route.params.destinationId)
  174. .then((response) => {
  175. description.value = response.data.data.description;
  176. destinationProfileImage.value = gp.$normalizeMultiImageUrl(
  177. response.data.data.destinationProfileImage
  178. );
  179. console.log(destinationProfileImage.value);
  180. downloadUrl.value = response.data.data.downloadUrl;
  181. destinationName.value = response.data.data.destinationName;
  182. useSeoMeta({
  183. title: destinationName.value,
  184. description: description.value,
  185. ogDescription: description.value,
  186. ogTitle: destinationName.value,
  187. ogImage:
  188. "https://www.hichinatravel.com/static/png/name-67280b81.png",
  189. twitterCard: "summary_large_image",
  190. });
  191. })
  192. .catch((e) => {
  193. console.log("get destination detail err");
  194. console.log(e);
  195. });
  196. }
  197. onMounted(() => {
  198. logPv();
  199. logView();
  200. console.log("on mounted destination detail page");
  201. loadDestinations();
  202. loadChildrenDestination();
  203. loadRelatedTours();
  204. });
  205. return {
  206. destinationName,
  207. destinationProfileImage,
  208. description,
  209. childDestinations,
  210. goDownload,
  211. };
  212. },
  213. };
  214. </script>