|
@@ -184,6 +184,8 @@ const chatArea = ref(null); // 對話框
|
|
|
|
|
|
// 捲軸滾動到使用者訊息底下第一則回覆
|
|
|
const scrollToMessage = () => {
|
|
|
+ console.log("scrollToMessage");
|
|
|
+
|
|
|
// 使用 nextTick 確保 DOM 已更新
|
|
|
nextTick(() => {
|
|
|
// 取得所有的 .message.message-out 元素
|
|
@@ -197,6 +199,9 @@ const scrollToMessage = () => {
|
|
|
const nextMessageContent =
|
|
|
lastMessageOut.parentElement.nextElementSibling;
|
|
|
|
|
|
+ console.log("lastMessageOut", lastMessageOut);
|
|
|
+ console.log("nextMessageContent", nextMessageContent);
|
|
|
+
|
|
|
if (
|
|
|
nextMessageContent &&
|
|
|
nextMessageContent.classList.contains("message-content")
|
|
@@ -211,7 +216,7 @@ const scrollToMessage = () => {
|
|
|
top: scrollTop - 45,
|
|
|
behavior: "smooth",
|
|
|
});
|
|
|
- }, 100);
|
|
|
+ }, 300);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
@@ -1464,7 +1469,7 @@ async function getArviews(route, text, type = "") {
|
|
|
|
|
|
messages.value.push({
|
|
|
label: "text",
|
|
|
- author: "ai",
|
|
|
+ author: "user",
|
|
|
body: text,
|
|
|
});
|
|
|
|
|
@@ -1517,13 +1522,17 @@ function changeLocation(item) {
|
|
|
// 顯示平面圖
|
|
|
function assignMapImg(item) {
|
|
|
console.log("顯示平面圖 item", item);
|
|
|
+
|
|
|
let name;
|
|
|
- if (item.value === "B1 平面圖") {
|
|
|
+
|
|
|
+ if (item === "B1 平面圖") {
|
|
|
name = "All_b1";
|
|
|
- } else if (item.value === "1F 平面圖") {
|
|
|
+ } else if (item === "1F 平面圖") {
|
|
|
name = "All_1";
|
|
|
- } else if (item.value === "2F 平面圖") {
|
|
|
+ } else if (item === "2F 平面圖") {
|
|
|
name = "All_2";
|
|
|
+ } else {
|
|
|
+ name = "1F";
|
|
|
}
|
|
|
|
|
|
messages.value.push({
|
|
@@ -1680,11 +1689,13 @@ async function selectCategory(value, index) {
|
|
|
body: t("location_guide"),
|
|
|
});
|
|
|
|
|
|
- messages.value.push({
|
|
|
- label: "text",
|
|
|
- author: "ai",
|
|
|
- body: t("select_location"),
|
|
|
- });
|
|
|
+ assignMapImg("1F"); // 顯示 1F 平面圖
|
|
|
+
|
|
|
+ // messages.value.push({
|
|
|
+ // label: "text",
|
|
|
+ // author: "ai",
|
|
|
+ // body: t("select_location"),
|
|
|
+ // });
|
|
|
|
|
|
// 掃描 QR Code
|
|
|
// messages.value.push({
|
|
@@ -1699,11 +1710,11 @@ async function selectCategory(value, index) {
|
|
|
// body: mapList,
|
|
|
// });
|
|
|
|
|
|
- messages.value.push({
|
|
|
- label: "location",
|
|
|
- author: "ai",
|
|
|
- body: locationList,
|
|
|
- });
|
|
|
+ // messages.value.push({
|
|
|
+ // label: "location",
|
|
|
+ // author: "ai",
|
|
|
+ // body: locationList,
|
|
|
+ // });
|
|
|
} else if (value === "美食/伴手禮") {
|
|
|
// getAd("美食伴手禮");
|
|
|
|
|
@@ -1908,13 +1919,13 @@ const menuList = reactive([
|
|
|
]);
|
|
|
|
|
|
// 美食伴手禮 or 購物及優惠類別篩選
|
|
|
-async function findBrand(value) {
|
|
|
+async function findBrand(value, text) {
|
|
|
console.log("findBrand", value);
|
|
|
|
|
|
messages.value.push({
|
|
|
label: "text",
|
|
|
author: "user",
|
|
|
- body: value,
|
|
|
+ body: t(text),
|
|
|
});
|
|
|
|
|
|
if (value === "館外店家") {
|
|
@@ -1970,13 +1981,13 @@ async function getStaticTickets(type) {
|
|
|
}
|
|
|
|
|
|
// 秘境花園觀景台對話
|
|
|
-function handleObservationDialog(value) {
|
|
|
+function handleObservationDialog(value, text) {
|
|
|
console.log("value", value);
|
|
|
|
|
|
messages.value.push({
|
|
|
label: "text",
|
|
|
author: "user",
|
|
|
- body: value,
|
|
|
+ body: t(text),
|
|
|
});
|
|
|
|
|
|
if (value === "線上購票") {
|
|
@@ -2084,7 +2095,12 @@ let gardenRouteList = reactive([
|
|
|
]);
|
|
|
|
|
|
// 購物及優惠對話
|
|
|
-function handleShoppingDialog(value) {
|
|
|
+function handleShoppingDialog(value, text) {
|
|
|
+ messages.value.push({
|
|
|
+ label: "text",
|
|
|
+ author: "ai",
|
|
|
+ body: t(text),
|
|
|
+ });
|
|
|
console.log("value", value);
|
|
|
|
|
|
if (value === "購物品牌查詢") {
|
|
@@ -2416,6 +2432,11 @@ function recStart() {
|
|
|
// 開始計時
|
|
|
timer = setInterval(() => {
|
|
|
recordTime.value += 1;
|
|
|
+
|
|
|
+ // 檢查錄音時間是否超過 60 秒
|
|
|
+ if (recordTime.value >= 60) {
|
|
|
+ recStop(); // 停止錄音
|
|
|
+ }
|
|
|
}, 1000);
|
|
|
|
|
|
rec.start();
|
|
@@ -2433,18 +2454,6 @@ function recStop() {
|
|
|
// rec.close(); // 釋放錄音資源 (若不釋放系統或瀏覽器將持續提示在錄音中)
|
|
|
// rec = null;
|
|
|
|
|
|
- // // 韓文轉 wav 格式,其他語言轉 mp3 格式
|
|
|
- // let lang = getLang();
|
|
|
- // if (lang === "ko") {
|
|
|
- // audioFile.value = new File([blob], "recording.wav", {
|
|
|
- // type: "audio/wav",
|
|
|
- // });
|
|
|
- // } else {
|
|
|
- // audioFile.value = new File([blob], "recording.mp3", {
|
|
|
- // type: "audio/mp3",
|
|
|
- // });
|
|
|
- // }
|
|
|
-
|
|
|
// 將 Blob 轉換為 File 對象
|
|
|
audioFile.value = new File([blob], "recording.mp3", {
|
|
|
type: "audio/mp3",
|
|
@@ -2711,28 +2720,46 @@ function videoSrcBg() {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-// 時間內無動作需 reload
|
|
|
+// 五分鐘內無動作需 reload
|
|
|
let timeout;
|
|
|
+let elapsedTime = 0; // 追蹤經過的時間
|
|
|
+let intervalId;
|
|
|
|
|
|
const resetTimeout = () => {
|
|
|
clearTimeout(timeout);
|
|
|
+ clearInterval(intervalId); // 停止當前的計時器
|
|
|
+ elapsedTime = 0; // 重置經過時間
|
|
|
+ intervalId = setInterval(() => {
|
|
|
+ elapsedTime += 1; // 每秒增加
|
|
|
+ if (elapsedTime >= 300) {
|
|
|
+ // 5 分鐘 = 300 秒
|
|
|
+ window.location.reload(); // 重新整理網頁
|
|
|
+ }
|
|
|
+ }, 1000); // 每秒更新一次
|
|
|
+
|
|
|
timeout = setTimeout(() => {
|
|
|
+ clearInterval(intervalId); // 取消計時
|
|
|
window.location.reload(); // 重新整理網頁
|
|
|
- }, 5 * 60 * 1000); // 5 分鐘 = 5 * 60 * 1000 毫秒
|
|
|
+ }, 300000); // 5 分鐘 = 5 * 60 * 1000 毫秒
|
|
|
};
|
|
|
|
|
|
const startInactivityTimer = () => {
|
|
|
- console.log("startInactivityTimer", timeout);
|
|
|
-
|
|
|
- window.addEventListener("mousemove", resetTimeout);
|
|
|
- window.addEventListener("keydown", resetTimeout);
|
|
|
+ window.addEventListener("mousemove", resetTimeout); // 桌面滑鼠移動
|
|
|
+ window.addEventListener("keydown", resetTimeout); // 桌面按鍵
|
|
|
+ window.addEventListener("touchstart", resetTimeout); // 手機觸碰
|
|
|
+ window.addEventListener("touchmove", resetTimeout); // 手機滑動
|
|
|
+ window.addEventListener("touchend", resetTimeout); // 手機觸碰結束
|
|
|
resetTimeout(); // 初始化計時器
|
|
|
};
|
|
|
|
|
|
const stopInactivityTimer = () => {
|
|
|
clearTimeout(timeout);
|
|
|
+ clearInterval(intervalId); // 停止計時
|
|
|
window.removeEventListener("mousemove", resetTimeout);
|
|
|
window.removeEventListener("keydown", resetTimeout);
|
|
|
+ window.removeEventListener("touchstart", resetTimeout);
|
|
|
+ window.removeEventListener("touchmove", resetTimeout);
|
|
|
+ window.removeEventListener("touchend", resetTimeout);
|
|
|
};
|
|
|
|
|
|
onMounted(() => {
|
|
@@ -2980,7 +3007,9 @@ onBeforeUnmount(() => {
|
|
|
>
|
|
|
<swiper-slide v-for="item in message.body">
|
|
|
<div class="btn-container">
|
|
|
- <button @click="handleObservationDialog(item.value)">
|
|
|
+ <button
|
|
|
+ @click="handleObservationDialog(item.value, item.text)"
|
|
|
+ >
|
|
|
{{ t(item.text) }}
|
|
|
</button>
|
|
|
</div>
|
|
@@ -2998,7 +3027,9 @@ onBeforeUnmount(() => {
|
|
|
>
|
|
|
<swiper-slide v-for="item in message.body">
|
|
|
<div class="btn-container">
|
|
|
- <button @click="handleObservationDialog(item.value)">
|
|
|
+ <button
|
|
|
+ @click="handleObservationDialog(item.value, item.text)"
|
|
|
+ >
|
|
|
{{ t(item.text) }}
|
|
|
</button>
|
|
|
</div>
|
|
@@ -3067,7 +3098,7 @@ onBeforeUnmount(() => {
|
|
|
</v-list>
|
|
|
</v-menu> -->
|
|
|
|
|
|
- <v-menu location="top">
|
|
|
+ <!-- <v-menu location="top">
|
|
|
<template v-slot:activator="{ props }">
|
|
|
<v-btn class="mt-5 py-2" color="primary" dark v-bind="props">
|
|
|
<p class="me-2">{{ t("view_floor_plan") }}</p>
|
|
@@ -3093,7 +3124,7 @@ onBeforeUnmount(() => {
|
|
|
</v-list-item-title>
|
|
|
</v-list-item>
|
|
|
</v-list>
|
|
|
- </v-menu>
|
|
|
+ </v-menu> -->
|
|
|
</div>
|
|
|
|
|
|
<!-- <v-select
|
|
@@ -3345,7 +3376,7 @@ onBeforeUnmount(() => {
|
|
|
>
|
|
|
<swiper-slide v-for="item in message.body">
|
|
|
<div class="btn-container">
|
|
|
- <button @click="findBrand(item.value)">
|
|
|
+ <button @click="findBrand(item.value, item.text)">
|
|
|
{{ t(item.text) }}
|
|
|
</button>
|
|
|
</div>
|
|
@@ -3371,7 +3402,7 @@ onBeforeUnmount(() => {
|
|
|
>
|
|
|
<swiper-slide v-for="item in message.body">
|
|
|
<div class="btn-container">
|
|
|
- <button @click="handleShoppingDialog(item.value)">
|
|
|
+ <button @click="handleShoppingDialog(item.value, item.text)">
|
|
|
{{ t(item.text) }}
|
|
|
</button>
|
|
|
</div>
|
|
@@ -3397,7 +3428,7 @@ onBeforeUnmount(() => {
|
|
|
>
|
|
|
<swiper-slide v-for="item in message.body">
|
|
|
<div class="btn-container">
|
|
|
- <button @click="findBrand(item.value)">
|
|
|
+ <button @click="findBrand(item.value, item.text)">
|
|
|
{{ t(item.text) }}
|
|
|
</button>
|
|
|
</div>
|
|
@@ -3833,7 +3864,14 @@ onBeforeUnmount(() => {
|
|
|
<tr v-for="(row, rowIndex) in menuList" :key="rowIndex">
|
|
|
<td v-for="(item, itemIndex) in row" :key="itemIndex">
|
|
|
<button @click="selectCategory(item.value, itemIndex)">
|
|
|
- <img :src="getImageUrl(item.imgSrc)" alt="" class="icon" />
|
|
|
+ <img
|
|
|
+ :src="getImageUrl(item.imgSrc)"
|
|
|
+ alt=""
|
|
|
+ class="icon"
|
|
|
+ :style="{
|
|
|
+ opacity: item.value === '趣味濾鏡' ? '0.5' : '1',
|
|
|
+ }"
|
|
|
+ />
|
|
|
<p>{{ t(`${item.text}`) }}</p>
|
|
|
</button>
|
|
|
</td>
|