MainLayout.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. <template>
  2. <q-layout view="hHh lpR fff">
  3. <q-header height-hint="76" reveal class="bg-white text-black">
  4. <q-toolbar class="GPL__toolbar" style="height: 76px">
  5. <div class="row">
  6. <div class="col-sm-12 col-xs-12 q-ml-xl q-mr-lg">
  7. <q-toolbar-title>
  8. <img
  9. @click="$router.push('/')"
  10. class="cursor-pointer float-left"
  11. src="~/assets/hichinalogo.png"
  12. />
  13. </q-toolbar-title>
  14. </div>
  15. </div>
  16. <q-btn
  17. @click="goPage('/guideintro')"
  18. flat
  19. no-caps
  20. no-wrap
  21. class="q-ml-xl"
  22. v-if="$q.screen.gt.xs"
  23. >
  24. {{ $t("guidebooks") }}
  25. </q-btn>
  26. <q-btn
  27. @click="goPage('/blog')"
  28. flat
  29. no-caps
  30. no-wrap
  31. class="q-ml-sm"
  32. v-if="$q.screen.gt.xs"
  33. >
  34. {{ $t("blogs_vlogs") }}
  35. </q-btn>
  36. <q-btn
  37. @click="goPage('/product')"
  38. flat
  39. no-caps
  40. no-wrap
  41. class="q-ml-sm"
  42. v-if="$q.screen.gt.xs"
  43. >
  44. {{ $t("travel_shop") }}
  45. </q-btn>
  46. <q-btn
  47. @click="goPage('/destination')"
  48. flat
  49. no-caps
  50. no-wrap
  51. class="q-ml-sm"
  52. v-if="$q.screen.gt.xs"
  53. >
  54. {{ $t("destinations") }}
  55. </q-btn>
  56. <!-- this is the trick -->
  57. <q-space />
  58. <div class="row no-wrap q-mr-md">
  59. <q-select
  60. v-model="locale"
  61. :options="localeOptions"
  62. :label="$t('language')"
  63. dense
  64. emit-value
  65. map-options
  66. options-dense
  67. style="min-width: 150px"
  68. @update:model-value="setLanguage"
  69. ><template v-slot:append> <q-icon name="public" /> </template
  70. ></q-select>
  71. </div>
  72. <div v-if="currentUser === ''" class="row no-wrap">
  73. <q-btn
  74. @click="goPage('/auth/register')"
  75. flat
  76. dense
  77. no-wrap
  78. no-caps
  79. color="primary"
  80. :label="$t('register')"
  81. class="q-mr-md"
  82. v-if="$q.screen.gt.xs"
  83. />
  84. <q-btn
  85. @click="goPage('/auth/login')"
  86. flat
  87. dense
  88. no-wrap
  89. no-caps
  90. color="primary"
  91. :label="$t('login')"
  92. class="q-mr-sm"
  93. v-if="$q.screen.gt.xs"
  94. />
  95. </div>
  96. <div v-if="currentUser != ''" class="row">
  97. <q-btn
  98. v-if="$q.screen.gt.xs"
  99. round
  100. flat
  101. @click="goPage('/user-info')"
  102. >
  103. <q-avatar size="36px">
  104. <img :src="currentProfileImage" />
  105. </q-avatar>
  106. <q-tooltip>{{ currentUser }}</q-tooltip>
  107. </q-btn>
  108. </div>
  109. <div v-if="currentUser != '' && $q.screen.gt.xs" class="row q-mr-xl">
  110. <q-btn icon="arrow_drop_down" flat dense>
  111. <q-menu
  112. style="width: 100px"
  113. transition-show="flip-right"
  114. transition-hide="flip-left"
  115. >
  116. <q-list dense class="text-grey-9 text-caption">
  117. <q-item clickable @click="goPage('/my-blogs')">
  118. <q-item-section>{{ $t("my_blogs") }}</q-item-section>
  119. </q-item>
  120. <q-item clickable @click="goPage('/my-orders')">
  121. <q-item-section>{{ $t("my_orders") }}</q-item-section>
  122. </q-item>
  123. <q-item clickable @click="goPage('/user-info')">
  124. <q-item-section>{{ $t("edit_profile") }}</q-item-section>
  125. </q-item>
  126. <q-separator />
  127. <q-item clickable @click="logout()">
  128. <q-item-section>{{ $t("logout") }}</q-item-section>
  129. </q-item>
  130. </q-list>
  131. </q-menu>
  132. </q-btn>
  133. </div>
  134. <!-- <div v-if="currentUser != ''" class="row no-wrap q-ml-md">
  135. <q-btn
  136. @click="logout()"
  137. v-if="$q.screen.gt.xs"
  138. round
  139. dense
  140. color="primary"
  141. class="q-mr-md"
  142. icon="logout"
  143. ><q-tooltip>logout</q-tooltip></q-btn
  144. >
  145. </div> -->
  146. <div>
  147. <q-btn
  148. v-if="!$q.screen.gt.xs"
  149. flat
  150. dense
  151. round
  152. @click="toggleLeftDrawer"
  153. icon="menu"
  154. aria-label="Menu"
  155. />
  156. </div>
  157. </q-toolbar>
  158. </q-header>
  159. <q-drawer v-model="leftDrawerOpen" bordered class="bg-grey-2" :width="240">
  160. <q-scroll-area class="fit">
  161. <q-list padding>
  162. <q-item v-ripple clickable @click="goPage('/guideintro')">
  163. <q-item-section avatar>
  164. <q-icon color="grey" name="fingerprint" />
  165. </q-item-section>
  166. <q-item-section>
  167. <q-item-label>{{ $t("guidebooks") }}</q-item-label>
  168. </q-item-section>
  169. </q-item>
  170. <q-item v-ripple clickable @click="goPage('/blog')">
  171. <q-item-section avatar>
  172. <q-icon color="grey" name="fingerprint" />
  173. </q-item-section>
  174. <q-item-section>
  175. <q-item-label>{{ $t("blogs_vlogs") }}</q-item-label>
  176. </q-item-section>
  177. </q-item>
  178. <q-item v-ripple clickable @click="goPage('/product')">
  179. <q-item-section avatar>
  180. <q-icon color="grey" name="fingerprint" />
  181. </q-item-section>
  182. <q-item-section>
  183. <q-item-label>{{ $t("travel_shop") }}</q-item-label>
  184. </q-item-section>
  185. </q-item>
  186. <q-item v-ripple clickable @click="goPage('/destination')">
  187. <q-item-section avatar>
  188. <q-icon color="grey" name="fingerprint" />
  189. </q-item-section>
  190. <q-item-section>
  191. <q-item-label>{{ $t("destinations") }}</q-item-label>
  192. </q-item-section>
  193. </q-item>
  194. <q-separator class="q-my-md" />
  195. <q-item
  196. v-if="currentUser === ''"
  197. v-ripple
  198. clickable
  199. @click="goPage('/auth/login')"
  200. >
  201. <q-item-section avatar>
  202. <q-icon color="grey" name="login" />
  203. </q-item-section>
  204. <q-item-section>
  205. <q-item-label>{{ $t("login") }}</q-item-label>
  206. </q-item-section>
  207. </q-item>
  208. <q-item
  209. v-if="currentUser === ''"
  210. v-ripple
  211. clickable
  212. @click="goPage('/auth/register')"
  213. >
  214. <q-item-section avatar>
  215. <q-icon color="grey" name="account_circle" />
  216. </q-item-section>
  217. <q-item-section>
  218. <q-item-label>{{ $t("register") }}</q-item-label>
  219. </q-item-section>
  220. </q-item>
  221. <q-item
  222. v-if="currentUser !== ''"
  223. v-ripple
  224. clickable
  225. @click="goPage('/my-blogs')"
  226. >
  227. <q-item-section avatar>
  228. <q-icon color="grey" name="rss_feed" />
  229. </q-item-section>
  230. <q-item-section>
  231. <q-item-label>{{ $t("my_blogs") }}</q-item-label>
  232. </q-item-section>
  233. </q-item>
  234. <q-item
  235. v-if="currentUser !== ''"
  236. v-ripple
  237. clickable
  238. @click="goPage('/my-orders')"
  239. >
  240. <q-item-section avatar>
  241. <q-icon color="grey" name="list_alt" />
  242. </q-item-section>
  243. <q-item-section>
  244. <q-item-label>{{ $t("my_orders") }}</q-item-label>
  245. </q-item-section>
  246. </q-item>
  247. <q-item
  248. v-if="currentUser !== ''"
  249. v-ripple
  250. clickable
  251. @click="goPage('/user-info')"
  252. >
  253. <q-item-section avatar>
  254. <q-icon color="grey" name="manage_accounts" />
  255. </q-item-section>
  256. <q-item-section>
  257. <q-item-label>{{ $t("edit_profile") }}</q-item-label>
  258. </q-item-section>
  259. </q-item>
  260. <q-item
  261. v-if="currentUser !== ''"
  262. v-ripple
  263. clickable
  264. @click="logout()"
  265. >
  266. <q-item-section avatar>
  267. <q-icon color="grey" name="logout" />
  268. </q-item-section>
  269. <q-item-section>
  270. <q-item-label>{{ $t("logout") }}</q-item-label>
  271. </q-item-section>
  272. </q-item>
  273. </q-list>
  274. </q-scroll-area>
  275. </q-drawer>
  276. <q-page-container>
  277. <router-view />
  278. <q-footer bordered class="bg-blue-6 text-white">
  279. <div class="q-pa-md">
  280. <div class="row justify-center">
  281. <div class="col-12 col-md-3 text-white q-pl-xl q-pt-md">
  282. <div class="text-subtitle1 text-weight-bold">
  283. About HiChinaTravel
  284. </div>
  285. <div class="text-body2 hover_underline_white q-mt-md">
  286. About Us
  287. </div>
  288. <div class="text-body2 hover_underline_white q-mt-md">
  289. Contact us
  290. </div>
  291. <div class="text-body2 hover_underline_white q-mt-md">
  292. Copyright
  293. </div>
  294. <div
  295. class="text-body2 hover_underline_white q-mt-md cursor-pointer"
  296. @click="goPage('/privacy')"
  297. >
  298. Privacy
  299. </div>
  300. <div class="text-body2 hover_underline_white q-mt-md">
  301. Join Us
  302. </div>
  303. </div>
  304. <div class="col-12 col-md-5 q-pt-xl">
  305. <div
  306. class="text-subtitle1 text-weight-medium hover_underline_white q-mt-md"
  307. >
  308. Facebook: HiChinaTravel
  309. </div>
  310. <div
  311. class="text-subtitle1 text-weight-medium hover_underline_white q-mt-md"
  312. >
  313. WeChat Official Account: HCTravel
  314. </div>
  315. <div
  316. class="text-subtitle1 text-weight-medium hover_underline_white q-mt-md"
  317. >
  318. Instagram: HiChinaTravel
  319. </div>
  320. <div
  321. class="text-subtitle1 text-weight-medium hover_underline_white q-mt-md"
  322. >
  323. Support: customerservice@hichinatrip.com
  324. </div>
  325. <div
  326. class="text-subtitle1 text-weight-medium hover_underline_white q-mt-md"
  327. >
  328. Address: 4,Zhixin Rd. Qixia District Nanjing City, Jiangsu
  329. Province, P.R China
  330. </div>
  331. </div>
  332. <div class="col-12 col-md-4"></div>
  333. </div>
  334. <div class="row justify-center q-mt-xl">
  335. <div class="text-body2 text-weight-medium hover_underline_white">
  336. COPYRIGHT © 2015-2023 WWW.HICHINATRAVEL.COM, ALL RIGHTS RESERVED
  337. </div>
  338. </div>
  339. <div class="row justify-center">
  340. <div class="text-body2 text-weight-medium hover_underline_white">
  341. 备案号: 京ICP备16006305号-1
  342. </div>
  343. </div>
  344. </div>
  345. </q-footer>
  346. </q-page-container>
  347. <!-- Messenger Chat Plugin Code -->
  348. <div id="fb-root"></div>
  349. <!-- Your Chat Plugin code -->
  350. <div id="fb-customer-chat" class="fb-customerchat"></div>
  351. </q-layout>
  352. </template>
  353. <script>
  354. import {
  355. defineComponent,
  356. onMounted,
  357. onBeforeMount,
  358. ref,
  359. getCurrentInstance,
  360. } from "vue";
  361. import { api } from "boot/axios";
  362. import { useI18n } from "vue-i18n";
  363. import { useQuasar } from "quasar";
  364. export default defineComponent({
  365. name: "MainLayout",
  366. components: {},
  367. setup() {
  368. const { locale } = useI18n({ useScope: "global" });
  369. const leftDrawerOpen = ref(false);
  370. const currentUser = ref("");
  371. const currentProfileImage = ref("");
  372. const instance = getCurrentInstance();
  373. const app = instance.appContext.app;
  374. const gp = app.config.globalProperties;
  375. const $q = useQuasar();
  376. function setLanguage() {
  377. // $q.lang.set(locale);
  378. console.log(locale);
  379. }
  380. function whoami() {
  381. api
  382. .get("/api/v1/user/whoamiv2")
  383. .then(function (response) {
  384. console.log("current user:" + response.data.data.username);
  385. console.log(
  386. "current user profile image:" + response.data.data.profileImageUrl
  387. );
  388. currentUser.value = response.data.data.username;
  389. currentProfileImage.value = response.data.data.profileImageUrl;
  390. if (
  391. currentProfileImage.value == null ||
  392. currentProfileImage.value.length < 1
  393. ) {
  394. currentProfileImage.value =
  395. "https://photoprism.hichinatravel.com/api/v1/t/8623903789c65a160279faa0b33159413cb18af4/32mcf2k4/fit_2048";
  396. }
  397. console.log("currentProfileImage.value");
  398. console.log(currentProfileImage.value);
  399. })
  400. .catch(function (error) {
  401. console.log("not logged in err");
  402. // router.push({name: 'home'})
  403. });
  404. }
  405. function logout() {
  406. console.log("logging out...");
  407. api
  408. .post(
  409. "/logout",
  410. {},
  411. { headers: { "Content-Type": "application/x-www-form-urlencoded" } }
  412. )
  413. .then((response) => {
  414. location.reload();
  415. })
  416. .catch((e) => {
  417. location.reload();
  418. });
  419. }
  420. function setupChat() {
  421. var chatbox = document.getElementById("fb-customer-chat");
  422. chatbox.setAttribute("page_id", "1534271790187013");
  423. chatbox.setAttribute("attribution", "biz_inbox");
  424. }
  425. function initFacebookSDK() {
  426. window.fbAsyncInit = function () {
  427. FB.init({
  428. xfbml: true,
  429. version: "v17.0",
  430. });
  431. };
  432. (function (d, s, id) {
  433. var js,
  434. fjs = d.getElementsByTagName(s)[0];
  435. if (d.getElementById(id)) return;
  436. js = d.createElement(s);
  437. js.id = id;
  438. js.src = "https://connect.facebook.net/en_US/sdk/xfbml.customerchat.js";
  439. fjs.parentNode.insertBefore(js, fjs);
  440. })(document, "script", "facebook-jssdk");
  441. }
  442. onMounted(() => {
  443. whoami();
  444. setupChat();
  445. });
  446. onBeforeMount(() => {
  447. initFacebookSDK();
  448. });
  449. return {
  450. setLanguage,
  451. locale,
  452. localeOptions: [
  453. { value: "en-US", label: "English" },
  454. { value: "th-TH", label: "ภาษาไทย" },
  455. { value: "ko-KR", label: "한국인" },
  456. { value: "ru-RU", label: "Русский" },
  457. ],
  458. leftDrawerOpen,
  459. currentUser,
  460. currentProfileImage,
  461. menu_profile: false,
  462. logout,
  463. toggleLeftDrawer() {
  464. leftDrawerOpen.value = !leftDrawerOpen.value;
  465. },
  466. };
  467. },
  468. });
  469. </script>
  470. <style lang="sass">
  471. .GPL
  472. &__toolbar
  473. height: 64px
  474. &__toolbar-input
  475. width: 35%
  476. &__drawer-item
  477. line-height: 24px
  478. border-radius: 0 24px 24px 0
  479. margin-right: 12px
  480. .q-item__section--avatar
  481. padding-left: 12px
  482. .q-icon
  483. color: #5f6368
  484. .q-item__label:not(.q-item__label--caption)
  485. color: #3c4043
  486. letter-spacing: .01785714em
  487. font-size: .875rem
  488. font-weight: 500
  489. line-height: 1.25rem
  490. &--storage
  491. border-radius: 0
  492. margin-right: 0
  493. padding-top: 24px
  494. padding-bottom: 24px
  495. &__side-btn
  496. &__label
  497. font-size: 12px
  498. line-height: 24px
  499. letter-spacing: .01785714em
  500. font-weight: 500
  501. @media (min-width: 1024px)
  502. &__page-container
  503. padding-left: 94px
  504. </style>