ProductDetailPage.vue 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823
  1. <template>
  2. <q-page>
  3. <div class="row" style="background-color: #e5f2fa; height: 50px">
  4. <div class="col-sm-3"></div>
  5. <div
  6. class="col-12 col-sm-9 text-body1 q-pl-md flex"
  7. style="align-items: center"
  8. >
  9. Travel shop > {{ productTypeName }} > {{ productName }}
  10. </div>
  11. </div>
  12. <div class="row q-pa-md" style="min-height: 340px">
  13. <div class="col-12 col-sm-4" style="height: 340px">
  14. <q-carousel
  15. v-if="renderComponent"
  16. v-model="slide"
  17. animated
  18. navigation
  19. infinite
  20. height="100%"
  21. :autoplay="autoplay"
  22. arrows
  23. transition-prev="slide-right"
  24. transition-next="slide-left"
  25. @mouseenter="autoplay = false"
  26. @mouseleave="autoplay = 2000"
  27. >
  28. <q-carousel-slide
  29. v-for="(imageUrl, index) in imageContainer.imageList"
  30. :key="index"
  31. :name="index"
  32. :img-src="imageUrl"
  33. >
  34. </q-carousel-slide>
  35. </q-carousel>
  36. </div>
  37. <div class="col-12 col-sm-8 column" style="height: 340px">
  38. <div class="text-h4 q-pl-md" style="min-height: 50px">
  39. {{ productName }}
  40. </div>
  41. <div class="q-ml-md q-pl-md col" style="background-color: #fff9c6">
  42. {{ removeHtmlTag(shortProductDescription).substring(0, 500) }}...
  43. </div>
  44. </div>
  45. </div>
  46. <div class="row" style="min-height: 500px">
  47. <div class="col-12 col-sm-4 row justify-center" style="height: 500px">
  48. <!-- https://github.com/shubhadip/vuejs3-datepicker#install -->
  49. <datepicker
  50. v-if="renderComponent && productTypeId != LOCALSPECIALTYPRODUCTTYPE"
  51. @selected="handleSelectDate"
  52. :use-utc="shouldUseUtc"
  53. :disabled-dates="state.disabledDates"
  54. v-model="selectedDate"
  55. :prevent-disable-date-selection="true"
  56. :inline="shouldInline"
  57. :icon-width="40"
  58. calendar-class="calendarclass"
  59. ></datepicker>
  60. </div>
  61. <div
  62. class="col-12 col-sm-8 q-pl-xl q-pt-md column"
  63. style="min-height: 500px"
  64. >
  65. <div class="row">
  66. <div class="col-3 col-sm-2 text-weight-bold text-h4">Days:</div>
  67. <div
  68. v-if="productTypeId != FLIGHTPRODUCTTYPE"
  69. class="col-11 col-sm-10 flex text-h6 q-pl-sm"
  70. style="align-items: center"
  71. >
  72. {{ days - 1 }} nights {{ days }} days
  73. </div>
  74. <div
  75. v-if="productTypeId === FLIGHTPRODUCTTYPE"
  76. class="col-11 col-sm-10 flex text-h6 q-pl-sm"
  77. style="align-items: center"
  78. >
  79. {{ days }} days
  80. </div>
  81. </div>
  82. <div class="row">
  83. <div class="col-3 col-sm-2 text-weight-bold text-h4">Option:</div>
  84. <div
  85. class="col-11 col-sm-10 flex text-h6 q-pl-sm"
  86. style="align-items: center"
  87. >
  88. <q-btn-toggle
  89. v-if="productTypeId != LOCALSPECIALTYPRODUCTTYPE"
  90. v-model="activeIndex"
  91. spread
  92. no-caps
  93. toggle-color="blue-6"
  94. color="white"
  95. text-color="black"
  96. @click="
  97. setActiveCate(packageCategories[activeIndex], activeIndex)
  98. "
  99. :options="labelValuePair(packageCategories)"
  100. />
  101. <q-btn-toggle
  102. v-if="productTypeId === LOCALSPECIALTYPRODUCTTYPE"
  103. v-model="activeIndex"
  104. spread
  105. no-caps
  106. toggle-color="blue-6"
  107. color="white"
  108. text-color="black"
  109. @click="
  110. setActiveCateForLocalSpecialty(
  111. packageCategories[activeIndex],
  112. activeIndex
  113. )
  114. "
  115. :options="labelValuePair(packageCategories)"
  116. />
  117. </div>
  118. </div>
  119. <div
  120. v-if="
  121. productTypeId == 'd7b95089-e278-4522-8f71-dacac41ba32f' ||
  122. productTypeId == '3a53caed-b788-4290-896d-7922532ad769' ||
  123. productTypeId == 'e05d07a3-a717-45b8-b009-47a349890e41'
  124. "
  125. class="row text-weight-bold text-h4 q-mt-xl"
  126. >
  127. Passengers
  128. </div>
  129. <div
  130. v-if="productTypeId == HOTELPRODUCTTYPE"
  131. class="row text-weight-bold text-h4 q-mt-xl"
  132. >
  133. Rooms
  134. </div>
  135. <div class="row q-mt-md" style="border-top: 2px solid black">
  136. <div
  137. v-if="productTypeId != LOCALSPECIALTYPRODUCTTYPE"
  138. class="row col-12 q-pr-md q-mt-md"
  139. >
  140. <div class="col-11 col-sm-4 q-pr-md">
  141. <q-input
  142. label="Adult Count"
  143. v-model.number="adultCount"
  144. type="number"
  145. min="1"
  146. outlined
  147. />
  148. </div>
  149. <div class="col-11 col-sm-4 q-pr-md">
  150. <q-input
  151. label="Child Count"
  152. v-model.number="childCount"
  153. type="number"
  154. min="0"
  155. outlined
  156. />
  157. </div>
  158. <div class="col-11 col-sm-4 q-pr-md">
  159. <q-input
  160. label="Infant Count(less than 2 years old)"
  161. v-model.number="infantCount"
  162. type="number"
  163. min="0"
  164. outlined
  165. />
  166. </div>
  167. </div>
  168. <div
  169. v-if="productTypeId === LOCALSPECIALTYPRODUCTTYPE"
  170. class="row col-12 q-pr-md q-mt-md"
  171. >
  172. <div class="col-4 q-pr-md">
  173. <q-input
  174. label="Count"
  175. v-model.number="buyCount"
  176. type="number"
  177. min="1"
  178. outlined
  179. />
  180. </div>
  181. </div>
  182. <div class="row col-12" style="height: 50px"></div>
  183. <div
  184. class="row col-12 q-pr-md q-mt-md"
  185. style="background-color: #e9e9e9; min-height: 50px"
  186. >
  187. <div
  188. class="col-4 text-blue-6 text-h5 flex"
  189. style="align-items: center; height: 50px"
  190. >
  191. CNY
  192. </div>
  193. <div
  194. v-if="
  195. productTypeId != LOCALSPECIALTYPRODUCTTYPE &&
  196. productTypeId != HOTELPRODUCTTYPE
  197. "
  198. class="col-4 col-sm-3 flex"
  199. style="align-items: center; height: 50px"
  200. >
  201. {{
  202. adultCount * adultUnitPrice +
  203. childCount * childUnitPrice +
  204. infantCount * infantUnitPrice
  205. }}
  206. </div>
  207. <div
  208. v-if="
  209. productTypeId == LOCALSPECIALTYPRODUCTTYPE ||
  210. productTypeId == HOTELPRODUCTTYPE
  211. "
  212. class="col-4 col-sm-3 flex"
  213. style="align-items: center; height: 50px"
  214. >
  215. {{ generalPrice * buyCount }}
  216. </div>
  217. <div
  218. class="col-7 col-sm-3 flex"
  219. style="align-items: center; height: 50px"
  220. >
  221. <q-btn
  222. class="glossy"
  223. rounded
  224. color="deep-orange"
  225. @click="goPage('/contact')"
  226. label="Pre-book Consult"
  227. />
  228. </div>
  229. <div
  230. class="col-5 col-sm-2 flex"
  231. style="align-items: center; height: 50px"
  232. >
  233. <q-btn
  234. class="glossy"
  235. @click="collectConfigAndGoPage()"
  236. rounded
  237. color="blue-6"
  238. label="Book Now"
  239. />
  240. </div>
  241. </div>
  242. </div>
  243. </div>
  244. </div>
  245. <div
  246. class="row q-pa-md"
  247. style="min-height: 500px; background-color: #d9d9d9"
  248. >
  249. <div
  250. class="col-12"
  251. style="background-color: white"
  252. v-html="productDescription"
  253. ></div>
  254. </div>
  255. </q-page>
  256. </template>
  257. <script>
  258. import { ref, onMounted, nextTick, getCurrentInstance } from "vue";
  259. import { api } from "boot/axios";
  260. import { useSeoMeta } from "unhead";
  261. import { useQuasar } from "quasar";
  262. import { useRoute, useRouter } from "vue-router";
  263. import Datepicker from "vuejs3-datepicker";
  264. import { bookParamStore } from "stores/bookParamStore";
  265. export default {
  266. name: "ProductDetailPage",
  267. components: {
  268. Datepicker,
  269. },
  270. setup() {
  271. const store = bookParamStore();
  272. const instance = getCurrentInstance();
  273. const app = instance.appContext.app;
  274. const gp = app.config.globalProperties;
  275. const $q = useQuasar();
  276. const route = useRoute();
  277. const router = useRouter();
  278. const buyCount = ref(1);
  279. const adultCount = ref(1);
  280. const childCount = ref(0);
  281. const infantCount = ref(0);
  282. const skusInGroup = { skus: [] };
  283. const productName = ref("");
  284. const productDescription = ref("");
  285. const shortProductDescription = ref("");
  286. const productTypeId = ref("");
  287. const productTypeName = ref("");
  288. const packageCategories = ref([]);
  289. const imageList = ref([]);
  290. const days = ref("");
  291. const imageContainer = { imageList: [] };
  292. const activeCategory = ref("");
  293. const filteredSkuIndex = ref([]);
  294. const activeIndex = ref(0);
  295. const adultUnitPrice = ref(0);
  296. const generalPrice = ref(0);
  297. const childUnitPrice = ref(0);
  298. const infantUnitPrice = ref(0);
  299. const singleMatchedSku = ref({});
  300. const selectedDate = ref(new Date());
  301. const maxNum = ref(1000000);
  302. const renderComponent = ref(true);
  303. const totalPrice = ref(0);
  304. const TYPEOFPACKAGEPROP = "11cd8b32-c4f6-47db-8b8a-486c992bf43b";
  305. const LOCALSPECIALTYPRODUCTTYPE = "fd264cab-ee8d-4571-a477-03d7e7c090b3";
  306. const HOTELPRODUCTTYPE = "a9f5adbe-c09b-49bc-a614-8a1c5d5e5337";
  307. const MAXNUM = "720f4f2e-e114-4003-9806-bc56a9366278";
  308. const PRODUCTIMAGEURLPROP = "2dea54f4-9b9c-413a-8b3a-0caf273283d2";
  309. const DAYSPROP = "8cc865ff-b30f-4f00-b426-9e64418e5100";
  310. const AVAILABLEDATEPROP = "f0b807e5-d1a6-4454-a400-7905a4fea492";
  311. const TOURPRODUCTTYPE = "3a53caed-b788-4290-896d-7922532ad769";
  312. const FLIGHTHOTELPRODUCTTYPE = "e05d07a3-a717-45b8-b009-47a349890e41";
  313. const FLIGHTPRODUCTTYPE = "d7b95089-e278-4522-8f71-dacac41ba32f";
  314. const GENERALPRICE = "e16df480-b17d-4442-91c2-d6c30d0d7ab0";
  315. const ADULTPRICEPROP = "e228b843-e054-41f8-91dd-19663460df54";
  316. const CHILDPRICEPROP = "c4c845a7-4bef-46d8-a5ad-d72a5464e8b1";
  317. const INFANTPRICEPROP = "448406cb-b68f-439e-9da8-148d78ae8404";
  318. const SHORTDESCRIPTIONPROP = "f3cac6ed-8e87-4fb9-a912-bd87f483b4d5";
  319. const forceRerender = async () => {
  320. // Remove MyComponent from the DOM
  321. renderComponent.value = false;
  322. // Wait for the change to get flushed to the DOM
  323. await nextTick();
  324. // Add the component back in
  325. renderComponent.value = true;
  326. };
  327. function getTotalPrice() {
  328. if (
  329. productTypeId.value != LOCALSPECIALTYPRODUCTTYPE &&
  330. productTypeId.value != HOTELPRODUCTTYPE
  331. ) {
  332. totalPrice.value =
  333. adultCount.value * adultUnitPrice.value +
  334. childCount.value * childUnitPrice.value +
  335. infantCount.value * infantUnitPrice.value;
  336. } else {
  337. totalPrice.value = generalPrice.value * buyCount.value;
  338. }
  339. console.log("this is the total price:");
  340. console.log(totalPrice.value);
  341. return totalPrice.value;
  342. }
  343. function collectConfigAndGoPage() {
  344. if (singleMatchedSku.value == null) {
  345. gp.$generalNotify($q, false, "Not all required config are selected");
  346. return;
  347. }
  348. var params = {};
  349. params.productTypeId = productTypeId.value;
  350. params.productName = productName.value;
  351. params.productSkuId =
  352. singleMatchedSku.value["hichinaProductBasicDTO"]["skuId"];
  353. params.productSkuGroupId = route.params.skuGroupId;
  354. params.packageCategory = activeCategory.value;
  355. params.productTypeName = productTypeName.value;
  356. params.profileImageUrl = extractAttributeValueFromProductPropertyBag(
  357. singleMatchedSku.value,
  358. PRODUCTIMAGEURLPROP
  359. );
  360. params.totalPrice = getTotalPrice();
  361. //params.totalPrice = 1;
  362. if (productTypeId.value != LOCALSPECIALTYPRODUCTTYPE) {
  363. params.adultCount = adultCount.value;
  364. params.childCount = childCount.value;
  365. params.infantCount = infantCount.value;
  366. }
  367. if (LOCALSPECIALTYPRODUCTTYPE != productTypeId.value) {
  368. params.selectedDate = selectedDate.value.toISOString().split("T")[0];
  369. }
  370. if (
  371. productTypeId.value == LOCALSPECIALTYPRODUCTTYPE ||
  372. productTypeId.value == HOTELPRODUCTTYPE
  373. ) {
  374. params.buyCount = buyCount.value;
  375. }
  376. console.log("params collected from product");
  377. console.log(params);
  378. // gp.$goPageWithParams(val, params);
  379. // router.push({ name: val, state: params });
  380. gp.$goPage("/book");
  381. store.setOrderDetail(params);
  382. }
  383. function handleSelectDate(dt) {
  384. console.log("dt");
  385. console.log(dt);
  386. console.log("filteredSkuIndex.value");
  387. console.log(filteredSkuIndex.value);
  388. console.log("skusInGroup.skus");
  389. console.log(skusInGroup.skus);
  390. var matchedIndex = -1;
  391. for (var i in filteredSkuIndex.value) {
  392. var index = filteredSkuIndex.value[i];
  393. var availableDatesInString =
  394. extractAttributeValueFromProductPropertyBag(
  395. skusInGroup.skus[index],
  396. AVAILABLEDATEPROP
  397. );
  398. console.log("availableDatesInString");
  399. console.log(availableDatesInString);
  400. var dateObjArray = multiDateString2DateObjectArray(
  401. availableDatesInString
  402. );
  403. console.log("dateObjArray");
  404. console.log(dateObjArray);
  405. if (dateObjArray.some((e) => e.toDateString() == dt.toDateString())) {
  406. matchedIndex = index;
  407. break;
  408. }
  409. }
  410. console.log("matchedIndex");
  411. console.log(matchedIndex);
  412. singleMatchedSku.value = skusInGroup.skus[matchedIndex];
  413. maxNum.value = extractAttributeValueFromProductPropertyBag(
  414. singleMatchedSku.value,
  415. MAXNUM
  416. );
  417. console.log("maxNum");
  418. console.log(maxNum.value);
  419. setPrice(singleMatchedSku.value);
  420. }
  421. var state = {
  422. disabledDates: {
  423. to: new Date(2023, 3, 15), // Disable all dates up to specific date
  424. from: new Date(2023, 4, 20), // Disable all dates after specific date
  425. dates: [
  426. // Disable an array of dates
  427. new Date(2016, 9, 16),
  428. new Date(2016, 9, 17),
  429. new Date(2016, 9, 18),
  430. ],
  431. preventDisableDateSelection: true,
  432. },
  433. };
  434. function logPv() {
  435. api
  436. .post(
  437. "/api/public/pagestats/pv/product-detail-" + route.params.skuGroupId
  438. )
  439. .then((res) => {
  440. console.log("log pv:");
  441. console.log(res.data);
  442. })
  443. .catch((err) => {
  444. console.error("Error:", err);
  445. });
  446. }
  447. function logView() {
  448. api
  449. .post("/api/public/pagestats/view-product/" + route.params.skuGroupId)
  450. .then((res) => {
  451. console.log("view cnt of this product:");
  452. console.log(res.data);
  453. })
  454. .catch((err) => {
  455. console.error("Error:", err);
  456. });
  457. }
  458. function extractAttributeValueFromProductPropertyBag(inputObject, attrId) {
  459. var valArray = inputObject.productPropertyBag.filter((obj) => {
  460. return obj.attributeId == attrId;
  461. });
  462. if (valArray.length > 0) {
  463. return valArray[0].attributeValue;
  464. }
  465. return null;
  466. }
  467. function getFilteredSkuIndexAndSetActiveValue(item, index) {
  468. activeCategory.value = item;
  469. filteredSkuIndex.value = [];
  470. for (var i in skusInGroup.skus) {
  471. var sku = skusInGroup.skus[i];
  472. var selectedSkuPropertyBags = sku["productPropertyBag"].filter(
  473. (obj) => {
  474. return obj.attributeId == TYPEOFPACKAGEPROP;
  475. }
  476. );
  477. if (selectedSkuPropertyBags[0].attributeValue == item) {
  478. filteredSkuIndex.value.push(i);
  479. }
  480. }
  481. activeIndex.value = index;
  482. }
  483. function multiDateString2DateObjectArray(candidateAvailableDates) {
  484. var dateObjArray = [];
  485. var datestringArray = candidateAvailableDates.split(";");
  486. for (var i in datestringArray) {
  487. var datestr = datestringArray[i];
  488. if (datestr != null && datestr.length > 0) {
  489. var dateObj = new Date(datestr);
  490. // console.log("...converting string to date:"+ datestr)
  491. // console.log(dateObj)
  492. dateObjArray.push(dateObj);
  493. }
  494. }
  495. return dateObjArray;
  496. }
  497. function compareDate(a, b) {
  498. if (a.getDate() < b.getDate()) {
  499. return -1;
  500. }
  501. if (a.getDate() > b.getDate()) {
  502. return 1;
  503. }
  504. return 0;
  505. }
  506. function setCandidateAvailableDates(indexArray) {
  507. var candidateAvailableDates = "";
  508. for (var i in indexArray) {
  509. var availableDates = extractAttributeValueFromProductPropertyBag(
  510. skusInGroup.skus[indexArray[i]],
  511. AVAILABLEDATEPROP
  512. );
  513. candidateAvailableDates = candidateAvailableDates + availableDates;
  514. }
  515. var dateObjArray = multiDateString2DateObjectArray(
  516. candidateAvailableDates
  517. );
  518. dateObjArray = dateObjArray.sort((a, b) => a - b);
  519. console.log("dateObjArray");
  520. console.log(dateObjArray);
  521. var minDate = dateObjArray[0];
  522. var maxDate = dateObjArray.slice(-1)[0];
  523. state.disabledDates.to = minDate;
  524. var nextDayOfMax = new Date(maxDate);
  525. nextDayOfMax.setDate(nextDayOfMax.getDate() + 1);
  526. state.disabledDates.from = nextDayOfMax;
  527. var segmentDisabledDates = [];
  528. var pointer = new Date(minDate);
  529. while (pointer < maxDate) {
  530. // check contains
  531. if (
  532. !dateObjArray.some((e) => e.toDateString() == pointer.toDateString())
  533. ) {
  534. segmentDisabledDates.push(new Date(pointer));
  535. } else {
  536. }
  537. // update pointer
  538. pointer.setDate(pointer.getDate() + 1);
  539. }
  540. state.disabledDates.dates = segmentDisabledDates;
  541. forceRerender();
  542. }
  543. function setPrice(sku) {
  544. if (sku == null) {
  545. if (
  546. productTypeId.value == LOCALSPECIALTYPRODUCTTYPE ||
  547. productTypeId.value == HOTELPRODUCTTYPE
  548. ) {
  549. console.log("001");
  550. generalPrice.value = 0;
  551. } else {
  552. adultUnitPrice.value = 0;
  553. childUnitPrice.value = 0;
  554. infantUnitPrice.value = 0;
  555. }
  556. } else {
  557. if (
  558. productTypeId.value == LOCALSPECIALTYPRODUCTTYPE ||
  559. productTypeId.value == HOTELPRODUCTTYPE
  560. ) {
  561. console.log("003");
  562. generalPrice.value = extractAttributeValueFromProductPropertyBag(
  563. sku,
  564. GENERALPRICE
  565. );
  566. } else {
  567. console.log("004");
  568. adultUnitPrice.value = extractAttributeValueFromProductPropertyBag(
  569. sku,
  570. ADULTPRICEPROP
  571. );
  572. childUnitPrice.value = extractAttributeValueFromProductPropertyBag(
  573. sku,
  574. CHILDPRICEPROP
  575. );
  576. infantUnitPrice.value = extractAttributeValueFromProductPropertyBag(
  577. sku,
  578. INFANTPRICEPROP
  579. );
  580. }
  581. }
  582. }
  583. function setActiveDescription() {
  584. var indexToFilter =
  585. filteredSkuIndex.value.length < 1 ? 0 : filteredSkuIndex.value[0];
  586. productDescription.value =
  587. skusInGroup.skus[indexToFilter]["hichinaProductBasicDTO"][
  588. "productContent"
  589. ];
  590. shortProductDescription.value =
  591. extractAttributeValueFromProductPropertyBag(
  592. skusInGroup.skus[indexToFilter],
  593. SHORTDESCRIPTIONPROP
  594. );
  595. }
  596. function setActiveCate(item, index) {
  597. console.log("picking package cate: " + item);
  598. console.log("index of it is: " + index);
  599. getFilteredSkuIndexAndSetActiveValue(item, index);
  600. setCandidateAvailableDates(filteredSkuIndex.value);
  601. setPrice(null);
  602. selectedDate.value = new Date();
  603. singleMatchedSku.value = null;
  604. setActiveDescription();
  605. }
  606. function setTourProductData(inputArray) {
  607. packageCategories.value = [];
  608. imageList.value = [];
  609. for (var index in inputArray) {
  610. var packageCat = extractAttributeValueFromProductPropertyBag(
  611. inputArray[index],
  612. TYPEOFPACKAGEPROP
  613. );
  614. if (packageCat != null) {
  615. packageCategories.value.push(packageCat);
  616. }
  617. var url = extractAttributeValueFromProductPropertyBag(
  618. inputArray[index],
  619. PRODUCTIMAGEURLPROP
  620. );
  621. if (url != null) {
  622. // imageContainer.imageList.push(url);
  623. imageList.value.push(url);
  624. }
  625. // this if means only need to calculate days once because all this property in the same sku group should be same
  626. if (days.value == "") {
  627. days.value = extractAttributeValueFromProductPropertyBag(
  628. inputArray[index],
  629. DAYSPROP
  630. );
  631. }
  632. }
  633. imageContainer.imageList = imageList.value;
  634. forceRerender();
  635. // trick: remove duplicates
  636. packageCategories.value = [...new Set(packageCategories.value)];
  637. // force select the first package category on entering page
  638. setActiveCate(packageCategories.value[0], 0);
  639. }
  640. function labelValuePair(valArray) {
  641. var ret = [];
  642. valArray.forEach((x, i) => {
  643. var obj = { label: x, value: i };
  644. ret.push(obj);
  645. });
  646. return ret;
  647. }
  648. function setFlightHotelProductData(inputArray) {
  649. setTourProductData(inputArray);
  650. }
  651. function setHotelProductData(inputArray) {
  652. setTourProductData(inputArray);
  653. }
  654. function setFligtProductData(inputArray) {
  655. setTourProductData(inputArray);
  656. }
  657. function setLocalSpecialtyProductData(inputArray) {
  658. packageCategories.value = [];
  659. imageList.value = [];
  660. for (var index in inputArray) {
  661. var packageCat = extractAttributeValueFromProductPropertyBag(
  662. inputArray[index],
  663. TYPEOFPACKAGEPROP
  664. );
  665. if (packageCat != null) {
  666. packageCategories.value.push(packageCat);
  667. }
  668. var url = extractAttributeValueFromProductPropertyBag(
  669. inputArray[index],
  670. PRODUCTIMAGEURLPROP
  671. );
  672. if (url != null) {
  673. imageList.value.push(url);
  674. }
  675. }
  676. imageContainer.imageList = imageList.value;
  677. // trick: remove duplicates
  678. packageCategories.value = [...new Set(packageCategories.value)];
  679. // force select the first package category on entering page
  680. setActiveCateForLocalSpecialty(packageCategories.value[0], 0);
  681. }
  682. function processSkuGroups(inputArray) {
  683. if (inputArray.length > 0) {
  684. productName.value =
  685. inputArray[0]["hichinaProductBasicDTO"]["productName"];
  686. productTypeId.value =
  687. inputArray[0]["hichinaProductBasicDTO"]["productTypeId"];
  688. productTypeName.value =
  689. inputArray[0]["hichinaProductBasicDTO"]["productTypeName"];
  690. useSeoMeta({
  691. title: productName.value,
  692. description: productName.value,
  693. ogDescription: productName.value,
  694. ogTitle: productName.value,
  695. ogImage: "https://www.hichinatravel.com/static/png/name-67280b81.png",
  696. twitterCard: "summary_large_image",
  697. });
  698. if (productTypeId.value === TOURPRODUCTTYPE) {
  699. // 跟团游,则按照跟团游的页面模板来设置各个参数
  700. setTourProductData(inputArray);
  701. } else if (productTypeId.value === FLIGHTHOTELPRODUCTTYPE) {
  702. // 机票酒店套餐
  703. setFlightHotelProductData(inputArray);
  704. } else if (productTypeId.value === FLIGHTPRODUCTTYPE) {
  705. // 机票
  706. setFligtProductData(inputArray);
  707. } else if (productTypeId.value === HOTELPRODUCTTYPE) {
  708. // 酒店
  709. setHotelProductData(inputArray);
  710. } else if (productTypeId.value === LOCALSPECIALTYPRODUCTTYPE) {
  711. // 本地土特产
  712. setLocalSpecialtyProductData(inputArray);
  713. }
  714. }
  715. }
  716. function loadSkusInGroup() {
  717. var params = {};
  718. params.skuGroupId = route.params.skuGroupId;
  719. api
  720. .get("/api/public/productsku/bygroupidwithpropertybag", {
  721. params: params,
  722. })
  723. .then(function (response) {
  724. skusInGroup.skus = response.data.data;
  725. console.log("skusInGroup.skus in loadSkusInGroup");
  726. console.log(skusInGroup.skus);
  727. processSkuGroups(skusInGroup.skus);
  728. })
  729. .catch(function (error) {
  730. console.log(error);
  731. });
  732. }
  733. onMounted(() => {
  734. logPv();
  735. logView();
  736. loadSkusInGroup();
  737. });
  738. return {
  739. productTypeName,
  740. productName,
  741. productTypeId,
  742. days,
  743. slide: ref(0),
  744. autoplay: ref(true),
  745. imageContainer,
  746. productDescription,
  747. shortProductDescription,
  748. handleSelectDate,
  749. selectedDate,
  750. state,
  751. renderComponent,
  752. LOCALSPECIALTYPRODUCTTYPE,
  753. FLIGHTPRODUCTTYPE,
  754. HOTELPRODUCTTYPE,
  755. packageCategories,
  756. labelValuePair,
  757. setActiveCate,
  758. activeIndex,
  759. adultCount,
  760. childCount,
  761. infantCount,
  762. adultUnitPrice,
  763. childUnitPrice,
  764. infantUnitPrice,
  765. generalPrice,
  766. buyCount,
  767. shouldUseUtc: ref(true),
  768. shouldInline: ref(true),
  769. collectConfigAndGoPage,
  770. };
  771. },
  772. };
  773. </script>