text-to-chart.js 87 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132213321342135213621372138213921402141214221432144214521462147214821492150215121522153215421552156215721582159216021612162216321642165216621672168216921702171217221732174217521762177217821792180218121822183218421852186218721882189219021912192219321942195219621972198219922002201220222032204220522062207220822092210221122122213221422152216221722182219222022212222222322242225222622272228222922302231223222332234223522362237223822392240224122422243224422452246224722482249225022512252225322542255225622572258225922602261226222632264226522662267226822692270227122722273227422752276227722782279228022812282228322842285228622872288228922902291229222932294229522962297229822992300230123022303230423052306230723082309231023112312231323142315231623172318231923202321232223232324232523262327232823292330233123322333233423352336233723382339234023412342234323442345234623472348234923502351235223532354235523562357235823592360236123622363236423652366236723682369237023712372237323742375237623772378237923802381238223832384238523862387238823892390239123922393239423952396239723982399240024012402240324042405240624072408240924102411241224132414241524162417241824192420242124222423242424252426242724282429243024312432243324342435243624372438243924402441244224432444244524462447244824492450245124522453245424552456245724582459246024612462246324642465246624672468246924702471247224732474247524762477247824792480248124822483248424852486248724882489249024912492249324942495249624972498249925002501250225032504250525062507250825092510251125122513251425152516251725182519252025212522252325242525252625272528252925302531253225332534253525362537253825392540254125422543254425452546254725482549255025512552255325542555255625572558255925602561256225632564256525662567256825692570257125722573257425752576257725782579258025812582258325842585258625872588258925902591259225932594259525962597259825992600260126022603260426052606260726082609261026112612261326142615261626172618261926202621262226232624262526262627262826292630263126322633263426352636263726382639264026412642264326442645264626472648264926502651265226532654265526562657265826592660266126622663266426652666266726682669267026712672267326742675267626772678267926802681268226832684268526862687268826892690269126922693269426952696269726982699270027012702270327042705270627072708270927102711271227132714271527162717271827192720272127222723272427252726272727282729273027312732273327342735273627372738273927402741274227432744274527462747274827492750275127522753275427552756275727582759276027612762276327642765276627672768276927702771277227732774277527762777277827792780278127822783278427852786278727882789279027912792279327942795279627972798279928002801280228032804280528062807280828092810281128122813281428152816281728182819282028212822282328242825282628272828282928302831283228332834283528362837283828392840284128422843284428452846284728482849285028512852285328542855285628572858285928602861286228632864
  1. console.log('text-to-chart');
  2. var accessToken = localStorage.getItem("access_token");
  3. console.log(accessToken);
  4. if (accessToken === null) {
  5. alert('請先登入')
  6. window.location.href = "./login.html";
  7. }
  8. var usernameEmail = localStorage.getItem("username");
  9. var unit_table_value
  10. var obj = {
  11. _text: ''
  12. };
  13. // let username = usernameEmail.split('@')[0];
  14. Chart.register(ChartDataLabels);
  15. var userNameSpan = document.getElementById("userName");
  16. var lineYfontSize = document.getElementById("lineYfontSize");
  17. userNameSpan.textContent = usernameEmail;
  18. var inputField = document.getElementById("keyword_data");
  19. var inputField_compare = document.getElementById("keyword_data_compare");
  20. var compare_box = document.getElementById("compare");
  21. var sendButton = document.getElementById("send_data");
  22. var send_data_compare = document.getElementById("data_compare");
  23. var canvasPng = document.getElementById("textToChart");
  24. var img_box = document.getElementById("img_box");
  25. var sidebar = document.getElementById('style_setting');
  26. var input_text_value;
  27. var chartType = 'line';
  28. var unit = '';
  29. var myChart = null;
  30. var chartColor = '#912B2B';
  31. var chartColorBar = '#'
  32. var displayXaxes = true;
  33. var ctx;
  34. var chartlabels;
  35. var chartdata;
  36. var XfontSizeValue = 18;
  37. var YfontSizeValue = 18;
  38. var TitlefontSizeValue = 32;
  39. var pieFontSize = 14;
  40. var PieMaxWidth = 100;
  41. var chart_bgimg_url = "url(./img/bg06.webp)";
  42. var vocab = [];
  43. var chartDiv = document.getElementById('chartdiv');
  44. var dataFirstValue;
  45. var dataLastValue;
  46. let cancelToken;
  47. let source;
  48. $('#chartdiv').hide();
  49. var clickExample = document.getElementById('example_box')
  50. document.addEventListener('DOMContentLoaded', function () {
  51. axios.get("https://cmm.ai:8080/vocab")
  52. .then(response => {
  53. // apiHideLoading();
  54. console.log(response);
  55. vocab = response.data;
  56. })
  57. .catch(error => {
  58. console.error('發生錯誤:', error);
  59. });
  60. });
  61. inputField.addEventListener('click', function () {
  62. // 確保輸入框處於焦點狀態
  63. inputField.focus();
  64. // 在控制台中顯示焦點狀態
  65. console.log('Input field focused');
  66. });
  67. document.addEventListener('DOMContentLoaded', function () {
  68. // const input = document.getElementById('stockInput');
  69. const suggestions = document.getElementById('suggestions');
  70. inputField.addEventListener('input', function () {
  71. const query = this.value.trim();
  72. suggestions.innerHTML = ''; // 清空之前的建議
  73. console.log('query', query);
  74. if (query) {
  75. // const filteredVocab = vocab.filter(item => item.toLowerCase().includes(query));
  76. const queryTerms = query.split(/\s+/); // 分割輸入的內容
  77. const lastQueryTerm = queryTerms[queryTerms.length - 1];
  78. const filteredVocab = vocab.filter(item => {
  79. // 對每個分割後的單詞進行匹配,只要其中有一個匹配成功就返回 true
  80. return item.toLowerCase().includes(lastQueryTerm.toLowerCase());
  81. });
  82. if (filteredVocab.length > 0) {
  83. suggestions.style.border = '1px solid #ccc';
  84. suggestions.style.display = 'block'; // 顯示建議容器
  85. } else {
  86. suggestions.style.border = 'none';
  87. suggestions.style.display = 'none'; // 隱藏建議容器
  88. }
  89. console.log(queryTerms[queryTerms.length - 1])
  90. filteredVocab.forEach(item => {
  91. const div = document.createElement('div');
  92. div.classList.add('autocomplete-suggestion');
  93. div.textContent = item;
  94. // div.addEventListener('click', function () {
  95. // inputField.value = item;
  96. // suggestions.innerHTML = ''; // 清空建議
  97. // suggestions.style.border = 'none';
  98. // suggestions.style.display = 'none'; // 隱藏建議容器
  99. // });
  100. // console.log('queryTerms', queryTerms);
  101. // console.log('filteredVocab', filteredVocab);
  102. div.addEventListener('click', function () {
  103. queryTerms.forEach((term, index) => {
  104. const regex = new RegExp(term, 'gi');
  105. const match = regex.exec(inputField.value);
  106. // console.log(queryTerms[queryTerms.length - 1]);
  107. // console.log(filteredVocab, queryTerms[queryTerms.length - 1])
  108. if (index === queryTerms.length - 1) {
  109. if (match) {
  110. var start = match.index;
  111. var end = start + match[0].length;
  112. console.log(start, end)
  113. const textBefore = inputField.value.substring(0, start);
  114. const textAfter = inputField.value.substring(end, inputField.value.length);
  115. console.log('start', start, 'end', end, 'textBefore', textBefore, 'textAfter', textAfter);
  116. // 捕獲被選中的部分並替換它
  117. const selectedText = inputField.value.substring(start, end);
  118. console.log('selectedText', selectedText);
  119. const newText = textBefore + item + textAfter.replace(selectedText, '');
  120. console.log('newText', newText);
  121. inputField.value = newText;
  122. // inputField.value = textBefore + item.replace(selectedText, '') + textAfter;
  123. suggestions.innerHTML = ''; // 清空建議
  124. suggestions.style.border = 'none';
  125. }
  126. }
  127. });
  128. // const start = inputField.selectionStart;
  129. // const end = inputField.selectionEnd;
  130. });
  131. suggestions.appendChild(div);
  132. });
  133. } else {
  134. suggestions.style.border = 'none';
  135. suggestions.style.display = 'none'; // 隱藏建議容器
  136. }
  137. });
  138. });
  139. document.addEventListener('DOMContentLoaded', function () {
  140. // const input = document.getElementById('stockInput');
  141. const suggestions_compare = document.getElementById('suggestions_compare');
  142. inputField_compare.addEventListener('input', function () {
  143. const query = this.value.trim();
  144. suggestions_compare.innerHTML = ''; // 清空之前的建議
  145. console.log('query', query);
  146. if (query) {
  147. // const filteredVocab = vocab.filter(item => item.toLowerCase().includes(query));
  148. const queryTerms = query.split(/\s+/); // 分割輸入的內容
  149. const lastQueryTerm = queryTerms[queryTerms.length - 1];
  150. const filteredVocab = vocab.filter(item => {
  151. // 對每個分割後的單詞進行匹配,只要其中有一個匹配成功就返回 true
  152. return item.toLowerCase().includes(lastQueryTerm.toLowerCase());
  153. });
  154. if (filteredVocab.length > 0) {
  155. suggestions_compare.style.border = '1px solid #ccc';
  156. suggestions_compare.style.display = 'block'; // 顯示建議容器
  157. } else {
  158. suggestions_compare.style.border = 'none';
  159. suggestions_compare.style.display = 'none'; // 隱藏建議容器
  160. }
  161. console.log(queryTerms[queryTerms.length - 1])
  162. filteredVocab.forEach(item => {
  163. const div = document.createElement('div');
  164. div.classList.add('autocomplete-suggestion');
  165. div.textContent = item;
  166. // div.addEventListener('click', function () {
  167. // inputField.value = item;
  168. // suggestions.innerHTML = ''; // 清空建議
  169. // suggestions.style.border = 'none';
  170. // suggestions.style.display = 'none'; // 隱藏建議容器
  171. // });
  172. // console.log('queryTerms', queryTerms);
  173. // console.log('filteredVocab', filteredVocab);
  174. div.addEventListener('click', function () {
  175. queryTerms.forEach((term, index) => {
  176. const regex = new RegExp(term, 'gi');
  177. const match = regex.exec(inputField_compare.value);
  178. // console.log(queryTerms[queryTerms.length - 1]);
  179. // console.log(filteredVocab, queryTerms[queryTerms.length - 1])
  180. if (index === queryTerms.length - 1) {
  181. if (match) {
  182. var start = match.index;
  183. var end = start + match[0].length;
  184. console.log(start, end)
  185. const textBefore = inputField_compare.value.substring(0, start);
  186. const textAfter = inputField_compare.value.substring(end, inputField_compare.value.length);
  187. console.log('start', start, 'end', end, 'textBefore', textBefore, 'textAfter', textAfter);
  188. // 捕獲被選中的部分並替換它
  189. const selectedText = inputField_compare.value.substring(start, end);
  190. console.log('selectedText', selectedText);
  191. const newText = textBefore + item + textAfter.replace(selectedText, '');
  192. console.log('newText', newText);
  193. inputField_compare.value = newText;
  194. // inputField.value = textBefore + item.replace(selectedText, '') + textAfter;
  195. suggestions_compare.innerHTML = ''; // 清空建議
  196. suggestions_compare.style.border = 'none';
  197. }
  198. }
  199. });
  200. // const start = inputField.selectionStart;
  201. // const end = inputField.selectionEnd;
  202. });
  203. suggestions_compare.appendChild(div);
  204. });
  205. } else {
  206. suggestions_compare.style.border = 'none';
  207. suggestions_compare.style.display = 'none'; // 隱藏建議容器
  208. }
  209. });
  210. });
  211. // 監聽輸入框的鍵盤事件
  212. // document.getElementById("keyword_data").addEventListener("keyup", function (event) {
  213. // // 判斷是否按下 Enter 鍵 (key code: 13)
  214. // if (event.keyCode === 13) {
  215. // // 觸發送出按鈕的點擊事件
  216. // document.getElementById("send_data").click();
  217. // }
  218. // });
  219. // 定義按鈕點擊事件處理函數
  220. function sendButtonClickHandler() {
  221. // inputField.value = "";
  222. console.log(inputField.value);
  223. var input_text_value = inputField.value;
  224. compare_box.style.display = "none";
  225. $('#chartdiv').hide();
  226. sidebar.classList.remove('show');
  227. sidebar.classList.add('hidden');
  228. // 在這裡添加你希望在按下按鈕時執行的其他代碼
  229. get_data(input_text_value);
  230. }
  231. sendButton.addEventListener("click", function () {
  232. sendButtonClickHandler();
  233. inputField.value = ''
  234. chartTypeBtn.style.display = "none";
  235. });
  236. let labels2 = [];
  237. let data2 = [];
  238. // 漲幅or單價
  239. const checkbox1 = document.getElementById('checkbox1');
  240. const checkbox2 = document.getElementById('checkbox2');
  241. var compare_type_input = document.querySelector('.checkbox-group')
  242. var compare_type = "";
  243. checkbox1.addEventListener('change', () => {
  244. if (checkbox1.checked) {
  245. checkbox2.checked = false;
  246. compare_type = '漲幅'
  247. console.log(compare_type);
  248. }
  249. });
  250. checkbox2.addEventListener('change', () => {
  251. if (checkbox2.checked) {
  252. checkbox1.checked = false;
  253. compare_type = '單價'
  254. console.log(compare_type);
  255. }
  256. });
  257. send_data_compare.addEventListener("click", function () {
  258. console.log('比較', inputField_compare.value);
  259. const stockName = inputField_compare.value;
  260. axios
  261. .get(`https://cmm.ai:8080/get_data_from_date?start_date=${dataFirstValue}&end_date=${dataLastValue}&stock_name=${inputField_compare.value}`)
  262. .then((response) => {
  263. console.log(response);
  264. var compare_data = response.data.data;
  265. labels2 = []; // Clear previous labels
  266. data2 = []; // Clear previous data
  267. if (compare_type === "") {
  268. alert('請輸入比較類型');
  269. return
  270. }
  271. // console.log('比較送出後 ', chartType)
  272. if (response.data.data === "無法產生圖表") {
  273. alert('無法產生圖表')
  274. return
  275. }
  276. if (chartType === 'line') {
  277. const colors = [
  278. 'rgba(130, 163, 63, 1)',
  279. 'rgba(47, 72, 123, 1)',
  280. '#df8c49',
  281. '#72598f',
  282. '#489fb6',
  283. '#99b0d5'
  284. ];
  285. let colorIndex = 0;
  286. for (var i = 0; i < compare_data.length; i++) {
  287. var item = compare_data[i];
  288. // 遍历当前项的属性
  289. for (var key in item) {
  290. // 如果属性名不在 labels 数组中,并且属性值不是对象,则将属性名添加到 labels 数组中
  291. if (typeof item[key] === 'string') {
  292. labels2.push(item[key]);
  293. }
  294. // 如果属性值是数值类型,则将其添加到 data 数组中
  295. if (typeof item[key] === 'number') {
  296. data2.push(item[key]);
  297. }
  298. }
  299. }
  300. function getRandomColor() {
  301. const randomIndex = Math.floor(Math.random() * colors.length);
  302. const selectedColor = colors[randomIndex];
  303. colors.splice(randomIndex, 1); // 从数组中移除已选颜色
  304. return selectedColor;
  305. }
  306. const normalize = (prices) => {
  307. const initialPrice = prices[0];
  308. // return prices.map(price => (price - initialPrice) / initialPrice * 100);
  309. return prices.map(price => {
  310. // Calculate the normalized price
  311. const normalized = (price - initialPrice) / initialPrice * 100;
  312. // Round to 2 decimal places and convert back to a number
  313. return parseFloat(normalized.toFixed(2));
  314. });
  315. };
  316. var stock2Normalized;
  317. var stock1Normalized;
  318. if (compare_type === '漲幅') {
  319. console.log('漲幅')
  320. stock2Normalized = normalize(data2);
  321. stock1Normalized = normalize(data);
  322. unitInput.value = '%';
  323. unit_value.textContent = '%'
  324. } else {
  325. console.log('單價')
  326. unitInput.value = unitInput.value;
  327. unit_value.textContent = unitInput.value;
  328. stock2Normalized = data2;
  329. stock1Normalized = data;
  330. }
  331. const color = getRandomColor();
  332. const newDataset = {
  333. label: stockName,
  334. backgroundColor: color,
  335. pointRadius: 0,
  336. pointHoverRadius: 0,
  337. pointBorderColor: '#fff',
  338. borderColor: color,
  339. data: stock2Normalized,
  340. fill: false
  341. };
  342. colorIndex = (colorIndex + 1) % colors.length;
  343. myChart.data.datasets.push(newDataset);
  344. myChart.data.datasets[0].data = stock1Normalized;
  345. console.log('多筆資料', myChart.data.datasets);
  346. // console.log('比較', myChart.options.plugins.legend);
  347. myChart.options.plugins.legend.display = true;
  348. // myChart.defaults.global.legend = {
  349. // display: true,
  350. // position: 'top', // 默认图例位置为顶部
  351. // labels: {
  352. // fontColor: 'rgb(255, 99, 132)', // 默认图例标签颜色
  353. // fontSize: 14 // 默认图例标签字体大小
  354. // }
  355. // };
  356. myChart.update();
  357. inputField_compare.value = '';
  358. compare_type_input.style.display = 'none'
  359. } else if (chartType === 'bar') {
  360. }
  361. })
  362. .catch((error) =>
  363. console.log(error)
  364. );
  365. });
  366. // sendButton.addEventListener("click", function () {
  367. // console.log(inputField.value);
  368. // input_text_value = inputField.value;
  369. // // if (myChart) {
  370. // // console.log('已存在')
  371. // // data = [];
  372. // // labels = [];
  373. // // myChart.removePlugin(Chart.pluginService.getPlugin('afterDraw'));
  374. // // myChart.destroy();
  375. // // }
  376. // get_data(input_text_value);
  377. // });
  378. var lastKeyPressTime = 0;
  379. var enterCount = 0;
  380. var ENTER_THRESHOLD = 500; // 設定連續按下 Enter 的時間閾值(毫秒)
  381. inputField.addEventListener("keyup", function (event) {
  382. // 判斷是否按下 Enter 鍵 (key code: 13)
  383. if (event.key === "Enter") {
  384. var currentTime = new Date().getTime();
  385. // 計算與上一次按鍵按下時間的差值
  386. var timeDiff = currentTime - lastKeyPressTime;
  387. lastKeyPressTime = currentTime;
  388. // 如果兩次 Enter 鍵按下的時間差小於閾值,則增加計數器
  389. if (timeDiff <= ENTER_THRESHOLD) {
  390. enterCount++;
  391. // 如果計數器為2,則觸發 API 請求並重置計數器
  392. if (enterCount === 2) {
  393. sendButtonClickHandler();
  394. enterCount = 0;
  395. }
  396. } else {
  397. // 如果時間差大於閾值,則重置計數器
  398. enterCount = 0;
  399. }
  400. } else {
  401. // 如果按下的不是 Enter 鍵,則重置計數器
  402. enterCount = 0;
  403. }
  404. });
  405. function number_format(number, decimals, dec_point, thousands_sep) {
  406. // * example: number_format(1234.56, 2, ',', ' ');
  407. // * return: '1 234,56'
  408. number = (number + '').replace(',', '').replace(' ', '');
  409. var n = !isFinite(+number) ? 0 : +number,
  410. prec = !isFinite(+decimals) ? 0 : Math.abs(decimals),
  411. sep = (typeof thousands_sep === 'undefined') ? ',' : thousands_sep,
  412. dec = (typeof dec_point === 'undefined') ? '.' : dec_point,
  413. s = '',
  414. toFixedFix = function (n, prec) {
  415. var k = Math.pow(10, prec);
  416. return '' + Math.round(n * k) / k;
  417. };
  418. // Fix for IE parseFloat(0.55).toFixed(0) = 0;
  419. s = (prec ? toFixedFix(n, prec) : '' + Math.round(n)).split('.');
  420. if (s[0].length > 3) {
  421. s[0] = s[0].replace(/\B(?=(?:\d{3})+(?!\d))/g, sep);
  422. }
  423. if ((s[1] || '').length < prec) {
  424. s[1] = s[1] || '';
  425. s[1] += new Array(prec - s[1].length + 1).join('0');
  426. }
  427. return s.join(dec);
  428. }
  429. var labels = [];
  430. var data = [];
  431. function apiLoading() {
  432. document.getElementById('chartDataLoading').style.display = 'block';
  433. }
  434. function apiHideLoading() {
  435. document.getElementById('chartDataLoading').style.display = 'none';
  436. }
  437. const DISPLAY = true;
  438. const BORDER = true;
  439. const CHART_AREA = true;
  440. const TICKS = true;
  441. var labels = [];
  442. var data = [];
  443. var dataArray;
  444. function generateChart(dataArray) {
  445. if (dataArray.length > 10) {
  446. lineYfontSize.style.display = 'none'
  447. } else {
  448. lineYfontSize.style.display = 'block'
  449. }
  450. // 创建一个空数组来存储 labels 和 data
  451. console.log(dataArray)
  452. // 遍历数据数组
  453. for (var i = 0; i < dataArray.length; i++) {
  454. var item = dataArray[i];
  455. // 遍历当前项的属性
  456. for (var key in item) {
  457. // 如果属性名不在 labels 数组中,并且属性值不是对象,则将属性名添加到 labels 数组中
  458. if (typeof item[key] === 'string') {
  459. labels.push(item[key]);
  460. } else {
  461. if (key === 'x') {
  462. labels.push(String(item[key]));
  463. }
  464. }
  465. if (key === 'y') {
  466. data.push(item[key]);
  467. }
  468. // 如果属性值是数值类型,则将其添加到 data 数组中
  469. // if (typeof item[key] === 'number') {
  470. // data.push(item[key]);
  471. // }
  472. }
  473. }
  474. console.log(data);
  475. dataFirstValue = labels[0];
  476. dataLastValue = labels[labels.length - 1]
  477. console.log('dataFirstValue', dataFirstValue);
  478. console.log('dataLastValue', dataLastValue);
  479. if (myChart) {
  480. console.log('已存在')
  481. myChart.destroy();
  482. }
  483. if (chartType === "doughnut") {
  484. // createChartDoughnut(chartType, data, labels);
  485. // createChartDoughnut2(data, labels)
  486. } else {
  487. createChart(chartType, data, labels);
  488. }
  489. // console.log(data);
  490. console.log(labels);
  491. }
  492. var amChartInstance;
  493. function extractAndGenerateChart(dataArray) {
  494. for (var i = 0; i < dataArray.length; i++) {
  495. var item = dataArray[i];
  496. // 遍历当前项的属性
  497. for (var key in item) {
  498. // 如果属性名不在 labels 数组中,并且属性值不是对象,则将属性名添加到 labels 数组中
  499. if (typeof item[key] === 'string') {
  500. labels.push(item[key]);
  501. }
  502. // 如果属性值是数值类型,则将其添加到 data 数组中
  503. if (typeof item[key] === 'number') {
  504. data.push(item[key]);
  505. }
  506. }
  507. }
  508. // console.log('key', key);
  509. generateBarChart(key, dataArray);
  510. }
  511. // 圓餅圖
  512. function generatePieChart(dataArray, pieFontSize, PieMaxWidth) {
  513. compare_box.style.display = "none";
  514. $('#chartdiv').show();
  515. bgImgelement.style.backgroundImage = chart_bgimg_url;
  516. // chartDiv.style.width = "100%"
  517. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  518. // Themes begin
  519. am4core.useTheme(am4themes_animated);
  520. // Themes end
  521. // Create chart instance
  522. var chart = am4core.create("chartdiv", am4charts.PieChart);
  523. chart.data = dataArray;
  524. chart.innerRadius = am4core.percent(50);
  525. // Add and configure Series
  526. var pieSeries = chart.series.push(new am4charts.PieSeries());
  527. pieSeries.dataFields.value = "y";
  528. pieSeries.dataFields.category = "x";
  529. pieSeries.slices.template.stroke = am4core.color("#fff");
  530. pieSeries.slices.template.strokeWidth = 2;
  531. pieSeries.slices.template.strokeOpacity = 1;
  532. // Configure labels
  533. pieSeries.labels.template.wrap = true;
  534. console.log('PieMaxWidth', PieMaxWidth);
  535. pieSeries.labels.template.maxWidth = PieMaxWidth;
  536. pieSeries.labels.template.truncate = false;
  537. pieSeries.labels.template.fontSize = 14;
  538. if (typeof PieMaxWidth === 'string') {
  539. console.log('The variable is a string.');
  540. } else {
  541. console.log('The variable is a number.');
  542. }
  543. // Configure label text to wrap and show percentages
  544. // pieSeries.labels.template.adapter.add("textOutput", function (text, target) {
  545. // if (target.dataItem && target.dataItem.values.value.percent) {
  546. // return "[font-size: 14px]" + target.dataItem.category + ": " + target.dataItem.values.value.percent.toFixed(1) + "%";
  547. // }
  548. // return text;
  549. // });
  550. pieSeries.labels.template.adapter.add("textOutput", function (text, target) {
  551. if (target.dataItem && target.dataItem.values.value.percent) {
  552. return "[font-size: " + pieFontSize + "px]" + target.dataItem.category + ": " + target.dataItem.values.value.percent.toFixed(1) + "%";
  553. }
  554. return text;
  555. });
  556. pieSeries.colors.list = [
  557. am4core.color('rgb(171, 51, 49)'),
  558. am4core.color('rgb(34, 83, 149)'),
  559. am4core.color('rgb(79, 148, 65)'),
  560. am4core.color('rgb(217, 195, 105)'),
  561. am4core.color('rgb(142, 124, 180)'),
  562. am4core.color('rgb(211, 183, 144)'),
  563. am4core.color('rgb(83, 84, 84)'),
  564. am4core.color('rgb(229, 147, 152)')
  565. ];
  566. // Ensure labels have background
  567. // pieSeries.labels.template.background.fillOpacity = 1;
  568. console.log(pieSeries.labels.template)
  569. // Set label background color to match corresponding slice color
  570. pieSeries.labels.template.adapter.add("background.fill", function (fill, target) {
  571. return target.dataItem.slice.fill;
  572. });
  573. // This creates initial animation
  574. pieSeries.hiddenState.properties.opacity = 1;
  575. pieSeries.hiddenState.properties.endAngle = -90;
  576. pieSeries.hiddenState.properties.startAngle = -90;
  577. // console.log('調位置')
  578. downloadButton.style.display = "inline-block";
  579. chartTypeBtn.style.display = "inline-block";
  580. }
  581. // var colorEven='#288D97';
  582. // var colorOdd='#427D7E'
  583. function adjustColorBrightness(color, amount) {
  584. let usePound = false;
  585. if (color[0] == "#") {
  586. color = color.slice(1);
  587. usePound = true;
  588. }
  589. let num = parseInt(color, 16);
  590. let r = (num >> 16) + amount;
  591. let g = ((num >> 8) & 0x00FF) + amount;
  592. let b = (num & 0x0000FF) + amount;
  593. if (r > 255) r = 255;
  594. else if (r < 0) r = 0;
  595. if (g > 255) g = 255;
  596. else if (g < 0) g = 0;
  597. if (b > 255) b = 255;
  598. else if (b < 0) b = 0;
  599. return (usePound ? "#" : "") + (r << 16 | g << 8 | b).toString(16).padStart(6, '0');
  600. }
  601. // 柱狀圖
  602. function generateBarChart(key, dataArray) {
  603. console.log('generateBarChart', dataArray)
  604. chartTypeBtn.style.display = "inline-block";
  605. compare_box.style.display = "none";
  606. $('#chartdiv').show();
  607. // var colorEven = document.getElementById('colorEven').value;
  608. // var colorOdd = document.getElementById('colorOdd').value;
  609. var baseColor = document.getElementById('borderColorInputBar').value;
  610. var categoryAxisFzValue = document.getElementById('XfontSizeBar').value;
  611. var valueAxisFzValue = document.getElementById('YfontSizeBar').value;
  612. var colorEven = am4core.color(baseColor);
  613. var colorOdd = adjustColorBrightness(baseColor, -20);
  614. chartDiv.style.fontFamily = "Arial, sans-serif";
  615. chartDiv.style.fontWeight = "900";
  616. bgImgelement.style.backgroundImage = chart_bgimg_url;
  617. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  618. console.log('labelDependent', labelDependent);
  619. // Themes begin
  620. am4core.useTheme(am4themes_animated);
  621. // Themes end
  622. // Create chart instance
  623. var chart = am4core.create("chartdiv", am4charts.XYChart3D);
  624. chart.data = dataArray;
  625. chart.angle = 30; // 圖表角度
  626. chart.depth = 25; // 圖表深度
  627. let categoryAxis = chart.xAxes.push(new am4charts.CategoryAxis());
  628. categoryAxis.dataFields.category = "x";
  629. categoryAxis.renderer.labels.template.rotation = 0;
  630. categoryAxis.renderer.labels.template.hideOversized = false;
  631. categoryAxis.renderer.minGridDistance = 20; // 間距
  632. // categoryAxis.renderer.labels.template.horizontalCenter = "left";
  633. // categoryAxis.renderer.labels.template.verticalCenter = "middle";
  634. // categoryAxis.tooltip.label.rotation = 90;
  635. // categoryAxis.tooltip.label.horizontalCenter = "right";
  636. // categoryAxis.tooltip.label.verticalCenter = "middle";
  637. // label 換行
  638. let label = categoryAxis.renderer.labels.template;
  639. label.wrap = true;
  640. if (dataArray.length > 5) {
  641. console.log('資料長度5')
  642. label.maxWidth = 60;
  643. } else if (dataArray.length < 7) {
  644. label.maxWidth = 70;
  645. } else {
  646. label.maxWidth = 100;
  647. console.log('資料長度<5')
  648. }
  649. // x軸字體大小
  650. categoryAxis.renderer.labels.template.fontSize = categoryAxisFzValue;
  651. categoryAxis.renderer.cellEndLocation = 0.5; // 減少值可以增加間距
  652. let valueAxis = chart.yAxes.push(new am4charts.ValueAxis());
  653. valueAxis.title.text = labelDependent; // Y軸
  654. valueAxis.title.fontWeight = "bold";
  655. // y軸字體大小
  656. valueAxis.renderer.labels.template.fontSize = valueAxisFzValue;
  657. // Create series
  658. var series = chart.series.push(new am4charts.ColumnSeries3D());
  659. series.dataFields.valueY = 'y';
  660. series.dataFields.categoryX = "x";
  661. series.name = "y";
  662. series.tooltipText = "{categoryX}: [bold]{valueY}[/]";
  663. series.columns.template.fillOpacity = .8;
  664. let columnTemplate = series.columns.template;
  665. columnTemplate.strokeWidth = 2;
  666. columnTemplate.strokeOpacity = 1;
  667. columnTemplate.stroke = am4core.color("#FFFFFF");
  668. // columnTemplate.adapter.add("fill", function (fill, target) {
  669. // return chart.colors.getIndex(target.dataItem.index);
  670. // })
  671. // 設置基數和偶數顏色
  672. columnTemplate.adapter.add("fill", function (fill, target) {
  673. return target.dataItem.index % 2 === 0 ? am4core.color(colorEven) : am4core.color(colorOdd);
  674. });
  675. columnTemplate.adapter.add("stroke", function (stroke, target) {
  676. return chart.colors.getIndex(target.dataItem.index);
  677. })
  678. // // 加上白色線條
  679. // series.columns.template.events.on("validated", function (event) {
  680. // let column = event.target;
  681. // let whiteLine = column.createChild(am4core.Line);
  682. // whiteLine.stroke = am4core.color("#ffffff");
  683. // whiteLine.strokeWidth = 1.3; // 線條寬度
  684. // whiteLine.strokeOpacity = 0.5;// 透明度
  685. // whiteLine.x1 = column.pixelWidth / 1;
  686. // whiteLine.y1 = 0;
  687. // whiteLine.x2 = column.pixelWidth / 1;
  688. // whiteLine.y2 = column.pixelHeight;
  689. // });
  690. chart.cursor = new am4charts.XYCursor();
  691. chart.cursor.lineX.strokeOpacity = 0;
  692. chart.cursor.lineY.strokeOpacity = 0;
  693. categoryAxis.renderer.grid.template.strokeOpacity = 0; // 隱藏 X 軸
  694. // valueAxis.renderer.grid.template.strokeOpacity = 0; // 隱藏 Y 軸
  695. // valueAxis.renderer.line.strokeOpacity = 0;
  696. // LineSeries 底部線條
  697. var lineSeries = chart.series.push(new am4charts.LineSeries());
  698. lineSeries.dataFields.valueY = "bottom";
  699. lineSeries.dataFields.categoryX = "0";
  700. lineSeries.stroke = am4core.color("#FFFFFF")
  701. lineSeries.strokeWidth = 2;
  702. lineSeries.strokeOpacity = 1
  703. var bottomData = [];
  704. for (var i = 0; i < dataArray.length; i++) {
  705. bottomData.push({ "x": dataArray[i].x, "bottom": 0 });
  706. }
  707. lineSeries.data = bottomData;
  708. downloadButton.style.display = "inline-block";
  709. chartTypeBtn.style.display = "inline-block";
  710. chartTypeBtn.style.opacity = "1";
  711. // 底部線條
  712. // let rectElements = document.querySelectorAll("rect");
  713. // // let widthValue = 0;
  714. // let fifthRectElement = rectElements[3];
  715. // // fifthRectElement.style.position = "relative";
  716. // setTimeout(() => {
  717. // let width = fifthRectElement.getAttribute("width");
  718. // // widthValue = parseFloat(width);
  719. // console.log('rectElements', rectElements);
  720. // console.log('fifthRectElement', fifthRectElement);
  721. // console.log('width', width);
  722. // let line = document.createElement("div");
  723. // line.style.position = "absolute";
  724. // line.style.bottom = "162px";
  725. // line.style.right = "40px";
  726. // line.style.width = `${parseFloat(width)}px`;
  727. // line.style.height = "2px";
  728. // line.style.backgroundColor = "#000000";
  729. // document.getElementById("chartdiv").appendChild(line);
  730. // }, 10)
  731. // end am4core.ready()
  732. }
  733. function handleClick(element) {
  734. const Clickvalue = element.dataset.value;
  735. console.log(Clickvalue); // 123
  736. get_data(Clickvalue);
  737. }
  738. // 圖表類型
  739. var chartTypeBtn = document.getElementById("chartType");
  740. chartTypeBtn.addEventListener('change', function () {
  741. const selectedValue = this.value;
  742. console.log('選擇的圖表類型:', selectedValue);
  743. obj.text = selectedValue;
  744. // 在此处添加处理选定图表类型的代码
  745. if (selectedValue === 'line') {
  746. // 处理折线图
  747. console.log('處理折線圖', dataArray);
  748. $('#chartdiv').hide();
  749. chartType = 'line';
  750. if (myChart) {
  751. data = [];
  752. labels = [];
  753. myChart.destroy();
  754. }
  755. generateChart(dataArray);
  756. } else if (selectedValue === 'bar') {
  757. chartType = 'bar';
  758. $('#textToChart').hide();
  759. // 处理柱状图
  760. console.log('處理柱狀圖', dataArray);
  761. extractAndGenerateChart(dataArray);
  762. } else if (selectedValue === 'doughnut') {
  763. chartType = 'doughnut';
  764. console.log('處理圓餅圖', dataArray);
  765. unit_value.textContent = "";
  766. $('#textToChart').hide();
  767. generatePieChart(dataArray, pieFontSize, PieMaxWidth)
  768. // 处理圆饼图
  769. }
  770. });
  771. var downloadButton = document.getElementById("downloadButton");
  772. // downloadButton.style.display = 'none';
  773. // 添加點擊事件監聽器
  774. downloadButton.addEventListener('click', function () {
  775. setTimeout(function () {
  776. // html2canvas(document.getElementById('data_chart_box')).then(function (canvas) {
  777. // // // 獲取圖片資料 URL
  778. // // var imageData = canvasPng.toDataURL("image/png");
  779. // // // 創建下載連結
  780. // // var link = document.createElement('a');
  781. // // link.download = 'chart.png'; // 下載的檔案名稱
  782. // // link.href = imageData; // 圖片資料 URL
  783. // // // 模擬點擊下載連結
  784. // // link.click();
  785. // // 創建一個&lt;a&gt;元素
  786. // const link = document.createElement('a');
  787. // // 將 Canvas 轉換為 Data URL
  788. // link.href = canvas.toDataURL('image/png');
  789. // // 設置下載文件名
  790. // link.download = 'chart.png';
  791. // // 模擬點擊下載
  792. // link.click();
  793. // });
  794. html2canvas(document.getElementById('data_chart_box'), {
  795. backgroundColor: null, // 確保背景透明
  796. scale: 2,
  797. useCORS: true,
  798. scrollX: 0,
  799. scrollY: 0
  800. }).then(function (canvas) {
  801. // 創建一個 <a> 元素
  802. const link = document.createElement('a');
  803. // 將 Canvas 轉換為 Data URL
  804. link.href = canvas.toDataURL('image/png');
  805. // 設置下載文件名
  806. link.download = 'chart.png';
  807. // 模擬點擊下載
  808. link.click();
  809. });
  810. }, 1000); // 等待 1 秒钟
  811. });
  812. var pieChartColors = [
  813. 'rgb(171, 51, 49)',
  814. 'rgb(34, 83, 149)',
  815. 'rgb(79, 148, 65)',
  816. 'rgb(217, 195, 105)',
  817. 'rgb(142, 124, 180)',
  818. 'rgb(211, 183, 144)',
  819. 'rgb(83, 84, 84)',
  820. 'rgb(229, 147, 152)'
  821. ];
  822. function createChartDoughnut2(data, labels) {
  823. const chartData = {
  824. type: 'outlabeledPie',
  825. data: {
  826. labels: labels,
  827. datasets: [
  828. {
  829. data: data,
  830. borderWidth: borderWidthValue,
  831. fill: false,
  832. backgroundColor: pieChartColors,
  833. },
  834. ],
  835. },
  836. options: {
  837. plugins: {
  838. legend: false,
  839. outlabels: {
  840. text: '%l %p',
  841. color: 'white',
  842. stretch: 35,
  843. font: {
  844. resizable: true,
  845. family: 'Arial',
  846. size: 48,
  847. weight: 'bold',
  848. style: 'italic',
  849. },
  850. },
  851. },
  852. },
  853. };
  854. const devicePixelRatio = 4;
  855. var chartUrl = `https://quickchart.io/chart?c=${encodeURIComponent(JSON.stringify(chartData))}&devicePixelRatio=${devicePixelRatio}&f=png&fontsize=20`;
  856. // 获取目标 div 元素
  857. const targetDiv = document.getElementById('data_chartJs_box');
  858. // 创建一个 img 元素来显示图表
  859. const chartImg = document.createElement('img');
  860. chartImg.src = chartUrl;
  861. chartImg.classList.add('outlabeled_img');
  862. // 清空目标 div 元素
  863. targetDiv.innerHTML = '';
  864. // 将图表 img 元素插入到目标 div 中
  865. targetDiv.appendChild(chartImg);
  866. console.log(chart_bgimg_url);
  867. bgImgelement.style.backgroundImage = chart_bgimg_url;
  868. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  869. }
  870. // 版型二源餅圖
  871. // // 创建 Chart.js 图表
  872. function createChartDoughnut(chartType, data, labels) {
  873. $('#textToChart').show();
  874. canvasPng.style.maxWidth = '450px';
  875. canvasPng.style.maxHeight = '450px';
  876. var windowWidth = window.innerWidth || document.documentElement.clientWidth;
  877. var canvasWidth = Math.min(500, windowWidth); // 限制最大宽度为 500px
  878. canvasPng.style.width = canvasWidth + 'px'; // 设置 canvas 宽度
  879. canvasPng.style.marginLeft = 'auto'; // 左外边距自动计算
  880. canvasPng.style.marginRight = 'auto'; // 右外边距自动计算
  881. var ctx = document.getElementById("textToChart").getContext('2d');
  882. myChart = new Chart(ctx, {
  883. type: chartType,
  884. data: {
  885. labels: labels,
  886. datasets: [{
  887. strokeColor: "rgba(220,220,220,1)",
  888. StrokeWidth: 5,
  889. data: data,
  890. borderWidth: borderWidthValue,
  891. pointRadius: 0,
  892. pointBorderColor: '#fff',
  893. pointBorderWidth: 3,
  894. pointHoverRadius: 0,
  895. fill: false,
  896. backgroundColor: pieChartColors,
  897. // barPercentage: 0.5,
  898. // categoryPercentage: 0.5
  899. }]
  900. },
  901. options: {
  902. // responsive: true, // 啟用響應式
  903. maintainAspectRatio: true,
  904. responsive: true,
  905. aspectRatio: 1,
  906. plugins: {
  907. legend: {
  908. display: false,
  909. },
  910. datalabels: {
  911. color: '#fff', // 数据标签的颜色
  912. anchor: 'end', // 数据标签的位置
  913. align: 'start', // 数据标签的对齐方式
  914. font: {
  915. size: 14, // 数据标签的字体大小
  916. },
  917. }
  918. },
  919. scales: {
  920. x: {
  921. display: false, // 显示 x 轴
  922. grid: {
  923. display: false,
  924. lineWidth: 5,
  925. color: '#584B3D'
  926. },
  927. ticks: {
  928. textStrokeColor: '#fff',
  929. textStrokeWidth: 5,
  930. font: {
  931. display: false,
  932. family: 'Arial', // 设置字体
  933. size: XfontSizeValue, // 设置字体大小
  934. weight: 'bold', // 设置字体粗细
  935. fontColor: 'rgba(255,255,255,0.8)', // X-axis font color
  936. shadowColor: 'rgba(0, 0, 0, 0.5)', // Shadow color
  937. shadowBlur: 10, // Shadow blur level
  938. shadowOffsetX: 5, // Horizontal shadow offset
  939. shadowOffsetY: 5
  940. },
  941. },
  942. },
  943. y: {
  944. display: false,
  945. beginAtZero: false,
  946. grid: {
  947. display: false,
  948. lineWidth: 5,
  949. color: '#584B3D'
  950. },
  951. ticks: {
  952. textStrokeColor: '#fff',
  953. textStrokeWidth: 5,
  954. font: {
  955. family: 'Arial', // 設置字體
  956. size: XfontSizeValue, // 設置字體大小
  957. weight: 'bold', // 設置字體粗細
  958. color: 'rgba(255,255,255,0.8)', // Y-axis 字體顏色
  959. shadowColor: 'rgba(0, 0, 0, 0.5)', // 陰影顏色
  960. shadowBlur: 10, // 陰影模糊級別
  961. shadowOffsetX: 5, // 水平陰影偏移
  962. shadowOffsetY: 5 // 垂直陰影偏移
  963. },
  964. }
  965. }
  966. },
  967. layout: {
  968. padding: {
  969. left: 50, // 调整图表左边距
  970. right: 50, // 调整图表右边距
  971. top: 10,
  972. bottom: 10,
  973. }
  974. },
  975. },
  976. });
  977. console.log(chart_bgimg_url);
  978. bgImgelement.style.backgroundImage = chart_bgimg_url;
  979. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  980. }
  981. var chartJsBox = document.getElementById('data_chartJs_box');
  982. function createTable(dataArray, unit, Label_dependent_variable, Label_independent_variable,) {
  983. $('#chartdiv').hide();
  984. $('#textToChart').hide();
  985. chartTitle.style.padding = "15px 0px 30px 30px";
  986. bgImgelement.style.paddingTop = "10px";
  987. var table = document.createElement('table');
  988. table.classList.add('table', 'table_template', 'dynamic-table'); // 添加 Bootstrap 的 table 样式,如果你在使用 Bootstrap
  989. // 创建表头
  990. var thead = table.createTHead();
  991. var headerRow = thead.insertRow();
  992. // Object.keys(dataArray[0]).forEach(function (key) {
  993. // var header = headerRow.insertCell();
  994. // header.textContent = key;
  995. // });
  996. var yearHeader = headerRow.insertCell();
  997. yearHeader.textContent = Label_independent_variable;
  998. var avgPriceHeader = headerRow.insertCell();
  999. avgPriceHeader.textContent = Label_dependent_variable;
  1000. // 创建表格主体
  1001. var tbody = table.createTBody();
  1002. dataArray.forEach(function (item) {
  1003. console.log(item)
  1004. var row = tbody.insertRow();
  1005. // var cell1 = row.insertCell();
  1006. // cell1.textContent = item.year;
  1007. // cell1.classList.add('year-class'); // 为年份单元格添加类
  1008. // var cell2 = row.insertCell();
  1009. Object.keys(item).forEach(function (key, index) {
  1010. var cell = row.insertCell();
  1011. if (index === 0) { // If it's the first cell, add 'year-class'
  1012. cell.textContent = item[key];
  1013. cell.classList.add('year-class');
  1014. } else { // Otherwise, add 'average-class'
  1015. if (key === 'Average_Close') {
  1016. // cell.textContent = item[key].toFixed(2) + unit;
  1017. cell.innerHTML = item[key].toFixed(2) + '<span class="unit_table">' + unit + '</span>';
  1018. } else {
  1019. // cell.textContent = item[key];
  1020. // cell.textContent = item[key].toFixed(2) + unit;
  1021. cell.innerHTML = item[key].toFixed(2) + '<span class="unit_table">' + unit + '</span>';
  1022. }
  1023. cell.classList.add('average-class');
  1024. }
  1025. });
  1026. // console.log(item.avg_price.toFixed(2))
  1027. // cell.textContent = item[key].toFixed(2) + unit;
  1028. // cell2.classList.add('average-class'); // 为年份单元格添加类
  1029. });
  1030. // 获取包含表格的 div 元素
  1031. bgImgelement.style.backgroundImage = chart_bgimg_url;
  1032. img_box.style.bottom = '60px';
  1033. img_box.style.right = '5px';
  1034. document.getElementById('img_box_url').style.width = "180px";
  1035. // chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  1036. // 将表格添加到指定的 div 元素中
  1037. chartJsBox.appendChild(table);
  1038. unit_table_value = document.querySelectorAll('.unit_table');
  1039. downloadButton.style.display = "inline-block";
  1040. chartTypeBtn.style.display = "inline-block";
  1041. chartTypeBtn.style.opacity = "0";
  1042. }
  1043. function getDatalabelsConfig(dataLength) {
  1044. if (dataLength < 10) {
  1045. return {
  1046. formatter: function (value, context) {
  1047. return value + ' unit'; // Add unit if data length is less than 10
  1048. },
  1049. textStrokeColor: '#fff',
  1050. textStrokeWidth: 5,
  1051. color: '#584B3D',
  1052. font: {
  1053. size: YfontSizeValue,
  1054. },
  1055. anchor: 'end',
  1056. align: 'end',
  1057. };
  1058. } else {
  1059. return null; // No datalabels if data length is 10 or more
  1060. }
  1061. }
  1062. var pointRadiusValue = 0;
  1063. var pointHoverRadiusValue = 0;
  1064. function getDatalabelsConfig(isEnabled) {
  1065. if (!isEnabled) {
  1066. return null;
  1067. }
  1068. return data.length <= 15 ? {
  1069. formatter: function (value, context) {
  1070. return value;
  1071. },
  1072. textStrokeColor: '#fff',
  1073. textStrokeWidth: 5,
  1074. color: '#584B3D',
  1075. font: {
  1076. size: YfontSizeValue,
  1077. },
  1078. anchor: 'end',
  1079. align: 'top'
  1080. } : null;
  1081. }
  1082. // 版型一折線圖
  1083. // // 创建 Chart.js 图表
  1084. function createChart(chartType, data, labels) {
  1085. $('#textToChart').show();
  1086. canvasPng.style.maxWidth = '';
  1087. canvasPng.style.maxHeight = '';
  1088. console.log('繪製圖表', data, labels);
  1089. if (data.length < 15) {
  1090. pointRadiusValue = 5;
  1091. pointHoverRadiusValue = 8;
  1092. togglePoints.checked = true;
  1093. }
  1094. const datalabelsConfig = data.length <= 15 ? {
  1095. formatter: function (value, context) {
  1096. // if (context.dataIndex === 0) {
  1097. // return '';
  1098. // }
  1099. // if (typeof value === 'number' && value.toString().includes('.')) {
  1100. // value = value.toFixed(4);
  1101. // }
  1102. // return value;
  1103. return value;
  1104. },
  1105. textStrokeColor: '#fff',
  1106. textStrokeWidth: 5,
  1107. color: '#584B3D',
  1108. font: {
  1109. size: YfontSizeValue,
  1110. },
  1111. anchor: 'end',
  1112. align: 'top'
  1113. } : null;
  1114. ctx = document.getElementById("textToChart");
  1115. myChart = new Chart(ctx, {
  1116. type: chartType,
  1117. data: {
  1118. labels: labels,
  1119. datasets: [{
  1120. label: stocknumber,
  1121. strokeColor: "rgba(220,220,220,1)",
  1122. StrokeWidth: 5,
  1123. data: data,
  1124. borderWidth: borderWidthValue,
  1125. pointRadius: pointRadiusValue,
  1126. pointBorderColor: '#fff',
  1127. pointBorderWidth: 3,
  1128. pointHoverRadius: pointHoverRadiusValue,
  1129. fill: false,
  1130. backgroundColor: chartColor,
  1131. borderColor: chartColor,
  1132. // barPercentage: 0.5,
  1133. // categoryPercentage: 0.5
  1134. }]
  1135. },
  1136. options: {
  1137. // responsive: true, // 啟用響應式
  1138. maintainAspectRatio: false,
  1139. plugins: {
  1140. chart3d: {
  1141. enabled: true, // 啟用 3D 插件
  1142. alpha: 25, // 圖表繞 x 軸的旋轉角度
  1143. beta: 25, // 圖表繞 y 軸的旋轉角度
  1144. depth: 15 // 圖表的深度
  1145. },
  1146. datalabels: datalabelsConfig,
  1147. backgroundColor: 'transparent',
  1148. // datalabels: {
  1149. // formatter: function (value, context) {
  1150. // // 添加单位
  1151. // return value;
  1152. // },
  1153. // textStrokeColor: '#fff',
  1154. // textStrokeWidth: 5,
  1155. // color: '#584B3D',
  1156. // font: {
  1157. // size: YfontSizeValue,
  1158. // },
  1159. // anchor: 'end',
  1160. // align: 'end',
  1161. // },
  1162. // datalabels: null,
  1163. // datalabels: data.length <= 10 ? {
  1164. // formatter: function (value, context) {
  1165. // // 添加单位
  1166. // return value + ' ' + unit;
  1167. // },
  1168. // textStrokeColor: '#fff',
  1169. // textStrokeWidth: 5,
  1170. // color: '#584B3D',
  1171. // font: {
  1172. // size: YfontSizeValue,
  1173. // },
  1174. // anchor: 'end',
  1175. // align: 'end',
  1176. // } : null,
  1177. customLabels: null,
  1178. legend: {
  1179. display: false,
  1180. },
  1181. },
  1182. scales: {
  1183. x: {
  1184. display: displayXaxes, // 显示 x 轴
  1185. grid: {
  1186. display: false,
  1187. lineWidth: 5,
  1188. color: '#584B3D'
  1189. },
  1190. ticks: {
  1191. textStrokeColor: '#fff',
  1192. textStrokeWidth: 5,
  1193. font: {
  1194. display: false,
  1195. family: 'Arial', // 设置字体
  1196. size: XfontSizeValue, // 设置字体大小
  1197. weight: 'bold', // 设置字体粗细
  1198. fontColor: 'rgba(255,255,255,0.8)', // X-axis font color
  1199. shadowColor: 'rgba(0, 0, 0, 0.5)', // Shadow color
  1200. shadowBlur: 10, // Shadow blur level
  1201. shadowOffsetX: 5, // Horizontal shadow offset
  1202. shadowOffsetY: 5
  1203. },
  1204. },
  1205. },
  1206. y: {
  1207. display: true,
  1208. beginAtZero: false,
  1209. grid: {
  1210. grid: {
  1211. display: true, // 显示 y 轴网格线
  1212. lineWidth: 1, // 网格线宽度
  1213. color: '#C0C5CC' // 网格线颜色
  1214. },
  1215. },
  1216. ticks: {
  1217. textStrokeColor: '#fff',
  1218. textStrokeWidth: 5,
  1219. font: {
  1220. family: 'Arial', // 設置字體
  1221. size: XfontSizeValue, // 設置字體大小
  1222. weight: 'bold', // 設置字體粗細
  1223. color: 'rgba(255,255,255,0.8)', // Y-axis 字體顏色
  1224. shadowColor: 'rgba(0, 0, 0, 0.5)', // 陰影顏色
  1225. shadowBlur: 10, // 陰影模糊級別
  1226. shadowOffsetX: 5, // 水平陰影偏移
  1227. shadowOffsetY: 5 // 垂直陰影偏移
  1228. },
  1229. }
  1230. }
  1231. },
  1232. layout: {
  1233. padding: {
  1234. left: 50, // 调整图表左边距
  1235. right: 50, // 调整图表右边距
  1236. top: 40,
  1237. bottom: 10,
  1238. }
  1239. },
  1240. },
  1241. });
  1242. console.log('第一次', myChart.options.legend)
  1243. console.log(chart_bgimg_url);
  1244. bgImgelement.style.backgroundImage = chart_bgimg_url;
  1245. chartWhitelement.style.background = "rgba(255,255,255,0.5)";
  1246. downloadButton.style.display = "inline-block";
  1247. chartTypeBtn.style.display = "inline-block";
  1248. chartTypeBtn.style.opacity = "1";
  1249. compare_box.style.display = "flex";
  1250. }
  1251. // createChart(chartType);
  1252. // createChart(chartType)
  1253. const ChartOptions = document.querySelectorAll('input[name="ChartOptions"]');
  1254. ChartOptions.forEach(button => {
  1255. button.addEventListener('click', function () {
  1256. // console.log(this.value);
  1257. chartType = this.value;
  1258. console.log(chartType);
  1259. if (myChart !== null) {
  1260. myChart.destroy();
  1261. data = [];
  1262. labels = [];
  1263. }
  1264. if (chartType === 'line') {
  1265. $('#chartdiv').hide();
  1266. generateChart(dataArray);
  1267. } else {
  1268. $('#textToChart').hide();
  1269. extractAndGenerateChart(dataArray);
  1270. }
  1271. // createChart(chartType, data, labels);
  1272. });
  1273. });
  1274. // 表格字體
  1275. var fontSizeInput = document.getElementById('table_fontSize');
  1276. // 添加事件监听器
  1277. fontSizeInput.addEventListener('input', function () {
  1278. // 获取输入框的值
  1279. var fontSize = fontSizeInput.value;
  1280. // 获取动态创建的表格元素
  1281. var dynamicTable = document.querySelector('.dynamic-table');
  1282. // 如果表格存在,则调整其字体大小
  1283. if (dynamicTable) {
  1284. dynamicTable.style.fontSize = fontSize + 'px';
  1285. }
  1286. });
  1287. // 圓餅圖字體
  1288. var pieFontSizeInput = document.getElementById('pie_fontSize');
  1289. // 添加事件监听器
  1290. pieFontSizeInput.addEventListener('input', function () {
  1291. // 获取输入框的值
  1292. pieFontSize = pieFontSizeInput.value;
  1293. console.log('圓餅圖字體', pieFontSize);
  1294. if (pieFontSize) {
  1295. generatePieChart(dataArray, pieFontSize, PieMaxWidth);
  1296. } else {
  1297. alert("Please enter a valid font size.");
  1298. }
  1299. // 如果表格存在,则调整其字体大小
  1300. // if (dynamicTable) {
  1301. // dynamicTable.style.fontSize = fontSize + 'px';
  1302. // }
  1303. });
  1304. document.getElementById('PieMaxWidthRange').addEventListener('input', function () {
  1305. PieMaxWidth = Number(document.getElementById('PieMaxWidthRange').value);
  1306. console.log('圓餅圖', PieMaxWidth)
  1307. generatePieChart(dataArray, pieFontSize, PieMaxWidth);
  1308. });
  1309. // 是否顯示point
  1310. var togglePoints = document.getElementById('togglePoints');
  1311. togglePoints.addEventListener('change', function () {
  1312. console.log('checked', "change")
  1313. // 遍历所有数据集,应用相应的pointRadius和pointHoverRadius
  1314. myChart.data.datasets.forEach(function (dataset) {
  1315. if (togglePoints.checked) {
  1316. dataset.pointRadius = 5;
  1317. dataset.pointHoverRadius = 8;
  1318. } else {
  1319. dataset.pointRadius = 0;
  1320. dataset.pointHoverRadius = 0;
  1321. }
  1322. });
  1323. myChart.update();
  1324. });
  1325. // 是否顯示數值
  1326. var toggleValue = document.getElementById('toggleValue');
  1327. let datalabelsConfig = getDatalabelsConfig(toggleValue.checked);
  1328. toggleValue.addEventListener('change', function () {
  1329. console.log('切換數值')
  1330. // 根据复选框状态更新 datalabelsConfig
  1331. datalabelsConfig = getDatalabelsConfig(this.checked);
  1332. // 更新图表
  1333. myChart.options.plugins.datalabels = datalabelsConfig;
  1334. myChart.update();
  1335. });
  1336. // 是否顯示單位
  1337. var toggleUnit = document.getElementById('toggleUnit');
  1338. toggleUnit.addEventListener('change', function () {
  1339. if (this.checked) {
  1340. unit_value_box.style.display = 'block';
  1341. } else {
  1342. unit_value_box.style.display = 'none';
  1343. }
  1344. });
  1345. // 單位
  1346. // =========================
  1347. var unitInput = document.getElementById('unit_data');
  1348. var unit_value = document.querySelector('.unit');
  1349. var unit_value_box = document.querySelector('#unit_box');
  1350. // console.log('單位', unitInput)
  1351. unitInput.addEventListener('input', function () {
  1352. // unitInput.value = unit_value.textContent;
  1353. if (chartType === 'table') {
  1354. // unit_table_value.textContent = this.value;
  1355. unit_table_value.forEach(span => {
  1356. span.textContent = this.value;
  1357. });
  1358. } else {
  1359. unit_value.textContent = this.value;
  1360. }
  1361. });
  1362. // ==========================
  1363. // y軸字體大小-折線圖
  1364. // ==========================
  1365. var YfontSize = document.getElementById('YfontSize');
  1366. YfontSize.addEventListener('input', function () {
  1367. YfontSizeValue = YfontSize.value;
  1368. if (myChart) {
  1369. myChart.destroy();
  1370. }
  1371. createChart(chartType, data, labels);
  1372. });
  1373. // ==========================
  1374. // x軸字體大小-折線圖
  1375. // ==========================
  1376. var XfontSize = document.getElementById('XfontSize');
  1377. XfontSize.addEventListener('input', function () {
  1378. XfontSizeValue = XfontSize.value;
  1379. console.log(XfontSizeValue)
  1380. if (myChart) {
  1381. myChart.destroy();
  1382. }
  1383. createChart(chartType, data, labels);
  1384. });
  1385. // ==========================
  1386. // x軸字體大小-柱狀圖
  1387. // ==========================
  1388. var XfontSizeBar = document.getElementById('XfontSizeBar');
  1389. XfontSizeBar.addEventListener('input', function () {
  1390. extractAndGenerateChart(dataArray);
  1391. });
  1392. // ==========================
  1393. // y軸字體大小-柱狀圖
  1394. // ==========================
  1395. var YfontSizeBar = document.getElementById('YfontSizeBar');
  1396. YfontSizeBar.addEventListener('input', function () {
  1397. extractAndGenerateChart(dataArray);
  1398. });
  1399. // ==========================
  1400. // 線條顏色-柱狀圖
  1401. // =========================
  1402. const colorInputBar = document.getElementById('borderColorInputBar');
  1403. // 添加事件监听器,当颜色选择发生变化时触发
  1404. colorInputBar.addEventListener('input', function () {
  1405. extractAndGenerateChart(dataArray);
  1406. });
  1407. // =========================
  1408. // 線條粗細
  1409. // =========================
  1410. var rangeInput = document.getElementById('borderWidthRange');
  1411. var rangeInputSpan = document.getElementById('borderWidthRangeValue');
  1412. var borderWidthValue;
  1413. rangeInputSpan.textContent = rangeInput.value;
  1414. // 添加事件监听器,当滑动条的值发生改变时更新显示的值
  1415. rangeInput.addEventListener('input', function () {
  1416. // 获取滑动条的当前值
  1417. borderWidthValue = rangeInput.value;
  1418. console.log(borderWidthValue);
  1419. rangeInputSpan.textContent = borderWidthValue;
  1420. if (myChart) {
  1421. myChart.destroy();
  1422. }
  1423. createChart(chartType, data, labels);
  1424. // myChart.data.datasets[0].borderWidth = borderWidthValue;
  1425. });
  1426. // =========================
  1427. // 線條顏色-折線圖
  1428. // =========================
  1429. const colorInput = document.getElementById('borderColorInput');
  1430. // 添加事件监听器,当颜色选择发生变化时触发
  1431. colorInput.addEventListener('input', function () {
  1432. // 获取当前选择的颜色值
  1433. chartColor = colorInput.value;
  1434. if (myChart) {
  1435. myChart.destroy();
  1436. }
  1437. console.log(chartColor);
  1438. createChart(chartType, data, labels);
  1439. });
  1440. // =========================
  1441. // 模板樣式
  1442. // =========================
  1443. // JavaScript
  1444. document.addEventListener("DOMContentLoaded", function () {
  1445. // 獲取所有帶有 templateImg class 的圖像元素
  1446. var templateImgs = document.querySelectorAll('.templateImg');
  1447. // 為每個圖像元素添加點擊事件監聽器
  1448. templateImgs.forEach(function (img) {
  1449. img.addEventListener('click', function () {
  1450. // 在點擊時印出圖像元素的 value 屬性值
  1451. console.log(this.getAttribute('value'));
  1452. });
  1453. });
  1454. });
  1455. // =========================
  1456. var bgImgelement = document.getElementById('data_chart_box');
  1457. var chartWhitelement = document.getElementById('data_chartJs_box');
  1458. // 背景樣式
  1459. // =========================
  1460. // JavaScript
  1461. document.addEventListener("DOMContentLoaded", function () {
  1462. // 獲取所有帶有 templateImg class 的圖像元素
  1463. var bgImgs = document.querySelectorAll('.bgImg');
  1464. // 為每個圖像元素添加點擊事件監聽器
  1465. bgImgs.forEach(function (img) {
  1466. img.addEventListener('click', function () {
  1467. // 在點擊時印出圖像元素的 value 屬性值
  1468. console.log(this.getAttribute('value'));
  1469. var bgImgUrl = this.getAttribute('value');
  1470. chart_bgimg_url = 'url(' + bgImgUrl + ')';
  1471. if (bgImgUrl === './img/bg07.webp') {
  1472. chartTitle.style.color = "#ffff";
  1473. chartTitle.style.padding = '0px'
  1474. }
  1475. else {
  1476. chartTitle.style.color = "#1c2d6d";
  1477. }
  1478. if (bgImgUrl === "") {
  1479. chartWhitelement.style.background = "";
  1480. console.log('無背景');
  1481. bgImgelement.style.backgroundImage = "";
  1482. chartWhitelement.style.background = 'rgba(255, 255, 255, 0) !important;'
  1483. } else {
  1484. bgImgelement.style.backgroundImage = chart_bgimg_url;
  1485. chartWhitelement.style.background = "rgba(255, 255, 255, 0.5)"
  1486. }
  1487. // bgImgelement.style.backgroundSize = 'cover'; // 調整背景圖片大小
  1488. // bgImgelement.style.backgroundPosition = 'center'; // 調整背景圖片位置
  1489. // bgImgelement.style.backgroundRepeat = 'no-repeat'; // 調整背景圖片重複
  1490. });
  1491. });
  1492. });
  1493. // =========================
  1494. const radioButtons = document.querySelectorAll('input[name="inlineRadioOptions"]');
  1495. var chartTitle = document.querySelector('.chart_title');
  1496. radioButtons.forEach(button => {
  1497. button.addEventListener('click', function () {
  1498. if (this.value === '0') {
  1499. chartTitle.style.textAlign = 'left';
  1500. } else if (this.value === '1') {
  1501. chartTitle.style.textAlign = 'center';
  1502. } else if (this.value === '2') {
  1503. chartTitle.style.textAlign = 'right';
  1504. }
  1505. });
  1506. });
  1507. // 圖表標題
  1508. var keywordInput = document.getElementById('title_data');
  1509. // 初始时将输入框的值设置为标题的文本内容
  1510. keywordInput.value = chartTitle.textContent;
  1511. // 监听输入框的输入事件,实现内容同步更新
  1512. keywordInput.addEventListener('input', function () {
  1513. chartTitle.textContent = this.value;
  1514. });
  1515. // 標題字體大小
  1516. // ==========================
  1517. var TitlefontSize = document.getElementById('TitlefontSize');
  1518. TitlefontSize.addEventListener('input', function () {
  1519. var TitlefontSizeValue = TitlefontSize.value;
  1520. chartTitle.style.fontSize = TitlefontSizeValue + 'px';;
  1521. });
  1522. // 標題字體顏色
  1523. var stocknumber;
  1524. // ==========================
  1525. let labelDependent = ""; // 圖表左側 Label
  1526. function get_data(input_text_value) {
  1527. clickExample.style.display = 'none'
  1528. // 重新创建取消令牌和源
  1529. cancelToken = axios.CancelToken;
  1530. source = cancelToken.source();
  1531. // inputField.value = "";
  1532. apiLoading();
  1533. var existingTable = chartJsBox.querySelector('table');
  1534. downloadButton.style.display = "none";
  1535. if (chartDiv.childElementCount != 0) {
  1536. data = [];
  1537. labels = [];
  1538. console.log('已存在')
  1539. chartTitle.textContent = "";
  1540. unit_value.textContent = "";
  1541. bgImgelement.style.backgroundImage = "";
  1542. chartWhitelement.style.background = "";
  1543. document.getElementById('img_box_url').src = "";
  1544. }
  1545. if (existingTable) {
  1546. unit_value.textContent = "";
  1547. chartJsBox.removeChild(existingTable);
  1548. chartTitle.textContent = "";
  1549. bgImgelement.style.backgroundImage = "";
  1550. chartWhitelement.style.background = "";
  1551. document.getElementById('img_box_url').src = "";
  1552. }
  1553. if (myChart) {
  1554. unit_value_box.style.display = 'none';
  1555. unit_value.textContent = "";
  1556. data = [];
  1557. labels = [];
  1558. chartTitle.textContent = "";
  1559. bgImgelement.style.backgroundImage = "";
  1560. chartWhitelement.style.background = "";
  1561. myChart.destroy();
  1562. document.getElementById('img_box_url').src = "";
  1563. }
  1564. const apiUrl = `https://cmm.ai:8080/answer_with_token?token=${accessToken}&question=${input_text_value}`;
  1565. axios.get(apiUrl, {
  1566. cancelToken: source.token // 将取消令牌传递给 Axios 请求
  1567. })
  1568. .then(response => {
  1569. apiHideLoading();
  1570. console.log(response);
  1571. if (response.data.data == "未達相似度搜尋標準") {
  1572. alert('無法生成圖表')
  1573. return
  1574. } else {
  1575. unit_value_box.style.display = "block";
  1576. var chart_info = response.data.chart_info.Title;
  1577. chartType = response.data.chart_info.Chart_type;
  1578. var finance_img_url = response.data.imageUrl_info.imageUrl;
  1579. if (response.data.imageUrl_info != null) {
  1580. console.log(finance_img_url);
  1581. document.getElementById('img_box_url').src = finance_img_url;
  1582. }
  1583. keywordInput.value = chart_info;
  1584. chartTitle.textContent = chart_info;
  1585. unit = response.data.chart_info.Unit_of_dependent_variable;
  1586. var Label_dependent_variable = response.data.chart_info.Label_dependent_variable;
  1587. var Label_independent_variable = response.data.chart_info.Label_independent_variable;
  1588. stocknumber = response.data.chart_info.args;
  1589. chartTypeBtn.value = chartType;
  1590. unit_data.value = response.data.chart_info.Unit_of_dependent_variable;
  1591. if (response.data.chart_info.Unit_of_dependent_variable == "") {
  1592. unit_value.textContent = ""
  1593. } else {
  1594. unit_value.textContent = response.data.chart_info.Unit_of_dependent_variable;
  1595. }
  1596. obj.text = chartType;
  1597. dataArray = response.data.data;
  1598. if (chartType === "table") {
  1599. unit_value.textContent = "";
  1600. unit_value_box.style.display = "none"
  1601. createTable(dataArray, unit, Label_dependent_variable, Label_independent_variable);
  1602. // document.getElementById('img_box_url').src = "";
  1603. } else if (chartType === "bar") {
  1604. unit_value_box.style.display = 'block';
  1605. console.log('圖表類型bar')
  1606. $('#textToChart').hide();
  1607. extractAndGenerateChart(dataArray);
  1608. } else if (chartType === "doughnut") {
  1609. unit_value.textContent = "";
  1610. $('#textToChart').hide();
  1611. generatePieChart(dataArray, pieFontSize, PieMaxWidth)
  1612. } else {
  1613. unit_value_box.style.display = "block"
  1614. $('#chartdiv').hide();
  1615. generateChart(dataArray);
  1616. }
  1617. // }
  1618. // 在這裡處理成功獲取 JSON 的情況
  1619. // console.log(response);
  1620. // switch (chartType) {
  1621. // case "line":
  1622. // document.getElementById("inlineRadio4").checked = true; // 折线图
  1623. // break;
  1624. // case "bar":
  1625. // document.getElementById("inlineRadio5").checked = true; // 柱状图
  1626. // labelDependent = response.data.chart_info.Label_dependent_variable;
  1627. // break;
  1628. // case "pie":
  1629. // document.getElementById("inlineRadio6").checked = true; // 圆饼图
  1630. // break;
  1631. // case "table":
  1632. // document.getElementById("inlineRadio7").checked = true; // 表格
  1633. // break;
  1634. // default:
  1635. // // 默认情况
  1636. }
  1637. // generateChart(dataArray);
  1638. })
  1639. .catch(error => {
  1640. // 错误处理逻辑,包括取消请求的情况
  1641. apiHideLoading();
  1642. if (axios.isCancel(error)) {
  1643. console.log('请求被取消:', error.message);
  1644. } else {
  1645. console.error('发生错误:', error);
  1646. }
  1647. });
  1648. }
  1649. // 取消请求的函数
  1650. function cancelRequest() {
  1651. if (source) {
  1652. source.cancel('取消了 API 请求');
  1653. apiHideLoading(); // 隐藏加载状态
  1654. }
  1655. }
  1656. document.getElementById('cancel_send_data').addEventListener('click', cancelRequest);
  1657. $(document).ready(function () {
  1658. $('#dataTable').DataTable();
  1659. });
  1660. function resizeChart() {
  1661. console.log(myChart)
  1662. if (myChart != null) {
  1663. myChart.resize();
  1664. }
  1665. }
  1666. document.addEventListener('visibilitychange', function () {
  1667. console.log('Visibility changed:', document.visibilityState);
  1668. if (document.visibilityState === 'visible') {
  1669. resizeChart();
  1670. }
  1671. });
  1672. // 监听窗口resize事件
  1673. window.addEventListener('resize', resizeChart);
  1674. var setting_button = document.getElementById('chart_seeting_button');
  1675. var closeButton = document.getElementById('style_setting_close');
  1676. var closeButton_table = document.getElementById('style_setting_close_table');
  1677. // var borderSetting = document.getElementById('borderSetting');
  1678. var lineChartSetting = document.getElementById('lineChartSetting');
  1679. var barChartSetting = document.getElementById('barChartSetting');
  1680. var table_setting = document.getElementById('table_setting');
  1681. var chart_category = document.getElementById('chart_category');
  1682. var pie_setting = document.getElementById('pie_setting');
  1683. // 获取 chartDiv 元素
  1684. setting_button.addEventListener('click', () => {
  1685. console.log('setting_button')
  1686. if (myChart === null && !chartJsBox.querySelector('table') && chartDiv.childElementCount === 0) {
  1687. alert('請先生成圖表');
  1688. return
  1689. }
  1690. if (chartType === 'table') {
  1691. console.log('表格');
  1692. sidebar.classList.toggle('show');
  1693. sidebar.classList.toggle('hidden');
  1694. barChartSetting.style.display = 'none';
  1695. lineChartSetting.style.display = 'none';
  1696. table_setting.style.display = 'block';
  1697. chart_category.style.display = 'none';
  1698. pie_setting.style.display = 'none';
  1699. } else if (chartType === "bar") {
  1700. console.log('柱狀圖')
  1701. sidebar.classList.toggle('show');
  1702. sidebar.classList.toggle('hidden');
  1703. barChartSetting.style.display = 'block';
  1704. lineChartSetting.style.display = 'none';
  1705. table_setting.style.display = 'none';
  1706. pie_setting.style.display = 'none';
  1707. } else if (chartType === "doughnut") {
  1708. console.log('快捷功能圓餅圖')
  1709. sidebar.classList.toggle('show');
  1710. sidebar.classList.toggle('hidden');
  1711. barChartSetting.style.display = 'none';
  1712. lineChartSetting.style.display = 'none';
  1713. pie_setting.style.display = 'block';
  1714. table_setting.style.display = 'none';
  1715. chart_category.style.display = 'none';
  1716. document.getElementById('unit_input').style.display = 'none';
  1717. } else {
  1718. sidebar.classList.toggle('show');
  1719. sidebar.classList.toggle('hidden');
  1720. table_setting.style.display = 'none';
  1721. barChartSetting.style.display = 'none';
  1722. lineChartSetting.style.display = 'block';
  1723. pie_setting.style.display = 'none';
  1724. }
  1725. });
  1726. closeButton.addEventListener('click', () => {
  1727. sidebar.classList.remove('show');
  1728. sidebar.classList.add('hidden');
  1729. });
  1730. const finance_options = [
  1731. { name: "原油油桶", imageUrl: "./img/finance/01.webp" },
  1732. { name: "黃金", imageUrl: "./img/finance/02.webp" },
  1733. { name: "房屋", imageUrl: "./img/finance/03.webp" },
  1734. { name: "車", imageUrl: "./img/finance/04.webp" },
  1735. { name: "股市", imageUrl: "./img/finance/05.webp" },
  1736. { name: "晶圓", imageUrl: "./img/finance/06.webp" },
  1737. { name: "TSMC 台積電", imageUrl: "./img/finance/07.webp" },
  1738. { name: "台幣", imageUrl: "./img/finance/08.webp" },
  1739. { name: "美金", imageUrl: "./img/finance/09.webp" },
  1740. { name: "購物車", imageUrl: "./img/finance/10.webp" },
  1741. { name: "美中貿易", imageUrl: "./img/finance/11.webp" },
  1742. { name: "世界地圖", imageUrl: "./img/finance/12.webp" },
  1743. { name: "小麥", imageUrl: "./img/finance/13.webp" },
  1744. { name: "玉米", imageUrl: "./img/finance/14.webp" },
  1745. { name: "智慧產線", imageUrl: "./img/finance/15.webp" }
  1746. ];
  1747. const selectOptions = document.getElementById('finance_options');
  1748. const selectOptions_table = document.getElementById('finance_options_table');
  1749. finance_options.forEach((option, index) => {
  1750. const optionElement = document.createElement('option');
  1751. optionElement.value = index + 1; // 值从 1 开始
  1752. optionElement.textContent = option.name;
  1753. optionElement.setAttribute('data-image-url', option.imageUrl); // 存储图片 URL
  1754. selectOptions.appendChild(optionElement);
  1755. });
  1756. selectOptions.addEventListener('change', function () {
  1757. const selectedOption = selectOptions.options[selectOptions.selectedIndex];
  1758. const imageUrl = selectedOption.getAttribute('data-image-url');
  1759. document.getElementById('img_box_url').src = imageUrl;
  1760. });
  1761. // finance_options.forEach((option, index) => {
  1762. // const optionElement = document.createElement('option');
  1763. // optionElement.value = index + 1; // 值从 1 开始
  1764. // optionElement.textContent = option.name;
  1765. // optionElement.setAttribute('data-image-url', option.imageUrl); // 存储图片 URL
  1766. // selectOptions_table.appendChild(optionElement);
  1767. // });
  1768. // selectOptions_table.addEventListener('change', function () {
  1769. // const selectedOption = selectOptions_table.options[selectOptions_table.selectedIndex];
  1770. // const imageUrl = selectedOption.getAttribute('data-image-url');
  1771. // document.getElementById('img_box_url').src = imageUrl;
  1772. // });
  1773. document.querySelectorAll('input[name="ChartOptions"]').forEach(function (radio) {
  1774. radio.addEventListener('change', function () {
  1775. // 檢查哪個單選按鈕被選中
  1776. if (this.value === 'line') {
  1777. // 如果選擇了折線圖,執行相應的操作
  1778. console.log("選擇了折線圖");
  1779. // 在這裡執行顯示折線圖的相關代碼
  1780. } else if (this.value === 'doughnut') {
  1781. // 如果選擇了柱狀圖,執行相應的操作
  1782. console.log("選擇了柱狀圖");
  1783. // 在這裡執行顯示柱狀圖的相關代碼
  1784. }
  1785. });
  1786. });
  1787. function tokencheck() {
  1788. if (!localStorage.getItem("access_token")) {
  1789. window.location.href = "./login.html";
  1790. } else {
  1791. console.log("存在");
  1792. }
  1793. }
  1794. tokencheck();
  1795. function logout() {
  1796. localStorage.removeItem("access_token");
  1797. }
  1798. $(document).on("click", "#logout", function (event) {
  1799. alert("登出成功");
  1800. logout();
  1801. location.reload();
  1802. });
  1803. // speech_to_text
  1804. var closeRecord = document.getElementById('recording_block_close');
  1805. var recording_block = document.getElementById('recording_block');
  1806. var PageLM = "2024-05-07 18:33";
  1807. var recording_button = document.getElementById('recording_button');
  1808. var stopButton = document.getElementById('stop');
  1809. var audio = document.getElementById('audio');
  1810. var recorder, audioBlob;
  1811. closeRecord.addEventListener('click', () => {
  1812. recPause();
  1813. console.log('record-click')
  1814. recording_block.classList.remove('show');
  1815. recording_block.classList.add('hidden');
  1816. });
  1817. function reclog(s, color) {
  1818. var now = new Date();
  1819. var t = ("0" + now.getHours()).substr(-2)
  1820. + ":" + ("0" + now.getMinutes()).substr(-2)
  1821. + ":" + ("0" + now.getSeconds()).substr(-2);
  1822. var div = document.createElement("div");
  1823. var elem = document.querySelector(".reclog");
  1824. elem.insertBefore(div, elem.firstChild);
  1825. div.innerHTML = '<div style="color:' + (!color ? "" : color == 1 ? "red" : color == 2 ? "#0b1" : color) + '">[' + t + ']' + s + '</div>';
  1826. };
  1827. window.onerror = function (message, url, lineNo, columnNo, error) {
  1828. //https://www.cnblogs.com/xianyulaodi/p/6201829.html
  1829. reclog('<span style="color:red">【Uncaught Error】' + message + '<pre>' + "at:" + lineNo + ":" + columnNo + " url:" + url + "\n" + (error && error.stack || Html_$T("kBaF::不能获得错误堆栈")) + '</pre></span>');
  1830. };
  1831. if (!window.Html_$T) {//没有提供本页面用的国际化多语言支持时 返回中文文本
  1832. window.Html_$T = function () {
  1833. var a = arguments, txt = a[0].replace(/^.+?::/, ""), n = 0;
  1834. for (var i = 0; i < a.length; i++) { if (typeof a[i] == "number") { n = i; break } }
  1835. txt = txt.replace(/\{(\d+)\}/g, function (v, f) { v = a[+f + n]; return v == null ? "" : v });
  1836. return txt;
  1837. }
  1838. window.Html_xT = function (v) { return v }
  1839. }
  1840. if (window.Recorder) {
  1841. // reclog(Html_$T('BL9u::頁面已準備好,請先點擊打開錄音,然後點擊錄製'), 2);
  1842. alert('錄音已準備好,請先點擊右下角打開錄音,即可開始錄製')
  1843. } else {
  1844. reclog(Html_$T("YzPd::js文件加载失败,请刷新重试!"), "#f00;font-size:50px");
  1845. console.log('js文件加载失敗,请刷新重试!')
  1846. }
  1847. recording_button.addEventListener('click', async () => {
  1848. recording_block.classList.toggle('show');
  1849. recording_block.classList.toggle('hidden');
  1850. recStart();
  1851. });
  1852. function recStart() {//打开了录音后才能进行start、stop调用
  1853. rec.start();
  1854. };
  1855. stopButton.addEventListener('click', async () => {
  1856. // recorder.stop();
  1857. // uploadAudio(audioBlob);
  1858. recStop();
  1859. });
  1860. window.onload = function () {
  1861. recOpen();
  1862. };
  1863. // 錄音程式
  1864. // =======================
  1865. var rec, wave, recBlob;
  1866. /**调用open打开录音请求好录音权限 Call open to open the recording and request the recording permission**/
  1867. var recOpen = function () {//一般在显示出录音按钮或相关的录音界面时进行此方法调用,后面用户点击开始录音时就能畅通无阻了
  1868. rec = null;
  1869. wave = null;
  1870. recBlob = null;
  1871. var newRec = Recorder({
  1872. type: "mp3", sampleRate: 16000, bitRate: 16 //mp3格式,指定采样率hz、比特率kbps,其他参数使用默认配置;注意:是数字的参数必须提供数字,不要用字符串;需要使用的type类型,需提前把格式支持文件加载进来,比如使用wav格式需要提前加载wav.js编码引擎
  1873. , onProcess: function (buffers, powerLevel, bufferDuration, bufferSampleRate, newBufferIdx, asyncEnd) {
  1874. // // //录音实时回调,大约1秒调用12次本回调
  1875. document.querySelector(".recpowerx").style.width = powerLevel + "%";
  1876. document.querySelector(".recpowert").innerText = formatMs(bufferDuration, 1) + " / " + powerLevel;
  1877. // //可视化图形绘制
  1878. // wave.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);
  1879. }
  1880. });
  1881. newRec.open(function () {//打开麦克风授权获得相关资源
  1882. rec = newRec;
  1883. //此处创建这些音频可视化图形绘制浏览器支持妥妥的
  1884. // wave = Recorder.FrequencyHistogramView({ elem: ".recwave" });
  1885. reclog(Html_$T("GVCa::已打開錄音,可以點擊錄制開始錄音"), 2);
  1886. }, function (msg, isUserNotAllow) {//用户拒绝未授权或不支持
  1887. reclog((isUserNotAllow ? "UserNotAllow, " : "") + Html_$T("TOOV::打開錄音失敗:") + msg, 1);
  1888. });
  1889. };
  1890. /**关闭录音,释放资源 Close recording, release resources**/
  1891. function recClose() {
  1892. if (rec) {
  1893. rec.close();
  1894. reclog(Html_$T("jqOs::已关闭"));
  1895. } else {
  1896. reclog(Html_$T("VOOw::未打开录音"), 1);
  1897. };
  1898. };
  1899. /**开始录音 Start recording**/
  1900. function recStart() {//打开了录音后才能进行start、stop调用
  1901. if (rec && Recorder.IsOpen()) {
  1902. recBlob = null;
  1903. rec.start();
  1904. reclog(Html_$T("CGdy::已經開始錄音...") + " " + rec.set.type + " " + rec.set.sampleRate + " " + rec.set.bitRate + "kbps");
  1905. } else {
  1906. reclog(Html_$T("ajKR::未打开錄音"), 1);
  1907. };
  1908. };
  1909. /**暂停录音 Passing recording**/
  1910. function recPause() {
  1911. if (rec && Recorder.IsOpen()) {
  1912. rec.pause();
  1913. reclog(Html_$T("GvCy::已暂停"));
  1914. } else {
  1915. reclog(Html_$T("gCAR::未打開錄音"), 1);
  1916. };
  1917. };
  1918. /**恢复录音 Resume recording**/
  1919. function recResume() {
  1920. if (rec && Recorder.IsOpen()) {
  1921. rec.resume();
  1922. reclog(Html_$T("5q1K::继续录音中..."));
  1923. } else {
  1924. reclog(Html_$T("Ob6S::未打开录音"), 1);
  1925. };
  1926. };
  1927. /**结束录音,得到音频文件 Stop recording and get audio files**/
  1928. function recStop() {
  1929. if (!(rec && Recorder.IsOpen())) {
  1930. reclog(Html_$T("5JuL::未打開錄音"), 1);
  1931. return;
  1932. };
  1933. rec.stop(function (blob, duration) {
  1934. console.log(blob, (window.URL || webkitURL).createObjectURL(blob), Html_xT(Html_$T("gOix::時長:{1}ms", 0, duration)));
  1935. recBlob = blob;
  1936. var file = new File([blob], "recording.mp3", { type: "audio/mp3" });
  1937. // console.log(file);
  1938. handleAudioToText(file);
  1939. reclog(Html_$T("0LHf::已錄製mp3:{1}ms {2}字節,可以點擊播放、上傳、本地下载了", 0, formatMs(duration), blob.size), 2);
  1940. }, function (msg) {
  1941. reclog(Html_$T("kGZO::錄音失敗:") + msg, 1);
  1942. });
  1943. };
  1944. /**播放 Play**/
  1945. function recPlay() {
  1946. if (!recBlob) {
  1947. reclog(Html_$T("tIke::请先錄音,然后停止後再播放"), 1);
  1948. return;
  1949. };
  1950. var cls = ("a" + Math.random()).replace(".", "");
  1951. reclog(Html_$T('GlWb::播放中: ') + '<span class="' + cls + '"></span>');
  1952. var audio = document.createElement("audio");
  1953. audio.controls = true;
  1954. document.querySelector("." + cls).appendChild(audio);
  1955. //简单利用URL生成播放地址,注意不用了时需要revokeObjectURL,否则霸占内存
  1956. audio.src = (window.URL || webkitURL).createObjectURL(recBlob);
  1957. audio.play();
  1958. setTimeout(function () {
  1959. (window.URL || webkitURL).revokeObjectURL(audio.src);
  1960. }, 5000);
  1961. };
  1962. /**上传 Upload**/
  1963. function recUpload() {
  1964. var blob = recBlob;
  1965. if (!blob) {
  1966. reclog(Html_$T("DUTn::请先录音,然后停止后再上传"), 1);
  1967. return;
  1968. };
  1969. //本例子假设使用原始XMLHttpRequest请求方式,实际使用中自行调整为自己的请求方式
  1970. //录音结束时拿到了blob文件对象,可以用FileReader读取出内容,或者用FormData上传
  1971. var api = "http://127.0.0.1:9528";
  1972. var onreadystatechange = function (xhr, title) {
  1973. return function () {
  1974. if (xhr.readyState == 4) {
  1975. if (xhr.status == 200) {
  1976. reclog(title + Html_$T("G2MU::上传成功") + ' <span style="color:#999">response: ' + xhr.responseText + '</span>', 2);
  1977. } else {
  1978. reclog(title + Html_$T("TUdi::没有完成上传,演示上传地址无需关注上传结果,只要浏览器控制台内Network面板内看到的请求数据结构是预期的就ok了。"), "#d8c1a0");
  1979. console.error(Html_xT(title + Html_$T("HjDi::上传失败")), xhr.status, xhr.responseText);
  1980. };
  1981. };
  1982. };
  1983. };
  1984. reclog(Html_$T("QnSI::开始上传到{1},请稍候... (你可以先到源码 /assets/node-localServer 目录内执行 npm run start 来运行本地测试服务器)", 0, api));
  1985. /***方式一:将blob文件转成base64纯文本编码,使用普通application/x-www-form-urlencoded表单上传***/
  1986. var reader = new FileReader();
  1987. reader.onloadend = function () {
  1988. var postData = "";
  1989. postData += "mime=" + encodeURIComponent(blob.type);//告诉后端,这个录音是什么格式的,可能前后端都固定的mp3可以不用写
  1990. postData += "&upfile_b64=" + encodeURIComponent((/.+;\s*base64\s*,\s*(.+)$/i.exec(reader.result) || [])[1]) //录音文件内容,后端进行base64解码成二进制
  1991. //...其他表单参数
  1992. var xhr = new XMLHttpRequest();
  1993. xhr.open("POST", api + "/uploadBase64");
  1994. xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
  1995. xhr.onreadystatechange = onreadystatechange(xhr, Html_$T("gG1f::上传方式一【Base64】"));
  1996. xhr.send(postData);
  1997. };
  1998. reader.readAsDataURL(blob);
  1999. /***方式二:使用FormData用multipart/form-data表单上传文件***/
  2000. var form = new FormData();
  2001. form.append("upfile", blob, "recorder.mp3"); //和普通form表单并无二致,后端接收到upfile参数的文件,文件名为recorder.mp3
  2002. //...其他表单参数
  2003. var xhr = new XMLHttpRequest();
  2004. xhr.open("POST", api + "/upload");
  2005. xhr.onreadystatechange = onreadystatechange(xhr, Html_$T("vDzB::上传方式二【FormData】"));
  2006. xhr.send(form);
  2007. };
  2008. /**本地下载 Local download**/
  2009. function recLocalDown() {
  2010. if (!recBlob) {
  2011. reclog(Html_$T("M86h::请先录音,然后停止后再下载"), 1);
  2012. return;
  2013. };
  2014. var cls = ("a" + Math.random()).replace(".", "");
  2015. recdown64.lastCls = cls;
  2016. reclog(Html_$T('vJPl::点击 ') + '<span class="' + cls + '"></span>' + Html_$T('Whtc:: 下载,或复制文本')
  2017. + '<button onclick="recdown64(\'' + cls + '\')">' + Html_$T('XK4l::生成Base64文本') + '</button><span class="' + cls + '_b64"></span>');
  2018. var fileName = "recorder-" + Date.now() + ".mp3";
  2019. var downA = document.createElement("A");
  2020. downA.innerHTML = Html_$T("g8Fy::下载 ") + fileName;
  2021. downA.href = (window.URL || webkitURL).createObjectURL(recBlob);
  2022. downA.download = fileName;
  2023. document.querySelector("." + cls).appendChild(downA);
  2024. if (/mobile/i.test(navigator.userAgent)) {
  2025. alert(Html_xT(Html_$T("DIEK::因移动端绝大部分国产浏览器未适配Blob Url的下载,所以本demo代码在移动端未调用downA.click()。请尝试点击日志中显示的下载链接下载")));
  2026. } else {
  2027. downA.click();
  2028. }
  2029. //不用了时需要revokeObjectURL,否则霸占内存
  2030. //(window.URL||webkitURL).revokeObjectURL(downA.href);
  2031. };
  2032. function recdown64(cls) {
  2033. var el = document.querySelector("." + cls + "_b64");
  2034. if (recdown64.lastCls != cls) {
  2035. el.innerHTML = '<span style="color:red">' + Html_$T("eKKx::老的数据没有保存,只支持最新的一条") + '</span>';
  2036. return;
  2037. }
  2038. var reader = new FileReader();
  2039. reader.onloadend = function () {
  2040. el.innerHTML = '<textarea></textarea>';
  2041. el.querySelector("textarea").value = reader.result;
  2042. };
  2043. reader.readAsDataURL(recBlob);
  2044. };
  2045. var formatMs = function (ms, all) {
  2046. var ss = ms % 1000; ms = (ms - ss) / 1000;
  2047. var s = ms % 60; ms = (ms - s) / 60;
  2048. var m = ms % 60; ms = (ms - m) / 60;
  2049. var h = ms;
  2050. var t = (h ? h + ":" : "")
  2051. + (all || h + m ? ("0" + m).substr(-2) + ":" : "")
  2052. + (all || h + m + s ? ("0" + s).substr(-2) + "″" : "")
  2053. + ("00" + ss).substr(-3);
  2054. return t;
  2055. };
  2056. // =======================
  2057. let mediaRecorder;
  2058. let audioChunks = [];
  2059. var recordText = document.getElementById('record_text');
  2060. async function handleAudioToText(file) {
  2061. console.log('tts')
  2062. let audioLang = "cmn-Hant-TW"; // 音訊語言
  2063. // let lang = localStorage.getItem("lang");
  2064. let lang = "zh-tw";
  2065. console.log("lang", lang);
  2066. switch (lang) {
  2067. case "zh-tw":
  2068. audioLang = "cmn-Hant-TW";
  2069. break;
  2070. case "en-us":
  2071. audioLang = "en-US";
  2072. break;
  2073. case "ja-jp":
  2074. audioLang = "ja-JP";
  2075. break;
  2076. case "ko-kr":
  2077. audioLang = "ko-KR";
  2078. break;
  2079. default:
  2080. break;
  2081. }
  2082. let url = `https://cmm.ai:9001/gcp/speech-to-text?language_code=${audioLang}`;
  2083. const formData = new FormData();
  2084. formData.append("file", file);
  2085. try {
  2086. const response = await axios.post(url, formData);
  2087. console.log("response", response);
  2088. input_text_value = response.data;
  2089. recording_block.classList.remove('show');
  2090. recording_block.classList.add('hidden');
  2091. if (input_text_value.length === 0) {
  2092. alert('請再說一次')
  2093. return
  2094. }
  2095. console.log(input_text_value[0]);
  2096. inputField.value = input_text_value[0];
  2097. // Show the result for a short time before executing get_data
  2098. get_data(input_text_value);
  2099. // get_data(input_text_value);
  2100. } catch (error) {
  2101. console.log("error", error);
  2102. }
  2103. }
  2104. Object.defineProperty(obj, 'text', {
  2105. get() {
  2106. return this._text;
  2107. },
  2108. set(newValue) {
  2109. console.log(`字符串从 "${this._text}" 变为 "${newValue}"`);
  2110. this._text = newValue;
  2111. }
  2112. });