HomeView.vue 28 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207
  1. <script setup>
  2. import { ref, reactive, onMounted, watch, nextTick } from "vue";
  3. import { useRoute } from "vue-router";
  4. // VR
  5. import "aframe";
  6. // i18n
  7. import { useI18n } from "vue-i18n";
  8. // Axios
  9. import axios from "axios";
  10. // Moment
  11. // import moment from "moment";
  12. // Animate
  13. import "animate.css";
  14. // Swiper
  15. // import { Swiper, SwiperSlide } from "swiper/vue";
  16. // import { Navigation, Autoplay } from "swiper/modules";
  17. // import "swiper/css";
  18. // import "swiper/css/navigation";
  19. // Recorder
  20. import Recorder from "recorder-core";
  21. import "recorder-core/src/engine/mp3";
  22. import "recorder-core/src/engine/mp3-engine";
  23. // QR Code
  24. // import { QrcodeStream, QrcodeDropZone, QrcodeCapture } from "vue-qrcode-reader";
  25. // Components
  26. import Navbar from "../components/Navbar.vue";
  27. // import TicketPurchase from "../components/TicketPurchase.vue";
  28. const { t, locale } = useI18n();
  29. const route = useRoute();
  30. const routeParam = ref(null);
  31. onMounted(() => {
  32. // 取得當前路由參數
  33. routeParam.value = route.path.replace("/", "");
  34. console.log("網址參數", routeParam.value);
  35. window.addEventListener("resize", handleResize);
  36. });
  37. const userMessage = ref("");
  38. // let modules = [Navigation, Autoplay]; // Swiper
  39. // AI 客服回覆訊息
  40. let messages = ref([]);
  41. watch(messages.value, (val) => {
  42. console.log("messages", val);
  43. scrollToBottom();
  44. updateMenuHeight();
  45. });
  46. let ad = ref({}); // 彈跳視窗廣告
  47. let qaQuery = reactive([]);
  48. const chatArea = ref(null); // 對話框
  49. // 滾動到對話框底部
  50. const scrollToBottom = () => {
  51. setTimeout(() => {
  52. console.log("chatArea.value", chatArea.value.scrollHeight);
  53. chatArea.value.scrollTo({
  54. top: chatArea.value.scrollHeight,
  55. behavior: "smooth",
  56. });
  57. }, 100);
  58. };
  59. let questionList = ref([]);
  60. // 傳送訊息
  61. async function sendMessage() {
  62. console.log("sendMessage", userMessage.value);
  63. // if (userMessage.value !== "") {
  64. // questionList.value.push({
  65. // q: userMessage.value,
  66. // a: "",
  67. // });
  68. // }
  69. let message = userMessage.value;
  70. // userMessage.value = "";
  71. let url = `https://cmm.ai:8081/answer_with_history?question=${userMessage.value}`;
  72. if (userMessage.value !== "") {
  73. questionList.value.push({
  74. q: message,
  75. a: "",
  76. });
  77. // 使用者訊息
  78. messages.value.push({
  79. label: "text",
  80. author: "user",
  81. body: message,
  82. });
  83. userMessage.value = "";
  84. scrollToBottom();
  85. messages.value.push({
  86. label: "text",
  87. author: "ai",
  88. body: "回覆中…",
  89. });
  90. } else {
  91. return;
  92. }
  93. try {
  94. const response = await axios.post(url, questionList.value);
  95. console.log("response", response);
  96. if (response.status === 200) {
  97. console.log("video_cache", response.data.video_cache);
  98. // 回傳影片
  99. if (response.data.Answer !== "line_oa" && response.data.video_cache) {
  100. if (response.data.video_cache !== "") {
  101. videoSrc.value = response.data.video_cache;
  102. }
  103. // 暫停當前音訊
  104. if (currentAudio.value) {
  105. currentAudio.value.pause();
  106. currentAudio.value.currentTime = 0;
  107. currentAudio.value = null;
  108. }
  109. setTimeout(() => {
  110. isVideoPause.value = true;
  111. videoPlay();
  112. }, 500);
  113. } else if (response.data.Answer !== "line_oa") {
  114. handleTTS(response.data.Answer); // 取得語音回覆
  115. }
  116. messages.value.splice(-1, 1); // 移除回覆中
  117. setTimeout(() => {
  118. // AI 客服回傳訊息
  119. if (response.data.Answer === "line_oa") {
  120. messages.value.push({
  121. label: "line",
  122. author: "ai",
  123. body: "",
  124. });
  125. } else {
  126. messages.value.push({
  127. label: "text",
  128. author: "ai",
  129. body: response.data.Answer,
  130. });
  131. }
  132. scrollToBottom();
  133. }, 300);
  134. questionList.value[questionList.value.length - 1].a =
  135. response.data.Answer;
  136. console.log("questionList.value", questionList.value);
  137. }
  138. console.log("messages.value", messages.value);
  139. } catch (error) {
  140. console.log("error", error);
  141. }
  142. }
  143. let hideMenu = ref(true); // 底部選單
  144. let showInput = ref(true); // 輸入框
  145. function getLang() {
  146. let lang = localStorage.getItem("lang");
  147. let langVal = "";
  148. switch (lang) {
  149. case "zh-tw":
  150. langVal = "ch";
  151. break;
  152. case "en-us":
  153. langVal = "en";
  154. break;
  155. default:
  156. break;
  157. }
  158. return langVal;
  159. }
  160. let showAd = ref(false);
  161. // 取得廣告
  162. async function setAd() {
  163. let lang = getLang();
  164. console.log("lang", lang);
  165. let url = `https://cmm.ai:9101/ad?language=${lang}`;
  166. try {
  167. const response = await axios.get(url);
  168. console.log("Ad response", response);
  169. ad.value = response.data.data.body;
  170. console.log("Ad", ad.value);
  171. } catch (error) {
  172. console.log("error", error);
  173. }
  174. }
  175. let video = ref(null);
  176. function videoPlay() {
  177. video.value.load();
  178. video.value.play();
  179. }
  180. // 底部選單
  181. const menu = ref(null);
  182. const menuHeight = ref(0);
  183. let isRotate = ref(false);
  184. let isLanguagePage = ref(true);
  185. let selectLang = ref("");
  186. let chatLoading = ref(true);
  187. watch(isLanguagePage, (val) => {
  188. if (!val) {
  189. setTimeout(updateMenuHeight, 500);
  190. }
  191. });
  192. const handleResize = () => {
  193. updateMenuHeight();
  194. };
  195. // 取得底部選單高度
  196. const updateMenuHeight = () => {
  197. nextTick(() => {
  198. if (menu.value) {
  199. menuHeight.value = menu.value.clientHeight;
  200. chatLoading.value = false;
  201. }
  202. });
  203. };
  204. function chooseLang(lang) {
  205. console.log("選擇語言:", lang);
  206. selectLang.value = lang;
  207. isLanguagePage.value = false;
  208. locale.value = lang; // i18n locale
  209. localStorage.setItem("lang", lang);
  210. messages.value.push({
  211. label: "text",
  212. author: "ai",
  213. body: t("prologue"),
  214. });
  215. messages.value.push({
  216. label: "text",
  217. author: "ai",
  218. body: t("service"),
  219. });
  220. // 判斷語言修改 title
  221. const language = localStorage.getItem("lang") || "zh-tw";
  222. console.log("language", language);
  223. if (language === "zh-tw") {
  224. document.title = "ChoozMo AI智能客服";
  225. } else if (language === "en-us") {
  226. document.title = "ChoozMo AI Intelligent Customer Service";
  227. }
  228. setAd();
  229. handleClick();
  230. // 影片路徑
  231. loadVideoSources();
  232. let sources;
  233. console.log("videoSources.value", videoSources.value);
  234. if (language === "zh-tw") {
  235. sources = videoSources.value;
  236. } else if (language === "en-us") {
  237. sources = videoSourcesEn.value;
  238. }
  239. const randomIndex = Math.floor(Math.random() * sources.length);
  240. videoSrc.value = sources[randomIndex];
  241. hideAnchorPrologue.value = true;
  242. videoIndex.value = randomIndex + 1;
  243. console.log("messages.value", messages.value);
  244. setTimeout(() => {
  245. showAd.value = true;
  246. videoPlay();
  247. }, 500);
  248. }
  249. // 動態引入視頻文件
  250. const videoSources = ref([]); // 開場白影片(中)
  251. const videoSourcesEn = ref([]); // 開場白影片(英)
  252. const videoMuteSources = ref([]); // 點頭影片(靜音)
  253. const videoSpeakSources = ref([]); // 動嘴型影片
  254. const loadVideoSources = async () => {
  255. videoSources.value = ["https://cmm.ai/chatbot/video/start.mp4"];
  256. videoSourcesEn.value = [""];
  257. videoMuteSources.value = ["https://cmm.ai/chatbot/video/mute.mp4"];
  258. videoSpeakSources.value = ["https://cmm.ai/chatbot/video/speak.mp4"];
  259. };
  260. let videoSrc = ref("");
  261. let hideAnchorPrologue = ref(false); // 顯示開場白 or 點頭影片
  262. let videoIndex = ref(null); // 影片編號
  263. // 計算使用次數
  264. async function handleClick() {
  265. let url = "https://cmm.ai:9101/click";
  266. try {
  267. const response = await axios.get(url);
  268. console.log("Click", response);
  269. } catch (error) {
  270. console.log("error", error);
  271. }
  272. }
  273. let langList = reactive([
  274. {
  275. lang: "中文",
  276. value: "zh-tw",
  277. },
  278. // {
  279. // lang: "English",
  280. // value: "en-us",
  281. // },
  282. // {
  283. // lang: "日本語",
  284. // value: "ja-jp",
  285. // },
  286. // {
  287. // lang: "한국어",
  288. // value: "ko-kr",
  289. // },
  290. ]);
  291. function getImageUrl(name) {
  292. return new URL(`../assets/img/icon/${name}`, import.meta.url).href;
  293. }
  294. let showAnchor = ref(false); // AI 主播影片
  295. let currentAudio = ref(null); // 當前音訊
  296. let audioDuration = ref(null); // 音訊秒數
  297. // 文字轉語音 (TTS)
  298. async function handleTTS(message) {
  299. console.log("handleTTS", message);
  300. let audioLang; // 音訊語言
  301. let lang = localStorage.getItem("lang");
  302. console.log("lang", lang);
  303. switch (lang) {
  304. case "zh-tw":
  305. audioLang = "cmn-TW";
  306. break;
  307. case "en-us":
  308. audioLang = "en-US";
  309. break;
  310. case "ja-jp":
  311. audioLang = "ja-JP";
  312. break;
  313. case "ko-kr":
  314. audioLang = "ko-KR";
  315. break;
  316. default:
  317. break;
  318. }
  319. let url = `https://cmm.ai:9001/gcp/text-to-speech?language_code=${audioLang}&gender=female`;
  320. const formData = new FormData();
  321. formData.append("text", message);
  322. try {
  323. const response = await axios.post(url, formData, { responseType: "blob" });
  324. console.log("TTS response", response);
  325. const blob = new Blob([response.data], { type: "audio/mp3" });
  326. const audioUrl = URL.createObjectURL(blob);
  327. console.log("audioUrl", audioUrl);
  328. cutVideo(); // 剪接影片
  329. // 取得 mp3 音訊秒數
  330. // audio.addEventListener("loadedmetadata", function () {
  331. // audioDuration.value = audio.duration;
  332. // cutVideo();
  333. // });
  334. // 暫停當前音訊
  335. if (currentAudio.value) {
  336. currentAudio.value.pause();
  337. currentAudio.value.currentTime = 0;
  338. }
  339. // 播放音檔
  340. currentAudio.value = new Audio(audioUrl);
  341. } catch (error) {
  342. console.log("error", error);
  343. }
  344. }
  345. let videoLoading = ref(false);
  346. let isAudioPlaying = ref(false); // 音訊播放狀態
  347. // 取得語音回覆 mp4
  348. async function cutVideo() {
  349. videoSrc.value = videoSpeakSources.value[videoIndex.value - 1];
  350. video.value.load(); // 重新讀取影片
  351. // 影片和音訊加載完成後播放
  352. video.value.oncanplay = () => {
  353. setTimeout(() => {
  354. // 監聽音訊播放結束
  355. currentAudio.value.addEventListener("ended", onAudioEnded);
  356. // 監聽音訊播放狀態
  357. currentAudio.value.addEventListener("play", onAudioPlay);
  358. currentAudio.value.addEventListener("pause", onAudioPause);
  359. video.value.play(); // 播放影片
  360. currentAudio.value.currentTime = 0; // 重置時間
  361. currentAudio.value.play(); // 播放音訊
  362. isVideoPause.value = true;
  363. videoLoading.value = false;
  364. }, 500);
  365. };
  366. }
  367. // 音訊結束後暫停影片播放
  368. const onAudioEnded = () => {
  369. video.value.pause();
  370. };
  371. // 判斷音訊是否為播放狀態
  372. const onAudioPlay = () => {
  373. isAudioPlaying.value = true;
  374. console.log("isAudioPlaying.value", isAudioPlaying.value);
  375. };
  376. const onAudioPause = () => {
  377. isAudioPlaying.value = false;
  378. isVideoPause.value = false;
  379. console.log("isAudioPlaying.value", isAudioPlaying.value);
  380. };
  381. const audioURL = ref(null);
  382. const audioFile = ref(null); // 音訊檔案
  383. let recordTime = ref(0); // 錄音時間
  384. let isRecording = ref(false); // 錄音狀態
  385. let timer;
  386. // 語音轉文字
  387. async function handleAudioToText() {
  388. isRecording.value = false;
  389. let audioLang; // 音訊語言
  390. let lang = localStorage.getItem("lang");
  391. console.log("lang", lang);
  392. switch (lang) {
  393. case "zh-tw":
  394. audioLang = "cmn-Hant-TW";
  395. break;
  396. case "en-us":
  397. audioLang = "en-US";
  398. break;
  399. case "ja-jp":
  400. audioLang = "ja-JP";
  401. break;
  402. case "ko-kr":
  403. audioLang = "ko-KR";
  404. break;
  405. default:
  406. break;
  407. }
  408. let url = `https://cmm.ai:9001/gcp/speech-to-text?language_code=${audioLang}`;
  409. const formData = new FormData();
  410. formData.append("file", audioFile.value);
  411. try {
  412. console.log("audioFile.value", audioFile.value);
  413. const response = await axios.post(url, formData);
  414. console.log("語音轉文字 response", response);
  415. // showAnchor.value = false; // 關閉主播視窗
  416. userMessage.value = response.data[0];
  417. // handleTTS(userMessage.value); // 取得語音回覆
  418. if (response.data[0] && response.data[0] !== "") {
  419. sendMessage(); // 傳送使用者訊息
  420. } else {
  421. if (showAnchor.value) {
  422. alert("語音辨識有誤,請重新錄製。");
  423. videoLoading.value = false;
  424. return;
  425. } else {
  426. messages.value.push({
  427. label: "text",
  428. author: "user",
  429. body: "語音辨識有誤,請重新錄製。",
  430. });
  431. }
  432. }
  433. } catch (error) {
  434. console.log("error", error);
  435. }
  436. }
  437. // 語音轉文字 (使用 recorder-core 錄音)
  438. let rec, wave;
  439. // 調用 open 請求錄音權限
  440. let recOpen = function (success) {
  441. rec = Recorder({
  442. type: "mp3",
  443. sampleRate: 16000,
  444. bitRate: 16,
  445. onProcess: function (
  446. buffers,
  447. powerLevel,
  448. bufferDuration,
  449. bufferSampleRate,
  450. newBufferIdx,
  451. asyncEnd
  452. ) {
  453. wave &&
  454. wave.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);
  455. },
  456. });
  457. rec.open(
  458. function () {
  459. if (Recorder.WaveView) wave = Recorder.WaveView({ elem: ".recwave" });
  460. success && success();
  461. },
  462. function (msg, isUserNotAllow) {
  463. // 使用者未授權或不支援
  464. console.log((isUserNotAllow ? "UserNotAllow," : "") + "無法錄音:" + msg);
  465. }
  466. );
  467. };
  468. /** 開始錄音 **/
  469. function recStart() {
  470. togglePause("pause"); // 暫停影片音訊
  471. // 需先呼叫 recOpen() 開啟錄音後才能調用 start、stop 方法
  472. console.log("開始錄音");
  473. recOpen(function () {
  474. isRecording.value = true;
  475. // 開始計時
  476. timer = setInterval(() => {
  477. recordTime.value += 1;
  478. }, 1000);
  479. rec.start();
  480. });
  481. }
  482. /** 結束錄音 **/
  483. function recStop() {
  484. videoLoading.value = true;
  485. rec.stop(
  486. function (blob, duration) {
  487. // 利用 URL 產生本地檔案位址,不用時需要 revokeObjectURL
  488. let localUrl = (window.URL || webkitURL).createObjectURL(blob); // 該 url 只能本地端使用 (例如給 audio.src 進行播放,或是給 a.href download 進行下載)
  489. console.log(blob, localUrl, "時長:" + duration + "ms");
  490. // rec.close(); // 釋放錄音資源 (若不釋放系統或瀏覽器將持續提示在錄音中)
  491. // rec = null;
  492. // 將 Blob 轉換為 File 對象
  493. audioFile.value = new File([blob], "recording.mp3", {
  494. type: "audio/mp3",
  495. });
  496. console.log("audioFile", audioFile.value);
  497. rec.close(); // 釋放錄音資源 (若不釋放系統或瀏覽器將持續提示在錄音中)
  498. rec = null;
  499. if (recordTime.value !== 0) {
  500. handleAudioToText(); // 語音轉文字
  501. } else {
  502. isRecording.value = false;
  503. }
  504. clearInterval(timer); // 清空計時秒數
  505. recordTime.value = 0;
  506. },
  507. function (msg) {
  508. console.log("錄音失敗:" + msg);
  509. rec.close(); // 可以透過 stop 方法的第 3 個參數來自動呼叫 close
  510. rec = null;
  511. }
  512. );
  513. }
  514. let videoCacheData = ref({});
  515. async function getVideoCache(messages) {
  516. let url = `https://cmm.ai:9101/video_cache?client_message=${messages}`;
  517. try {
  518. const response = await axios.post(url);
  519. console.log("response", response);
  520. console.log("response.status", response.status);
  521. if (response.data.state === 200) {
  522. videoCacheData.value = response.data.message[0];
  523. console.log("videoCacheData.value", videoCacheData.value);
  524. return true;
  525. } else {
  526. return false;
  527. }
  528. } catch (error) {
  529. console.log("error", error);
  530. }
  531. }
  532. // 播放 Video Cache
  533. function handleVideoCache() {
  534. console.log("播放 Video Cache", videoCacheData.value);
  535. // AI 客服回傳訊息
  536. messages.value.push({
  537. label: "text",
  538. author: "ai",
  539. body: videoCacheData.value.answer,
  540. });
  541. // 播放 Cache 影片
  542. videoSrc.value = `https://cmm.ai:9101${videoCacheData.value.video_url}`;
  543. video.value.load();
  544. // 清空音訊
  545. if (currentAudio.value) {
  546. currentAudio.value.pause();
  547. currentAudio.value.currentTime = 0;
  548. currentAudio.value = null;
  549. }
  550. video.value.play();
  551. isVideoPause.value = true;
  552. setTimeout(() => {
  553. videoLoading.value = false;
  554. }, 1000);
  555. }
  556. // 關閉主播視窗 (結束錄音)
  557. function closeRec() {
  558. video.value.pause();
  559. isVideoPause.value = true;
  560. recordTime.value = 0;
  561. clearInterval(timer); // 清空計時秒數
  562. if (isRecording.value) {
  563. recStop();
  564. }
  565. showAnchor.value = false;
  566. }
  567. let isVideoPause = ref(true);
  568. // AI 主播影片播放 & 暫停
  569. function togglePause(val) {
  570. if (val === "pause") {
  571. // video.value.pause();
  572. isVideoPause.value = false;
  573. if (video.value) {
  574. video.value.pause();
  575. }
  576. if (currentAudio.value) {
  577. currentAudio.value.pause(); // 暫停音訊
  578. }
  579. } else {
  580. isVideoPause.value = true;
  581. if (video.value) {
  582. video.value.play();
  583. }
  584. if (currentAudio.value) {
  585. currentAudio.value.play(); // 播放音訊
  586. currentAudio.value.addEventListener("ended", onAudioEnded);
  587. }
  588. }
  589. }
  590. // 回傳未收錄問題
  591. async function messageNotInCache(question, answer) {
  592. let url = `https://cmm.ai:9101/message_not_in_cache?question=${question}&answer=${answer}&client_id=0`;
  593. try {
  594. const response = await axios.post(url);
  595. console.log("messageNotInCache response", response);
  596. } catch (error) {
  597. console.log("error", error);
  598. }
  599. }
  600. </script>
  601. <template>
  602. <Navbar />
  603. <main>
  604. <div v-if="isLanguagePage" class="lang-content">
  605. <button
  606. v-for="(item, index) in langList"
  607. :key="index"
  608. @click="chooseLang(item.value)"
  609. class="main-btn"
  610. >
  611. {{ item.lang }}
  612. </button>
  613. </div>
  614. <div v-else class="main-containar">
  615. <div class="video-content">
  616. <video ref="video" preload playsinline>
  617. <source :src="videoSrc" type="video/mp4" />
  618. <!-- <source src="../assets/video/start_1.mp4" type="video/mp4" /> -->
  619. Your browser does not support the video tag.
  620. </video>
  621. <div v-if="videoLoading" class="video-progress">
  622. <v-progress-circular
  623. color="primary"
  624. indeterminate
  625. ></v-progress-circular>
  626. </div>
  627. <button
  628. v-if="isVideoPause"
  629. @click="togglePause('pause')"
  630. class="control-btn"
  631. >
  632. <img src="../assets/img/pause-button.png" alt="" />
  633. </button>
  634. <button v-else @click="togglePause('play')" class="control-btn">
  635. <img src="../assets/img/play-button.png" alt="" />
  636. </button>
  637. <div class="control-item">
  638. <div class="d-flex flex-column align-center">
  639. <audio v-if="audioURL" :src="audioURL" controls></audio>
  640. <!-- <p
  641. v-if="!isRecording"
  642. class="mb-3 text-center"
  643. v-html="t('system_construction')"
  644. ></p> -->
  645. <p v-if="!isRecording" class="mb-3">
  646. {{ t("tap_to_record") }}
  647. </p>
  648. <p v-else class="mb-3">錄音中:{{ recordTime }} 秒</p>
  649. <!-- 錄音按鈕 -->
  650. <v-btn
  651. v-if="!isRecording"
  652. @click="recStart"
  653. icon="mdi-circle"
  654. size="large"
  655. >
  656. <v-icon icon="mdi-circle" color="red" size="large"></v-icon>
  657. </v-btn>
  658. <v-btn
  659. v-else
  660. @click="recStop"
  661. icon="mdi-circle"
  662. size="large"
  663. color="success"
  664. >
  665. <v-icon icon="mdi-square" size="large"></v-icon>
  666. </v-btn>
  667. </div>
  668. </div>
  669. </div>
  670. <div class="chat-content">
  671. <div class="headline">
  672. <h1>{{ t("title") }}</h1>
  673. <!-- <button @click="isRotate = !isRotate">
  674. <img
  675. :class="{ rotate: isRotate }"
  676. src="../assets/img/angles-up-solid.svg"
  677. alt=""
  678. />
  679. </button> -->
  680. </div>
  681. <section
  682. ref="chatArea"
  683. class="chat-area"
  684. :class="{ 'area-open': isRotate, 'hide-menu': hideMenu }"
  685. :style="{
  686. paddingBottom: !hideMenu ? menuHeight + 20 + 'px' : '70px',
  687. }"
  688. >
  689. <div v-for="message in messages" class="message-content">
  690. <p
  691. v-if="message.label === 'text'"
  692. class="message animate__animated"
  693. :class="{
  694. 'message-out': message.author === 'user',
  695. 'message-in': message.author !== 'user',
  696. animate__fadeInRight: message.author === 'user',
  697. animate__fadeInLeft: message.author !== 'user',
  698. }"
  699. v-html="message.body"
  700. ></p>
  701. <div v-if="message.label === 'line'" class="line-item">
  702. <img
  703. src="../assets/img/line_oa_qrcode.png"
  704. alt="Line OA Qrcode"
  705. />
  706. </div>
  707. </div>
  708. </section>
  709. <!-- 底部選單 -->
  710. <div ref="menu" class="menu">
  711. <div class="d-flex align-center position-relative">
  712. <div class="w-100 d-flex align-center justify-center">
  713. <button
  714. v-if="!showInput"
  715. @click="
  716. hideMenu = true;
  717. showInput = true;
  718. "
  719. class="d-flex align-center question-btn"
  720. >
  721. <img
  722. class="me-2"
  723. src="../assets/img/icon/素材-03.png"
  724. alt=""
  725. width="45"
  726. />
  727. {{ t("question") }}
  728. </button>
  729. <!-- 對話輸入框 -->
  730. <form
  731. v-else
  732. @submit.prevent="sendMessage()"
  733. class="chat-inputs"
  734. :class="{ 'd-none': !showInput }"
  735. >
  736. <input
  737. v-model="userMessage"
  738. type="text"
  739. placeholder="Type a message..."
  740. />
  741. <button type="submit" class="submit">
  742. <img
  743. width="20"
  744. src="../assets/img/paper-plane-solid.svg"
  745. alt=""
  746. />
  747. </button>
  748. </form>
  749. </div>
  750. </div>
  751. </div>
  752. </div>
  753. </div>
  754. </main>
  755. </template>
  756. <style scoped lang="scss">
  757. main {
  758. height: 100vh;
  759. display: flex;
  760. flex-direction: column;
  761. justify-content: center;
  762. align-items: center;
  763. /* background-color: var(--sub-color); */
  764. background-color: rgba(0, 0, 0, 0.6);
  765. background-blend-mode: multiply;
  766. background-image: url("@/assets/img/banner.jpg");
  767. background-size: cover;
  768. background-position: center center;
  769. }
  770. .main-btn {
  771. padding: 16px 70px;
  772. font-size: 22px;
  773. font-weight: 600;
  774. border: none;
  775. border-radius: 100px;
  776. letter-spacing: 2px;
  777. color: white;
  778. background-color: var(--main-color);
  779. cursor: pointer;
  780. transition: all 0.3s;
  781. &:hover {
  782. opacity: 0.7;
  783. }
  784. }
  785. .lang-content {
  786. display: flex;
  787. flex-direction: column;
  788. .main-btn {
  789. margin-bottom: 40px;
  790. &:last-child {
  791. margin-bottom: 0;
  792. }
  793. }
  794. }
  795. .main-containar {
  796. width: 100%;
  797. height: 100%;
  798. display: flex;
  799. flex-direction: column;
  800. overflow-x: hidden;
  801. .video-content {
  802. height: 73vh;
  803. display: flex;
  804. justify-content: center;
  805. transform: scale(1);
  806. video {
  807. width: 100%;
  808. height: 100%;
  809. position: fixed;
  810. top: 55px;
  811. left: 0;
  812. right: 0;
  813. z-index: 10;
  814. @media (max-width: 575px) {
  815. top: 15px;
  816. }
  817. @media (max-width: 375px) {
  818. top: 40px;
  819. }
  820. }
  821. .control-btn {
  822. width: 33px;
  823. height: 33px;
  824. display: flex;
  825. align-items: center;
  826. justify-content: center;
  827. position: absolute;
  828. z-index: 50;
  829. top: 80px;
  830. right: 7px;
  831. background: var(--main-color);
  832. border: none;
  833. border-radius: 100px;
  834. img {
  835. width: 25px;
  836. filter: invert(100%) sepia(0%) saturate(0%) hue-rotate(93deg)
  837. brightness(103%) contrast(103%);
  838. }
  839. }
  840. .control-item {
  841. display: flex;
  842. justify-content: end;
  843. flex-direction: column;
  844. z-index: 10;
  845. position: absolute;
  846. bottom: 30px;
  847. @media (max-width: 375px) {
  848. height: 29vh;
  849. }
  850. p {
  851. color: #fff;
  852. font-size: 0.875rem;
  853. letter-spacing: 1px;
  854. text-shadow: 1px 1px 2px #333;
  855. }
  856. }
  857. }
  858. .chat-content {
  859. width: 100%;
  860. height: 60vh;
  861. margin-top: -20px;
  862. position: relative;
  863. z-index: 100;
  864. letter-spacing: 1px;
  865. border-radius: 10px 10px 0 0;
  866. .headline {
  867. display: flex;
  868. align-items: center;
  869. justify-content: space-between;
  870. padding: 10px 25px;
  871. border-radius: 15px 15px 0 0;
  872. background-color: var(--main-color);
  873. h1 {
  874. font-size: 1.125rem;
  875. font-weight: 500;
  876. color: white;
  877. @media (max-width: 375px) {
  878. font-size: 0.875rem;
  879. }
  880. }
  881. button {
  882. padding-top: 3px;
  883. display: flex;
  884. align-items: center;
  885. border: none;
  886. background-color: transparent;
  887. cursor: pointer;
  888. img {
  889. width: 25px;
  890. height: 20px;
  891. transition: all 0.3s;
  892. transform: rotate(0deg);
  893. &.rotate {
  894. transform: rotate(180deg);
  895. }
  896. }
  897. }
  898. }
  899. .chat-area {
  900. display: flex;
  901. flex-direction: column;
  902. background: var(--sub-color);
  903. height: 40vh;
  904. padding: 0 1em 2em;
  905. overflow-x: hidden;
  906. overflow-y: auto;
  907. transition: all 0.3s;
  908. &.area-open {
  909. height: 75vh;
  910. @media (max-width: 400px) {
  911. height: 67vh;
  912. }
  913. }
  914. }
  915. .message-content {
  916. display: flex;
  917. flex-direction: column;
  918. .message {
  919. max-width: 45%;
  920. border-radius: 20px;
  921. padding: 0.5em 1.2em;
  922. white-space: pre-line;
  923. &:first-child {
  924. margin-top: 0;
  925. }
  926. @media (max-width: 600px) {
  927. max-width: 80%;
  928. font-size: 0.875rem;
  929. }
  930. &.message-out {
  931. margin-left: auto;
  932. background: var(--bg-grey);
  933. color: white;
  934. }
  935. &.message-in {
  936. margin-right: auto;
  937. background: #f1f0f0;
  938. color: black;
  939. }
  940. &.message-in,
  941. &.message-out {
  942. margin-top: 20px;
  943. }
  944. }
  945. .line-item {
  946. margin-top: 20px;
  947. img {
  948. max-width: 300px;
  949. }
  950. }
  951. }
  952. .chat-inputs {
  953. width: 100%;
  954. display: flex;
  955. padding: 13px 20px;
  956. background-color: white;
  957. input {
  958. width: 100%;
  959. border: none;
  960. &:focus-visible {
  961. outline: none;
  962. }
  963. }
  964. button {
  965. border: none;
  966. background: white;
  967. cursor: pointer;
  968. &:hover {
  969. img {
  970. opacity: 0.8;
  971. }
  972. }
  973. img {
  974. padding-top: 5px;
  975. transition: all 0.3s;
  976. filter: invert(16%) sepia(40%) saturate(4127%) hue-rotate(286deg)
  977. brightness(108%) contrast(122%);
  978. }
  979. }
  980. ::placeholder {
  981. font-size: 1rem;
  982. font-weight: 500;
  983. color: var(--main-color);
  984. opacity: 1; /* Firefox */
  985. }
  986. ::-ms-input-placeholder {
  987. /* Edge 12 -18 */
  988. color: var(--sub-color);
  989. }
  990. }
  991. }
  992. }
  993. .video-progress {
  994. position: absolute;
  995. left: 50%;
  996. top: 40%;
  997. z-index: 100;
  998. transform: translate(-50%, -50%);
  999. }
  1000. .v-expansion-panel-text {
  1001. font-size: 0.875rem;
  1002. line-height: 1.7;
  1003. }
  1004. /* 底部選單 */
  1005. .menu {
  1006. position: fixed;
  1007. z-index: 300;
  1008. left: 0;
  1009. bottom: 0px;
  1010. right: 0;
  1011. color: var(--text-color);
  1012. background-color: white;
  1013. &.hide-menu table {
  1014. height: 0;
  1015. z-index: -1;
  1016. }
  1017. .icon {
  1018. width: 80px;
  1019. @media (max-width: 767px) {
  1020. width: 50px;
  1021. }
  1022. }
  1023. .question-btn {
  1024. color: var(--main-color);
  1025. font-size: 1.25rem;
  1026. font-weight: 500;
  1027. letter-spacing: 0.1rem;
  1028. @media (max-width: 414px) {
  1029. font-size: 1rem;
  1030. }
  1031. }
  1032. }
  1033. </style>