text-to-chart.js 53 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643
  1. console.log('text-to-chart');
  2. Chart.register(ChartDataLabels);
  3. var inputField = document.getElementById("keyword_data");
  4. var sendButton = document.getElementById("send_data");
  5. var canvasPng = document.getElementById("textToChart");
  6. var img_box = document.getElementById("img_box");
  7. var sidebar = document.getElementById('style_setting');
  8. var input_text_value;
  9. var chartType = 'line';
  10. var unit = '';
  11. var myChart = null;
  12. var chartColor = '#912B2B';
  13. var displayXaxes = true;
  14. var ctx;
  15. var chartlabels;
  16. var chartdata;
  17. var XfontSizeValue = 18;
  18. var YfontSizeValue = 18;
  19. var TitlefontSizeValue = 32;
  20. var chart_bgimg_url = "url(./img/bg.webp)";
  21. var vocab = [];
  22. var chartDiv = document.getElementById('chartdiv');
  23. document.addEventListener('DOMContentLoaded', function () {
  24. axios.get("https://cmm.ai:8080/vocab")
  25. .then(response => {
  26. // apiHideLoading();
  27. console.log(response);
  28. vocab = response.data;
  29. })
  30. .catch(error => {
  31. console.error('發生錯誤:', error);
  32. });
  33. });
  34. inputField.addEventListener('click', function () {
  35. // 確保輸入框處於焦點狀態
  36. inputField.focus();
  37. // 在控制台中顯示焦點狀態
  38. console.log('Input field focused');
  39. });
  40. document.addEventListener('DOMContentLoaded', function () {
  41. // const input = document.getElementById('stockInput');
  42. const suggestions = document.getElementById('suggestions');
  43. inputField.addEventListener('input', function () {
  44. const query = this.value.trim();
  45. suggestions.innerHTML = ''; // 清空之前的建議
  46. console.log('query', query);
  47. if (query) {
  48. // const filteredVocab = vocab.filter(item => item.toLowerCase().includes(query));
  49. const queryTerms = query.split(/\s+/); // 分割輸入的內容
  50. const lastQueryTerm = queryTerms[queryTerms.length - 1];
  51. const filteredVocab = vocab.filter(item => {
  52. // 對每個分割後的單詞進行匹配,只要其中有一個匹配成功就返回 true
  53. return item.toLowerCase().includes(lastQueryTerm.toLowerCase());
  54. });
  55. if (filteredVocab.length > 0) {
  56. suggestions.style.border = '1px solid #ccc';
  57. suggestions.style.display = 'block'; // 顯示建議容器
  58. } else {
  59. suggestions.style.border = 'none';
  60. suggestions.style.display = 'none'; // 隱藏建議容器
  61. }
  62. console.log(queryTerms[queryTerms.length - 1])
  63. filteredVocab.forEach(item => {
  64. const div = document.createElement('div');
  65. div.classList.add('autocomplete-suggestion');
  66. div.textContent = item;
  67. // div.addEventListener('click', function () {
  68. // inputField.value = item;
  69. // suggestions.innerHTML = ''; // 清空建議
  70. // suggestions.style.border = 'none';
  71. // suggestions.style.display = 'none'; // 隱藏建議容器
  72. // });
  73. console.log('queryTerms', queryTerms);
  74. // console.log('filteredVocab', filteredVocab);
  75. div.addEventListener('click', function () {
  76. queryTerms.forEach((term, index) => {
  77. const regex = new RegExp(term, 'gi');
  78. const match = regex.exec(inputField.value);
  79. // console.log(queryTerms[queryTerms.length - 1]);
  80. // console.log(filteredVocab, queryTerms[queryTerms.length - 1])
  81. if (index === queryTerms.length - 1) {
  82. if (match) {
  83. var start = match.index;
  84. var end = start + match[0].length;
  85. console.log(start, end)
  86. const textBefore = inputField.value.substring(0, start);
  87. const textAfter = inputField.value.substring(end, inputField.value.length);
  88. console.log('start', start, 'end', end, 'textBefore', textBefore, 'textAfter', textAfter);
  89. // 捕獲被選中的部分並替換它
  90. const selectedText = inputField.value.substring(start, end);
  91. console.log('selectedText', selectedText);
  92. const newText = textBefore + item + textAfter.replace(selectedText, '');
  93. console.log('newText', newText);
  94. inputField.value = newText;
  95. // inputField.value = textBefore + item.replace(selectedText, '') + textAfter;
  96. suggestions.innerHTML = ''; // 清空建議
  97. suggestions.style.border = 'none';
  98. }
  99. }
  100. });
  101. // const start = inputField.selectionStart;
  102. // const end = inputField.selectionEnd;
  103. });
  104. suggestions.appendChild(div);
  105. });
  106. } else {
  107. suggestions.style.border = 'none';
  108. suggestions.style.display = 'none'; // 隱藏建議容器
  109. }
  110. });
  111. });
  112. // 監聽輸入框的鍵盤事件
  113. // document.getElementById("keyword_data").addEventListener("keyup", function (event) {
  114. // // 判斷是否按下 Enter 鍵 (key code: 13)
  115. // if (event.keyCode === 13) {
  116. // // 觸發送出按鈕的點擊事件
  117. // document.getElementById("send_data").click();
  118. // }
  119. // });
  120. // 定義按鈕點擊事件處理函數
  121. function sendButtonClickHandler() {
  122. console.log(inputField.value);
  123. var input_text_value = inputField.value;
  124. $('#chartdiv').hide();
  125. sidebar.classList.remove('show');
  126. sidebar.classList.add('hidden');
  127. // 在這裡添加你希望在按下按鈕時執行的其他代碼
  128. get_data(input_text_value);
  129. }
  130. sendButton.addEventListener("click", sendButtonClickHandler);
  131. // sendButton.addEventListener("click", function () {
  132. // console.log(inputField.value);
  133. // input_text_value = inputField.value;
  134. // // if (myChart) {
  135. // // console.log('已存在')
  136. // // data = [];
  137. // // labels = [];
  138. // // myChart.removePlugin(Chart.pluginService.getPlugin('afterDraw'));
  139. // // myChart.destroy();
  140. // // }
  141. // get_data(input_text_value);
  142. // });
  143. var lastKeyPressTime = 0;
  144. var enterCount = 0;
  145. var ENTER_THRESHOLD = 500; // 設定連續按下 Enter 的時間閾值(毫秒)
  146. inputField.addEventListener("keyup", function (event) {
  147. // 判斷是否按下 Enter 鍵 (key code: 13)
  148. if (event.key === "Enter") {
  149. var currentTime = new Date().getTime();
  150. // 計算與上一次按鍵按下時間的差值
  151. var timeDiff = currentTime - lastKeyPressTime;
  152. lastKeyPressTime = currentTime;
  153. // 如果兩次 Enter 鍵按下的時間差小於閾值,則增加計數器
  154. if (timeDiff <= ENTER_THRESHOLD) {
  155. enterCount++;
  156. // 如果計數器為2,則觸發 API 請求並重置計數器
  157. if (enterCount === 2) {
  158. sendButtonClickHandler();
  159. enterCount = 0;
  160. }
  161. } else {
  162. // 如果時間差大於閾值,則重置計數器
  163. enterCount = 0;
  164. }
  165. } else {
  166. // 如果按下的不是 Enter 鍵,則重置計數器
  167. enterCount = 0;
  168. }
  169. });
  170. function number_format(number, decimals, dec_point, thousands_sep) {
  171. // * example: number_format(1234.56, 2, ',', ' ');
  172. // * return: '1 234,56'
  173. number = (number + '').replace(',', '').replace(' ', '');
  174. var n = !isFinite(+number) ? 0 : +number,
  175. prec = !isFinite(+decimals) ? 0 : Math.abs(decimals),
  176. sep = (typeof thousands_sep === 'undefined') ? ',' : thousands_sep,
  177. dec = (typeof dec_point === 'undefined') ? '.' : dec_point,
  178. s = '',
  179. toFixedFix = function (n, prec) {
  180. var k = Math.pow(10, prec);
  181. return '' + Math.round(n * k) / k;
  182. };
  183. // Fix for IE parseFloat(0.55).toFixed(0) = 0;
  184. s = (prec ? toFixedFix(n, prec) : '' + Math.round(n)).split('.');
  185. if (s[0].length > 3) {
  186. s[0] = s[0].replace(/\B(?=(?:\d{3})+(?!\d))/g, sep);
  187. }
  188. if ((s[1] || '').length < prec) {
  189. s[1] = s[1] || '';
  190. s[1] += new Array(prec - s[1].length + 1).join('0');
  191. }
  192. return s.join(dec);
  193. }
  194. var labels = [];
  195. var data = [];
  196. function apiLoading() {
  197. document.getElementById('chartDataLoading').style.display = 'block';
  198. }
  199. function apiHideLoading() {
  200. document.getElementById('chartDataLoading').style.display = 'none';
  201. }
  202. const DISPLAY = true;
  203. const BORDER = true;
  204. const CHART_AREA = true;
  205. const TICKS = true;
  206. var labels = [];
  207. var data = [];
  208. var dataArray;
  209. function generateChart(dataArray) {
  210. // 创建一个空数组来存储 labels 和 data
  211. console.log(dataArray)
  212. // 遍历数据数组
  213. for (var i = 0; i < dataArray.length; i++) {
  214. var item = dataArray[i];
  215. // 遍历当前项的属性
  216. for (var key in item) {
  217. // 如果属性名不在 labels 数组中,并且属性值不是对象,则将属性名添加到 labels 数组中
  218. if (typeof item[key] === 'string') {
  219. labels.push(item[key]);
  220. }
  221. // 如果属性值是数值类型,则将其添加到 data 数组中
  222. if (typeof item[key] === 'number') {
  223. data.push(item[key]);
  224. }
  225. }
  226. }
  227. console.log(data);
  228. if (myChart) {
  229. console.log('已存在')
  230. myChart.destroy();
  231. }
  232. if (chartType === "doughnut") {
  233. // createChartDoughnut(chartType, data, labels);
  234. // createChartDoughnut2(data, labels)
  235. } else {
  236. createChart(chartType, data, labels);
  237. }
  238. // console.log(data);
  239. console.log(labels);
  240. }
  241. var amChartInstance;
  242. function extractAndGenerateChart(dataArray) {
  243. for (var i = 0; i < dataArray.length; i++) {
  244. var item = dataArray[i];
  245. // 遍历当前项的属性
  246. for (var key in item) {
  247. // 如果属性名不在 labels 数组中,并且属性值不是对象,则将属性名添加到 labels 数组中
  248. if (typeof item[key] === 'string') {
  249. labels.push(item[key]);
  250. }
  251. // 如果属性值是数值类型,则将其添加到 data 数组中
  252. if (typeof item[key] === 'number') {
  253. data.push(item[key]);
  254. }
  255. }
  256. }
  257. // console.log('key', key);
  258. generateBarChart(key, dataArray);
  259. }
  260. // 圓餅圖
  261. function generatePieChart(dataArray) {
  262. $('#chartdiv').show();
  263. bgImgelement.style.backgroundImage = chart_bgimg_url;
  264. // chartDiv.style.width = "100%"
  265. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  266. // Themes begin
  267. am4core.useTheme(am4themes_animated);
  268. // Themes end
  269. // Create chart instance
  270. var chart = am4core.create("chartdiv", am4charts.PieChart);
  271. chart.data = dataArray;
  272. chart.innerRadius = am4core.percent(50);
  273. // Add and configure Series
  274. var pieSeries = chart.series.push(new am4charts.PieSeries());
  275. pieSeries.dataFields.value = "y";
  276. pieSeries.dataFields.category = "x";
  277. pieSeries.slices.template.stroke = am4core.color("#fff");
  278. pieSeries.slices.template.strokeWidth = 2;
  279. pieSeries.slices.template.strokeOpacity = 1;
  280. // Configure labels
  281. pieSeries.labels.template.wrap = true;
  282. pieSeries.labels.template.maxWidth = 100;
  283. pieSeries.labels.template.truncate = false;
  284. pieSeries.labels.template.fontSize = 14;
  285. // Configure label text to wrap and show percentages
  286. pieSeries.labels.template.adapter.add("textOutput", function (text, target) {
  287. if (target.dataItem && target.dataItem.values.value.percent) {
  288. return "[font-size: 14px]" + target.dataItem.category + ": " + target.dataItem.values.value.percent.toFixed(1) + "%";
  289. }
  290. return text;
  291. });
  292. pieSeries.colors.list = [
  293. am4core.color('rgb(171, 51, 49)'),
  294. am4core.color('rgb(34, 83, 149)'),
  295. am4core.color('rgb(79, 148, 65)'),
  296. am4core.color('rgb(217, 195, 105)'),
  297. am4core.color('rgb(142, 124, 180)'),
  298. am4core.color('rgb(211, 183, 144)'),
  299. am4core.color('rgb(83, 84, 84)'),
  300. am4core.color('rgb(229, 147, 152)')
  301. ];
  302. // Ensure labels have background
  303. // pieSeries.labels.template.background.fillOpacity = 1;
  304. console.log(pieSeries.labels.template)
  305. // Set label background color to match corresponding slice color
  306. pieSeries.labels.template.adapter.add("background.fill", function (fill, target) {
  307. return target.dataItem.slice.fill;
  308. });
  309. // This creates initial animation
  310. pieSeries.hiddenState.properties.opacity = 1;
  311. pieSeries.hiddenState.properties.endAngle = -90;
  312. pieSeries.hiddenState.properties.startAngle = -90;
  313. // console.log('調位置')
  314. }
  315. // 柱狀圖
  316. function generateBarChart(key, dataArray) {
  317. $('#chartdiv').show();
  318. // var colorEven = document.getElementById('colorEven').value;
  319. // var colorOdd = document.getElementById('colorOdd').value;
  320. chartDiv.style.fontFamily = "Arial, sans-serif";
  321. chartDiv.style.fontWeight = "900";
  322. bgImgelement.style.backgroundImage = chart_bgimg_url;
  323. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  324. console.log('labelDependent', labelDependent);
  325. // Themes begin
  326. am4core.useTheme(am4themes_animated);
  327. // Themes end
  328. // Create chart instance
  329. var chart = am4core.create("chartdiv", am4charts.XYChart3D);
  330. console.log(dataArray);
  331. chart.data = dataArray;
  332. chart.angle = 30; // 圖表角度
  333. chart.depth = 25; // 圖表深度
  334. let categoryAxis = chart.xAxes.push(new am4charts.CategoryAxis());
  335. categoryAxis.dataFields.category = "x";
  336. categoryAxis.renderer.labels.template.rotation = 0;
  337. categoryAxis.renderer.labels.template.hideOversized = false;
  338. categoryAxis.renderer.minGridDistance = 20; // 間距
  339. // categoryAxis.renderer.labels.template.horizontalCenter = "left";
  340. // categoryAxis.renderer.labels.template.verticalCenter = "middle";
  341. // categoryAxis.tooltip.label.rotation = 90;
  342. // categoryAxis.tooltip.label.horizontalCenter = "right";
  343. // categoryAxis.tooltip.label.verticalCenter = "middle";
  344. // label 換行
  345. let label = categoryAxis.renderer.labels.template;
  346. label.wrap = true;
  347. label.maxWidth = 100;
  348. categoryAxis.renderer.cellEndLocation = 0.5; // 減少值可以增加間距
  349. let valueAxis = chart.yAxes.push(new am4charts.ValueAxis());
  350. valueAxis.title.text = labelDependent; // Y軸
  351. valueAxis.title.fontWeight = "bold";
  352. // Create series
  353. var series = chart.series.push(new am4charts.ColumnSeries3D());
  354. series.dataFields.valueY = 'y';
  355. series.dataFields.categoryX = "x";
  356. series.name = "y";
  357. series.tooltipText = "{categoryX}: [bold]{valueY}[/]";
  358. series.columns.template.fillOpacity = .8;
  359. let columnTemplate = series.columns.template;
  360. columnTemplate.strokeWidth = 2;
  361. columnTemplate.strokeOpacity = 1;
  362. columnTemplate.stroke = am4core.color("#FFFFFF");
  363. // columnTemplate.adapter.add("fill", function (fill, target) {
  364. // return chart.colors.getIndex(target.dataItem.index);
  365. // })
  366. // 設置基數和偶數顏色
  367. columnTemplate.adapter.add("fill", function (fill, target) {
  368. return target.dataItem.index % 2 === 0 ? am4core.color("#288D97") : am4core.color("#427D7E");
  369. });
  370. columnTemplate.adapter.add("stroke", function (stroke, target) {
  371. return chart.colors.getIndex(target.dataItem.index);
  372. })
  373. // // 加上白色線條
  374. // series.columns.template.events.on("validated", function (event) {
  375. // let column = event.target;
  376. // let whiteLine = column.createChild(am4core.Line);
  377. // whiteLine.stroke = am4core.color("#ffffff");
  378. // whiteLine.strokeWidth = 1.3; // 線條寬度
  379. // whiteLine.strokeOpacity = 0.5;// 透明度
  380. // whiteLine.x1 = column.pixelWidth / 1;
  381. // whiteLine.y1 = 0;
  382. // whiteLine.x2 = column.pixelWidth / 1;
  383. // whiteLine.y2 = column.pixelHeight;
  384. // });
  385. chart.cursor = new am4charts.XYCursor();
  386. chart.cursor.lineX.strokeOpacity = 0;
  387. chart.cursor.lineY.strokeOpacity = 0;
  388. categoryAxis.renderer.grid.template.strokeOpacity = 0; // 隱藏 X 軸
  389. // valueAxis.renderer.grid.template.strokeOpacity = 0; // 隱藏 Y 軸
  390. // valueAxis.renderer.line.strokeOpacity = 0;
  391. // LineSeries 底部線條
  392. var lineSeries = chart.series.push(new am4charts.LineSeries());
  393. lineSeries.dataFields.valueY = "bottom";
  394. lineSeries.dataFields.categoryX = "0";
  395. lineSeries.stroke = am4core.color("#FFFFFF")
  396. lineSeries.strokeWidth = 2;
  397. lineSeries.strokeOpacity = 1
  398. var bottomData = [];
  399. for (var i = 0; i < dataArray.length; i++) {
  400. bottomData.push({ "x": dataArray[i].x, "bottom": 0 });
  401. }
  402. lineSeries.data = bottomData;
  403. // 底部線條
  404. // let rectElements = document.querySelectorAll("rect");
  405. // // let widthValue = 0;
  406. // let fifthRectElement = rectElements[3];
  407. // // fifthRectElement.style.position = "relative";
  408. // setTimeout(() => {
  409. // let width = fifthRectElement.getAttribute("width");
  410. // // widthValue = parseFloat(width);
  411. // console.log('rectElements', rectElements);
  412. // console.log('fifthRectElement', fifthRectElement);
  413. // console.log('width', width);
  414. // let line = document.createElement("div");
  415. // line.style.position = "absolute";
  416. // line.style.bottom = "162px";
  417. // line.style.right = "40px";
  418. // line.style.width = `${parseFloat(width)}px`;
  419. // line.style.height = "2px";
  420. // line.style.backgroundColor = "#000000";
  421. // document.getElementById("chartdiv").appendChild(line);
  422. // }, 10)
  423. // end am4core.ready()
  424. }
  425. var downloadButton = document.getElementById("downloadButton");
  426. // 添加點擊事件監聽器
  427. downloadButton.addEventListener('click', function () {
  428. setTimeout(function () {
  429. html2canvas(document.getElementById('data_chart_box')).then(function (canvas) {
  430. // // 獲取圖片資料 URL
  431. // var imageData = canvasPng.toDataURL("image/png");
  432. // // 創建下載連結
  433. // var link = document.createElement('a');
  434. // link.download = 'chart.png'; // 下載的檔案名稱
  435. // link.href = imageData; // 圖片資料 URL
  436. // // 模擬點擊下載連結
  437. // link.click();
  438. // 創建一個&lt;a&gt;元素
  439. const link = document.createElement('a');
  440. // 將 Canvas 轉換為 Data URL
  441. link.href = canvas.toDataURL('image/png');
  442. // 設置下載文件名
  443. link.download = 'chart.png';
  444. // 模擬點擊下載
  445. link.click();
  446. });
  447. }, 1000); // 等待 1 秒钟
  448. });
  449. var pieChartColors = [
  450. 'rgb(171, 51, 49)',
  451. 'rgb(34, 83, 149)',
  452. 'rgb(79, 148, 65)',
  453. 'rgb(217, 195, 105)',
  454. 'rgb(142, 124, 180)',
  455. 'rgb(211, 183, 144)',
  456. 'rgb(83, 84, 84)',
  457. 'rgb(229, 147, 152)'
  458. ];
  459. function createChartDoughnut2(data, labels) {
  460. const chartData = {
  461. type: 'outlabeledPie',
  462. data: {
  463. labels: labels,
  464. datasets: [
  465. {
  466. data: data,
  467. borderWidth: borderWidthValue,
  468. fill: false,
  469. backgroundColor: pieChartColors,
  470. },
  471. ],
  472. },
  473. options: {
  474. plugins: {
  475. legend: false,
  476. outlabels: {
  477. text: '%l %p',
  478. color: 'white',
  479. stretch: 35,
  480. font: {
  481. resizable: true,
  482. family: 'Arial',
  483. size: 48,
  484. weight: 'bold',
  485. style: 'italic',
  486. },
  487. },
  488. },
  489. },
  490. };
  491. const devicePixelRatio = 4;
  492. var chartUrl = `https://quickchart.io/chart?c=${encodeURIComponent(JSON.stringify(chartData))}&devicePixelRatio=${devicePixelRatio}&f=png&fontsize=20`;
  493. // 获取目标 div 元素
  494. const targetDiv = document.getElementById('data_chartJs_box');
  495. // 创建一个 img 元素来显示图表
  496. const chartImg = document.createElement('img');
  497. chartImg.src = chartUrl;
  498. chartImg.classList.add('outlabeled_img');
  499. // 清空目标 div 元素
  500. targetDiv.innerHTML = '';
  501. // 将图表 img 元素插入到目标 div 中
  502. targetDiv.appendChild(chartImg);
  503. console.log(chart_bgimg_url);
  504. bgImgelement.style.backgroundImage = chart_bgimg_url;
  505. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  506. }
  507. // 版型二源餅圖
  508. // // 创建 Chart.js 图表
  509. function createChartDoughnut(chartType, data, labels) {
  510. $('#textToChart').show();
  511. canvasPng.style.maxWidth = '450px';
  512. canvasPng.style.maxHeight = '450px';
  513. var windowWidth = window.innerWidth || document.documentElement.clientWidth;
  514. var canvasWidth = Math.min(500, windowWidth); // 限制最大宽度为 500px
  515. canvasPng.style.width = canvasWidth + 'px'; // 设置 canvas 宽度
  516. canvasPng.style.marginLeft = 'auto'; // 左外边距自动计算
  517. canvasPng.style.marginRight = 'auto'; // 右外边距自动计算
  518. var ctx = document.getElementById("textToChart").getContext('2d');
  519. myChart = new Chart(ctx, {
  520. type: chartType,
  521. data: {
  522. labels: labels,
  523. datasets: [{
  524. strokeColor: "rgba(220,220,220,1)",
  525. StrokeWidth: 5,
  526. data: data,
  527. borderWidth: borderWidthValue,
  528. pointRadius: 0,
  529. pointBorderColor: '#fff',
  530. pointBorderWidth: 3,
  531. pointHoverRadius: 0,
  532. fill: false,
  533. backgroundColor: pieChartColors,
  534. // barPercentage: 0.5,
  535. // categoryPercentage: 0.5
  536. }]
  537. },
  538. options: {
  539. // responsive: true, // 啟用響應式
  540. maintainAspectRatio: true,
  541. responsive: true,
  542. aspectRatio: 1,
  543. plugins: {
  544. legend: {
  545. display: false,
  546. },
  547. datalabels: {
  548. color: '#fff', // 数据标签的颜色
  549. anchor: 'end', // 数据标签的位置
  550. align: 'start', // 数据标签的对齐方式
  551. font: {
  552. size: 14, // 数据标签的字体大小
  553. },
  554. }
  555. },
  556. scales: {
  557. x: {
  558. display: false, // 显示 x 轴
  559. grid: {
  560. display: false,
  561. lineWidth: 5,
  562. color: '#584B3D'
  563. },
  564. ticks: {
  565. textStrokeColor: '#fff',
  566. textStrokeWidth: 5,
  567. font: {
  568. display: false,
  569. family: 'Arial', // 设置字体
  570. size: XfontSizeValue, // 设置字体大小
  571. weight: 'bold', // 设置字体粗细
  572. fontColor: 'rgba(255,255,255,0.8)', // X-axis font color
  573. shadowColor: 'rgba(0, 0, 0, 0.5)', // Shadow color
  574. shadowBlur: 10, // Shadow blur level
  575. shadowOffsetX: 5, // Horizontal shadow offset
  576. shadowOffsetY: 5
  577. },
  578. },
  579. },
  580. y: {
  581. display: false,
  582. beginAtZero: false,
  583. grid: {
  584. display: false,
  585. lineWidth: 5,
  586. color: '#584B3D'
  587. },
  588. ticks: {
  589. textStrokeColor: '#fff',
  590. textStrokeWidth: 5,
  591. font: {
  592. family: 'Arial', // 設置字體
  593. size: XfontSizeValue, // 設置字體大小
  594. weight: 'bold', // 設置字體粗細
  595. color: 'rgba(255,255,255,0.8)', // Y-axis 字體顏色
  596. shadowColor: 'rgba(0, 0, 0, 0.5)', // 陰影顏色
  597. shadowBlur: 10, // 陰影模糊級別
  598. shadowOffsetX: 5, // 水平陰影偏移
  599. shadowOffsetY: 5 // 垂直陰影偏移
  600. },
  601. }
  602. }
  603. },
  604. layout: {
  605. padding: {
  606. left: 50, // 调整图表左边距
  607. right: 50, // 调整图表右边距
  608. top: 10,
  609. bottom: 10,
  610. }
  611. },
  612. },
  613. });
  614. console.log(chart_bgimg_url);
  615. bgImgelement.style.backgroundImage = chart_bgimg_url;
  616. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  617. }
  618. var chartJsBox = document.getElementById('data_chartJs_box');
  619. function createTable(dataArray, unit, Label_dependent_variable, Label_independent_variable,) {
  620. $('#chartdiv').hide();
  621. $('#textToChart').hide();
  622. var table = document.createElement('table');
  623. table.classList.add('table', 'table_template', 'dynamic-table'); // 添加 Bootstrap 的 table 样式,如果你在使用 Bootstrap
  624. // 创建表头
  625. var thead = table.createTHead();
  626. var headerRow = thead.insertRow();
  627. // Object.keys(dataArray[0]).forEach(function (key) {
  628. // var header = headerRow.insertCell();
  629. // header.textContent = key;
  630. // });
  631. var yearHeader = headerRow.insertCell();
  632. yearHeader.textContent = Label_independent_variable;
  633. var avgPriceHeader = headerRow.insertCell();
  634. avgPriceHeader.textContent = Label_dependent_variable;
  635. // 创建表格主体
  636. var tbody = table.createTBody();
  637. dataArray.forEach(function (item) {
  638. console.log(item)
  639. var row = tbody.insertRow();
  640. // var cell1 = row.insertCell();
  641. // cell1.textContent = item.year;
  642. // cell1.classList.add('year-class'); // 为年份单元格添加类
  643. // var cell2 = row.insertCell();
  644. Object.keys(item).forEach(function (key, index) {
  645. var cell = row.insertCell();
  646. if (index === 0) { // If it's the first cell, add 'year-class'
  647. cell.textContent = item[key];
  648. cell.classList.add('year-class');
  649. } else { // Otherwise, add 'average-class'
  650. if (key === 'Average_Close') {
  651. cell.textContent = item[key].toFixed(2) + unit;
  652. } else {
  653. // cell.textContent = item[key];
  654. cell.textContent = item[key].toFixed(2) + unit;
  655. }
  656. cell.classList.add('average-class');
  657. }
  658. });
  659. // console.log(item.avg_price.toFixed(2))
  660. // cell.textContent = item[key].toFixed(2) + unit;
  661. // cell2.classList.add('average-class'); // 为年份单元格添加类
  662. });
  663. // 获取包含表格的 div 元素
  664. bgImgelement.style.backgroundImage = chart_bgimg_url;
  665. img_box.style.bottom = '60px';
  666. img_box.style.right = '5px';
  667. document.getElementById('img_box_url').style.width = "180px";
  668. // chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  669. // 将表格添加到指定的 div 元素中
  670. chartJsBox.appendChild(table);
  671. }
  672. // 版型一折線圖
  673. // // 创建 Chart.js 图表
  674. function createChart(chartType, data, labels) {
  675. $('#textToChart').show();
  676. canvasPng.style.maxWidth = '';
  677. canvasPng.style.maxHeight = '';
  678. ctx = document.getElementById("textToChart");
  679. myChart = new Chart(ctx, {
  680. type: chartType,
  681. data: {
  682. labels: labels,
  683. datasets: [{
  684. strokeColor: "rgba(220,220,220,1)",
  685. StrokeWidth: 5,
  686. data: data,
  687. borderWidth: borderWidthValue,
  688. pointRadius: 0,
  689. pointBorderColor: '#fff',
  690. pointBorderWidth: 3,
  691. pointHoverRadius: 0,
  692. fill: false,
  693. backgroundColor: chartColor,
  694. borderColor: chartColor,
  695. // barPercentage: 0.5,
  696. // categoryPercentage: 0.5
  697. }]
  698. },
  699. options: {
  700. // responsive: true, // 啟用響應式
  701. maintainAspectRatio: false,
  702. plugins: {
  703. chart3d: {
  704. enabled: true, // 啟用 3D 插件
  705. alpha: 25, // 圖表繞 x 軸的旋轉角度
  706. beta: 25, // 圖表繞 y 軸的旋轉角度
  707. depth: 15 // 圖表的深度
  708. },
  709. // datalabels: {
  710. // formatter: function (value, context) {
  711. // // 添加单位
  712. // return value + ' ' + unit;
  713. // },
  714. // textStrokeColor: '#fff',
  715. // textStrokeWidth: 5,
  716. // color: '#584B3D',
  717. // font: {
  718. // size: YfontSizeValue,
  719. // },
  720. // anchor: 'end',
  721. // align: 'end',
  722. // },
  723. datalabels: null,
  724. // datalabels: data.length <= 10 ? {
  725. // formatter: function (value, context) {
  726. // // 添加单位
  727. // return value + ' ' + unit;
  728. // },
  729. // textStrokeColor: '#fff',
  730. // textStrokeWidth: 5,
  731. // color: '#584B3D',
  732. // font: {
  733. // size: YfontSizeValue,
  734. // },
  735. // anchor: 'end',
  736. // align: 'end',
  737. // } : null,
  738. customLabels: null,
  739. legend: {
  740. display: false,
  741. },
  742. // subtitle: {
  743. // display: true,
  744. // align: 'left',
  745. // position: 'top',
  746. // text: 'Custom Chart Subtitle'
  747. // },
  748. // afterDatasetsDraw(chart) {
  749. // var ctx = chart.ctx;
  750. // var xAxis = chart.scales.x;
  751. // var yAxis = chart.scales.y;
  752. // var dataj = myChart.data.datasets[0].data;
  753. // console.log('after', dataj)
  754. // ctx.save();
  755. // ctx.strokeStyle = '#000'; // 虚线颜色
  756. // ctx.lineWidth = 1; // 虚线宽度
  757. // ctx.setLineDash([5, 5]); // 设置虚线样式,[虚线段长度, 间隔长度]
  758. // dataj.forEach(function (value, index) {
  759. // var xPos = xAxis.getPixelForValue(labels[index]); // 获取 x 轴位置
  760. // var yPos = yAxis.getPixelForValue(value); // 获取 y 轴位置
  761. // // console.log(xPos)
  762. // // 绘制虚线
  763. // // ctx.beginPath();
  764. // // ctx.moveTo(xPos, yPos);
  765. // // ctx.lineTo(xPos, xAxis.bottom);
  766. // // ctx.stroke();
  767. // var xLineEnd = Math.min(yPos, yAxis.bottom);
  768. // // console.log('第' + index + '條線', xPos, yPos, 'xLineEnd', xLineEnd, 'yAxis.getPixelForValue(value)', yAxis.getPixelForValue(value));
  769. // // console.log('yAxis.getPixelForValue(value)', yAxis.getPixelForValue(value))
  770. // // 绘制虚线
  771. // ctx.beginPath();
  772. // ctx.moveTo(xPos, yAxis.getPixelForValue(value));
  773. // ctx.lineTo(xPos, xAxis.top);
  774. // ctx.stroke();
  775. // // 添加日期标签
  776. // ctx.fillStyle = '#000'; // 文字颜色
  777. // ctx.textAlign = 'center';
  778. // // ctx.fillText(labels[index], xPos, yAxis.bottom + 20); // 将日期标签绘制在虚线下方
  779. // });
  780. // ctx.restore();
  781. // }
  782. },
  783. scales: {
  784. x: {
  785. display: displayXaxes, // 显示 x 轴
  786. grid: {
  787. display: false,
  788. lineWidth: 5,
  789. color: '#584B3D'
  790. },
  791. ticks: {
  792. textStrokeColor: '#fff',
  793. textStrokeWidth: 5,
  794. font: {
  795. display: false,
  796. family: 'Arial', // 设置字体
  797. size: XfontSizeValue, // 设置字体大小
  798. weight: 'bold', // 设置字体粗细
  799. fontColor: 'rgba(255,255,255,0.8)', // X-axis font color
  800. shadowColor: 'rgba(0, 0, 0, 0.5)', // Shadow color
  801. shadowBlur: 10, // Shadow blur level
  802. shadowOffsetX: 5, // Horizontal shadow offset
  803. shadowOffsetY: 5
  804. },
  805. },
  806. },
  807. y: {
  808. display: true,
  809. beginAtZero: false,
  810. grid: {
  811. grid: {
  812. display: true, // 显示 y 轴网格线
  813. lineWidth: 1, // 网格线宽度
  814. color: '#C0C5CC' // 网格线颜色
  815. },
  816. },
  817. ticks: {
  818. textStrokeColor: '#fff',
  819. textStrokeWidth: 5,
  820. font: {
  821. family: 'Arial', // 設置字體
  822. size: XfontSizeValue, // 設置字體大小
  823. weight: 'bold', // 設置字體粗細
  824. color: 'rgba(255,255,255,0.8)', // Y-axis 字體顏色
  825. shadowColor: 'rgba(0, 0, 0, 0.5)', // 陰影顏色
  826. shadowBlur: 10, // 陰影模糊級別
  827. shadowOffsetX: 5, // 水平陰影偏移
  828. shadowOffsetY: 5 // 垂直陰影偏移
  829. },
  830. }
  831. }
  832. },
  833. layout: {
  834. padding: {
  835. left: 50, // 调整图表左边距
  836. right: 50, // 调整图表右边距
  837. top: 40,
  838. bottom: 10,
  839. }
  840. },
  841. },
  842. });
  843. // 注册 afterDraw 事件
  844. // Chart.register({
  845. // id: 'afterDraw',
  846. // afterDraw: function (chart) {
  847. // // console.log('處發', chart)
  848. // // 在这里添加你的绘制代码
  849. // var ctx = chart.ctx;
  850. // var xAxis = chart.scales.x;
  851. // var yAxis = chart.scales.y;
  852. // var dataj = myChart.data.datasets[0].data;
  853. // console.log('after', dataj)
  854. // ctx.save();
  855. // ctx.strokeStyle = '#000'; // 虚线颜色
  856. // ctx.lineWidth = 1; // 虚线宽度
  857. // ctx.setLineDash([5, 5]); // 设置虚线样式,[虚线段长度, 间隔长度]
  858. // dataj.forEach(function (value, index) {
  859. // var xPos = xAxis.getPixelForValue(labels[index]); // 获取 x 轴位置
  860. // var yPos = yAxis.getPixelForValue(value); // 获取 y 轴位置
  861. // // console.log(xPos)
  862. // // 绘制虚线
  863. // // ctx.beginPath();
  864. // // ctx.moveTo(xPos, yPos);
  865. // // ctx.lineTo(xPos, xAxis.bottom);
  866. // // ctx.stroke();
  867. // var xLineEnd = Math.min(yPos, yAxis.bottom);
  868. // // console.log('第' + index + '條線', xPos, yPos, 'xLineEnd', xLineEnd, 'yAxis.getPixelForValue(value)', yAxis.getPixelForValue(value));
  869. // // console.log('yAxis.getPixelForValue(value)', yAxis.getPixelForValue(value))
  870. // // 绘制虚线
  871. // ctx.beginPath();
  872. // ctx.moveTo(xPos, yAxis.getPixelForValue(value));
  873. // ctx.lineTo(xPos, xAxis.top);
  874. // ctx.stroke();
  875. // // 添加日期标签
  876. // ctx.fillStyle = '#000'; // 文字颜色
  877. // ctx.textAlign = 'center';
  878. // // ctx.fillText(labels[index], xPos, yAxis.bottom + 20); // 将日期标签绘制在虚线下方
  879. // });
  880. // ctx.restore();
  881. // }
  882. // });
  883. console.log(chart_bgimg_url);
  884. bgImgelement.style.backgroundImage = chart_bgimg_url;
  885. chartWhitelement.style.background = "rgba(255,255,255,0.5)"
  886. }
  887. // createChart(chartType);
  888. // createChart(chartType)
  889. const ChartOptions = document.querySelectorAll('input[name="ChartOptions"]');
  890. ChartOptions.forEach(button => {
  891. button.addEventListener('click', function () {
  892. // console.log(this.value);
  893. chartType = this.value;
  894. console.log(chartType);
  895. if (myChart !== null) {
  896. myChart.destroy();
  897. data = [];
  898. labels = [];
  899. }
  900. if (chartType === 'line') {
  901. $('#chartdiv').hide();
  902. generateChart(dataArray);
  903. } else {
  904. $('#textToChart').hide();
  905. extractAndGenerateChart(dataArray);
  906. }
  907. // createChart(chartType, data, labels);
  908. });
  909. });
  910. // 表格字體
  911. var fontSizeInput = document.getElementById('table_fontSize');
  912. // 添加事件监听器
  913. fontSizeInput.addEventListener('input', function () {
  914. // 获取输入框的值
  915. var fontSize = fontSizeInput.value;
  916. // 获取动态创建的表格元素
  917. var dynamicTable = document.querySelector('.dynamic-table');
  918. // 如果表格存在,则调整其字体大小
  919. if (dynamicTable) {
  920. dynamicTable.style.fontSize = fontSize + 'px';
  921. }
  922. });
  923. // 是否顯示point
  924. var togglePoints = document.getElementById('togglePoints');
  925. togglePoints.addEventListener('change', function () {
  926. console.log('checked')
  927. if (this.checked) {
  928. myChart.data.datasets[0].pointRadius = 5;
  929. myChart.data.datasets[0].pointHoverRadius = 8;
  930. } else {
  931. myChart.data.datasets[0].pointRadius = 0;
  932. myChart.data.datasets[0].pointHoverRadius = 0;
  933. }
  934. myChart.update();
  935. });
  936. // 單位標題
  937. // =========================
  938. var unitInput = document.getElementById('unit_data');
  939. unitInput.addEventListener('input', function () {
  940. unit = unitInput.value;
  941. // 重新渲染图表
  942. myChart.update();
  943. });
  944. // ==========================
  945. // y軸字體大小
  946. // ==========================
  947. var YfontSize = document.getElementById('YfontSize');
  948. YfontSize.addEventListener('input', function () {
  949. YfontSizeValue = YfontSize.value;
  950. if (myChart) {
  951. myChart.destroy();
  952. }
  953. createChart(chartType, data, labels);
  954. });
  955. // ==========================
  956. // x軸字體大小
  957. // ==========================
  958. var XfontSize = document.getElementById('XfontSize');
  959. XfontSize.addEventListener('input', function () {
  960. XfontSizeValue = XfontSize.value;
  961. console.log(XfontSizeValue)
  962. if (myChart) {
  963. myChart.destroy();
  964. }
  965. createChart(chartType, data, labels);
  966. });
  967. // ==========================
  968. // 線條粗細
  969. // =========================
  970. var rangeInput = document.getElementById('borderWidthRange');
  971. var rangeInputSpan = document.getElementById('borderWidthRangeValue');
  972. var borderWidthValue;
  973. rangeInputSpan.textContent = rangeInput.value;
  974. // 添加事件监听器,当滑动条的值发生改变时更新显示的值
  975. rangeInput.addEventListener('input', function () {
  976. // 获取滑动条的当前值
  977. borderWidthValue = rangeInput.value;
  978. console.log(borderWidthValue);
  979. rangeInputSpan.textContent = borderWidthValue;
  980. if (myChart) {
  981. myChart.destroy();
  982. }
  983. createChart(chartType, data, labels);
  984. // myChart.data.datasets[0].borderWidth = borderWidthValue;
  985. });
  986. // =========================
  987. // 線條顏色
  988. // =========================
  989. const colorInput = document.getElementById('borderColorInput');
  990. // 添加事件监听器,当颜色选择发生变化时触发
  991. colorInput.addEventListener('input', function () {
  992. // 获取当前选择的颜色值
  993. chartColor = colorInput.value;
  994. if (myChart) {
  995. myChart.destroy();
  996. }
  997. console.log(chartColor);
  998. createChart(chartType, data, labels);
  999. });
  1000. // =========================
  1001. // 模板樣式
  1002. // =========================
  1003. // JavaScript
  1004. document.addEventListener("DOMContentLoaded", function () {
  1005. // 獲取所有帶有 templateImg class 的圖像元素
  1006. var templateImgs = document.querySelectorAll('.templateImg');
  1007. // 為每個圖像元素添加點擊事件監聽器
  1008. templateImgs.forEach(function (img) {
  1009. img.addEventListener('click', function () {
  1010. // 在點擊時印出圖像元素的 value 屬性值
  1011. console.log(this.getAttribute('value'));
  1012. });
  1013. });
  1014. });
  1015. // =========================
  1016. var bgImgelement = document.getElementById('data_chart_box');
  1017. var chartWhitelement = document.getElementById('data_chartJs_box');
  1018. // 背景樣式
  1019. // =========================
  1020. // JavaScript
  1021. document.addEventListener("DOMContentLoaded", function () {
  1022. // 獲取所有帶有 templateImg class 的圖像元素
  1023. var bgImgs = document.querySelectorAll('.bgImg');
  1024. // 為每個圖像元素添加點擊事件監聽器
  1025. bgImgs.forEach(function (img) {
  1026. img.addEventListener('click', function () {
  1027. // 在點擊時印出圖像元素的 value 屬性值
  1028. console.log(this.getAttribute('value'));
  1029. var bgImgUrl = this.getAttribute('value');
  1030. chart_bgimg_url = 'url(' + bgImgUrl + ')';
  1031. bgImgelement.style.backgroundImage = chart_bgimg_url;
  1032. // bgImgelement.style.backgroundSize = 'cover'; // 調整背景圖片大小
  1033. // bgImgelement.style.backgroundPosition = 'center'; // 調整背景圖片位置
  1034. // bgImgelement.style.backgroundRepeat = 'no-repeat'; // 調整背景圖片重複
  1035. });
  1036. });
  1037. });
  1038. // =========================
  1039. const radioButtons = document.querySelectorAll('input[name="inlineRadioOptions"]');
  1040. var chartTitle = document.querySelector('.chart_title');
  1041. radioButtons.forEach(button => {
  1042. button.addEventListener('click', function () {
  1043. if (this.value === '0') {
  1044. chartTitle.style.textAlign = 'left';
  1045. } else if (this.value === '1') {
  1046. chartTitle.style.textAlign = 'center';
  1047. } else if (this.value === '2') {
  1048. chartTitle.style.textAlign = 'right';
  1049. }
  1050. });
  1051. });
  1052. var keywordInput = document.getElementById('title_data');
  1053. // 初始时将输入框的值设置为标题的文本内容
  1054. keywordInput.value = chartTitle.textContent;
  1055. // 监听输入框的输入事件,实现内容同步更新
  1056. keywordInput.addEventListener('input', function () {
  1057. chartTitle.textContent = this.value;
  1058. });
  1059. // 標題字體大小
  1060. // ==========================
  1061. var TitlefontSize = document.getElementById('TitlefontSize');
  1062. TitlefontSize.addEventListener('input', function () {
  1063. var TitlefontSizeValue = TitlefontSize.value;
  1064. chartTitle.style.fontSize = TitlefontSizeValue + 'px';;
  1065. });
  1066. // ==========================
  1067. let labelDependent = ""; // 圖表左側 Label
  1068. function get_data(input_text_value) {
  1069. inputField.value = "";
  1070. apiLoading();
  1071. var existingTable = chartJsBox.querySelector('table');
  1072. if (chartDiv.childElementCount != 0) {
  1073. data = [];
  1074. labels = [];
  1075. console.log('已存在')
  1076. chartTitle.textContent = "";
  1077. bgImgelement.style.backgroundImage = "";
  1078. chartWhitelement.style.background = "";
  1079. document.getElementById('img_box_url').src = "";
  1080. }
  1081. if (existingTable) {
  1082. chartJsBox.removeChild(existingTable);
  1083. chartTitle.textContent = "";
  1084. bgImgelement.style.backgroundImage = "";
  1085. chartWhitelement.style.background = "";
  1086. document.getElementById('img_box_url').src = "";
  1087. }
  1088. if (myChart) {
  1089. data = [];
  1090. labels = [];
  1091. chartTitle.textContent = "";
  1092. bgImgelement.style.backgroundImage = "";
  1093. chartWhitelement.style.background = "";
  1094. myChart.destroy();
  1095. document.getElementById('img_box_url').src = "";
  1096. }
  1097. axios.get('https://cmm.ai:8080/answer?question=' + input_text_value)
  1098. .then(response => {
  1099. apiHideLoading();
  1100. // 在這裡處理成功獲取 JSON 的情況
  1101. console.log(response);
  1102. var chart_info = response.data.chart_info.Title;
  1103. chartType = response.data.chart_info.Chart_type;
  1104. var finance_img_url = response.data.imageUrl_info.imageUrl;
  1105. if (response.data.imageUrl_info != null) {
  1106. console.log(finance_img_url);
  1107. document.getElementById('img_box_url').src = finance_img_url;
  1108. }
  1109. // switch (chartType) {
  1110. // case "line":
  1111. // document.getElementById("inlineRadio4").checked = true; // 折线图
  1112. // break;
  1113. // case "bar":
  1114. // document.getElementById("inlineRadio5").checked = true; // 柱状图
  1115. // labelDependent = response.data.chart_info.Label_dependent_variable;
  1116. // break;
  1117. // case "pie":
  1118. // document.getElementById("inlineRadio6").checked = true; // 圆饼图
  1119. // break;
  1120. // case "table":
  1121. // document.getElementById("inlineRadio7").checked = true; // 表格
  1122. // break;
  1123. // default:
  1124. // // 默认情况
  1125. // }
  1126. keywordInput.value = chart_info;
  1127. chartTitle.textContent = chart_info;
  1128. unit = response.data.chart_info.Unit_of_dependent_variable;
  1129. var Label_dependent_variable = response.data.chart_info.Label_dependent_variable;
  1130. var Label_independent_variable = response.data.chart_info.Label_independent_variable;
  1131. unit_data.value = response.data.chart_info.Unit_of_dependent_variable;
  1132. dataArray = response.data.data;
  1133. if (chartType === "table") {
  1134. createTable(dataArray, unit, Label_dependent_variable, Label_independent_variable);
  1135. // document.getElementById('img_box_url').src = "";
  1136. } else if (chartType === "bar") {
  1137. console.log('圖表類型bar')
  1138. $('#textToChart').hide();
  1139. extractAndGenerateChart(dataArray);
  1140. } else if (chartType === "doughnut") {
  1141. $('#textToChart').hide();
  1142. generatePieChart(dataArray)
  1143. } else {
  1144. $('#chartdiv').hide();
  1145. generateChart(dataArray);
  1146. }
  1147. // generateChart(dataArray);
  1148. })
  1149. .catch(error => {
  1150. apiHideLoading();
  1151. // 在這裡處理請求失敗的情況
  1152. console.error('發生錯誤:', error);
  1153. });
  1154. }
  1155. $(document).ready(function () {
  1156. $('#dataTable').DataTable();
  1157. });
  1158. function resizeChart() {
  1159. if (myChart) {
  1160. myChart.resize();
  1161. }
  1162. }
  1163. document.addEventListener('visibilitychange', function () {
  1164. console.log('Visibility changed:', document.visibilityState);
  1165. if (document.visibilityState === 'visible') {
  1166. resizeChart();
  1167. }
  1168. });
  1169. // 监听窗口resize事件
  1170. window.addEventListener('resize', resizeChart);
  1171. var seeting_button = document.getElementById('chart_seeting_button');
  1172. var closeButton = document.getElementById('style_setting_close');
  1173. var closeButton_table = document.getElementById('style_setting_close_table');
  1174. // var borderSetting = document.getElementById('borderSetting');
  1175. var lineChartSetting = document.getElementById('lineChartSetting');
  1176. var table_setting = document.getElementById('table_setting');
  1177. var chart_category = document.getElementById('chart_category');
  1178. // 获取 chartDiv 元素
  1179. seeting_button.addEventListener('click', () => {
  1180. console.log('setting_button')
  1181. if (myChart === null && !chartJsBox.querySelector('table') && chartDiv.childElementCount === 0) {
  1182. alert('請先生成圖表');
  1183. return
  1184. }
  1185. if (chartType === 'table') {
  1186. console.log('表格');
  1187. sidebar.classList.toggle('show');
  1188. sidebar.classList.toggle('hidden');
  1189. lineChartSetting.style.display = 'none';
  1190. table_setting.style.display = 'block';
  1191. chart_category.style.display = 'none';
  1192. } else if (chartType === "bar") {
  1193. console.log('柱狀圖')
  1194. sidebar.classList.toggle('show');
  1195. sidebar.classList.toggle('hidden');
  1196. lineChartSetting.style.display = 'none';
  1197. table_setting.style.display = 'none';
  1198. } else if (chartType === "doughnut") {
  1199. console.log('快捷功能圓餅圖')
  1200. sidebar.classList.toggle('show');
  1201. sidebar.classList.toggle('hidden');
  1202. lineChartSetting.style.display = 'none';
  1203. table_setting.style.display = 'none';
  1204. chart_category.style.display = 'none';
  1205. } else {
  1206. sidebar.classList.toggle('show');
  1207. sidebar.classList.toggle('hidden');
  1208. table_setting.style.display = 'none';
  1209. lineChartSetting.style.display = 'block';
  1210. }
  1211. });
  1212. closeButton.addEventListener('click', () => {
  1213. sidebar.classList.remove('show');
  1214. sidebar.classList.add('hidden');
  1215. });
  1216. const finance_options = [
  1217. { name: "原油油桶", imageUrl: "./img/finance/01.webp" },
  1218. { name: "黃金", imageUrl: "./img/finance/02.webp" },
  1219. { name: "房屋", imageUrl: "./img/finance/03.webp" },
  1220. { name: "車", imageUrl: "./img/finance/04.webp" },
  1221. { name: "股市", imageUrl: "./img/finance/05.webp" },
  1222. { name: "晶圓", imageUrl: "./img/finance/06.webp" },
  1223. { name: "TSMC 台積電", imageUrl: "./img/finance/07.webp" },
  1224. { name: "台幣", imageUrl: "./img/finance/08.webp" },
  1225. { name: "美金", imageUrl: "./img/finance/09.webp" },
  1226. { name: "購物車", imageUrl: "./img/finance/10.webp" },
  1227. { name: "美中貿易", imageUrl: "./img/finance/11.webp" },
  1228. { name: "世界地圖", imageUrl: "./img/finance/12.webp" },
  1229. { name: "小麥", imageUrl: "./img/finance/13.webp" },
  1230. { name: "玉米", imageUrl: "./img/finance/14.webp" },
  1231. { name: "智慧產線", imageUrl: "./img/finance/15.webp" }
  1232. ];
  1233. const selectOptions = document.getElementById('finance_options');
  1234. const selectOptions_table = document.getElementById('finance_options_table');
  1235. finance_options.forEach((option, index) => {
  1236. const optionElement = document.createElement('option');
  1237. optionElement.value = index + 1; // 值从 1 开始
  1238. optionElement.textContent = option.name;
  1239. optionElement.setAttribute('data-image-url', option.imageUrl); // 存储图片 URL
  1240. selectOptions.appendChild(optionElement);
  1241. });
  1242. selectOptions.addEventListener('change', function () {
  1243. const selectedOption = selectOptions.options[selectOptions.selectedIndex];
  1244. const imageUrl = selectedOption.getAttribute('data-image-url');
  1245. document.getElementById('img_box_url').src = imageUrl;
  1246. });
  1247. // finance_options.forEach((option, index) => {
  1248. // const optionElement = document.createElement('option');
  1249. // optionElement.value = index + 1; // 值从 1 开始
  1250. // optionElement.textContent = option.name;
  1251. // optionElement.setAttribute('data-image-url', option.imageUrl); // 存储图片 URL
  1252. // selectOptions_table.appendChild(optionElement);
  1253. // });
  1254. // selectOptions_table.addEventListener('change', function () {
  1255. // const selectedOption = selectOptions_table.options[selectOptions_table.selectedIndex];
  1256. // const imageUrl = selectedOption.getAttribute('data-image-url');
  1257. // document.getElementById('img_box_url').src = imageUrl;
  1258. // });
  1259. document.querySelectorAll('input[name="ChartOptions"]').forEach(function (radio) {
  1260. radio.addEventListener('change', function () {
  1261. // 檢查哪個單選按鈕被選中
  1262. if (this.value === 'line') {
  1263. // 如果選擇了折線圖,執行相應的操作
  1264. console.log("選擇了折線圖");
  1265. // 在這裡執行顯示折線圖的相關代碼
  1266. } else if (this.value === 'doughnut') {
  1267. // 如果選擇了柱狀圖,執行相應的操作
  1268. console.log("選擇了柱狀圖");
  1269. // 在這裡執行顯示柱狀圖的相關代碼
  1270. }
  1271. });
  1272. });