index.vue 59 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800
  1. <!-- -->
  2. <template>
  3. <div class="chart-show">
  4. <header class="chart-header" @click="openNew">
  5. <span
  6. class="chart-title"
  7. @click.stop
  8. @dblclick="copyText"
  9. :style="chartInfo.ChartThemeStyle?`
  10. text-align:${JSON.parse(chartInfo.ChartThemeStyle).titleOptions.align};
  11. font-size:${JSON.parse(chartInfo.ChartThemeStyle).titleOptions.style.fontSize}px;
  12. color:${JSON.parse(chartInfo.ChartThemeStyle).titleOptions.style.color}
  13. `:''"
  14. >
  15. {{ language === 'ch'?chartInfo.ChartName: chartInfo.ChartNameEn}}
  16. </span>
  17. </header>
  18. <template v-if="haveData">
  19. <div
  20. class="chart-wrapper"
  21. id="chart-wrapper"
  22. v-loading="loading"
  23. element-loading-text="加载中..."
  24. >
  25. <chart :options="options" :chartId="chartInfo.ChartInfoId || 0" :chartInfo="chartInfo"/>
  26. <div class="mark"></div>
  27. </div>
  28. </template>
  29. <div class="chart-wrapper notfound" v-else>
  30. <i class="el-icon-warning"></i>哎吆,你的图飞了,赶快去找管理员救命吧~
  31. </div>
  32. <!-- 图表来源说明 -->
  33. <div
  34. class="chart-bottom-info bootom-source"
  35. v-if="chartInfo.Instructions&&JSON.parse(chartInfo.Instructions).isShow
  36. ">
  37. <!-- 图表说明 -->
  38. <div
  39. class="chart-instruction text_oneLine"
  40. v-text="JSON.parse(chartInfo.Instructions).text"
  41. :style="`
  42. color: ${JSON.parse(chartInfo.Instructions).color};
  43. fontSize: ${ JSON.parse(chartInfo.Instructions).fontSize }px
  44. `"
  45. ></div>
  46. </div>
  47. <div class="bootom-source">
  48. <!-- 自定义来源 -->
  49. <div class="chart-source"
  50. v-if="chartInfo.SourcesFrom&&JSON.parse(chartInfo.SourcesFrom).isShow"
  51. :style="`
  52. color: ${ JSON.parse(chartInfo.SourcesFrom).color };
  53. fontSize: ${ JSON.parse(chartInfo.SourcesFrom).fontSize }px;
  54. `"
  55. >
  56. source:<em>{{ JSON.parse(chartInfo.SourcesFrom).text}}</em>
  57. </div>
  58. <temeplate v-else>
  59. <div
  60. v-if="$route.query.source=='smartReportGetImg'"
  61. class="text_oneLine"
  62. :style="`
  63. color: ${chartInfo.ChartThemeStyle&&JSON.parse(chartInfo.ChartThemeStyle).markerOptions.style.color };
  64. font-size: ${chartInfo.ChartThemeStyle&&JSON.parse(chartInfo.ChartThemeStyle).markerOptions.style.fontSize }px;
  65. `"
  66. >
  67. source:<span style="display:inline">{{language === 'ch' ? chartInfo.ChartSource : chartInfo.ChartSourceEn}}</span>
  68. </div>
  69. <strong
  70. v-else
  71. class="text_oneLine"
  72. :style="`
  73. flex:1;
  74. color: ${chartInfo.ChartThemeStyle&&JSON.parse(chartInfo.ChartThemeStyle).markerOptions.style.color };
  75. font-size: ${chartInfo.ChartThemeStyle&&JSON.parse(chartInfo.ChartThemeStyle).markerOptions.style.fontSize }px;
  76. `"
  77. >
  78. source: <em> {{language === 'ch' ? chartInfo.ChartSource : chartInfo.ChartSourceEn}}</em>
  79. </strong>
  80. </temeplate>
  81. <ul class="right-action" @click.stop>
  82. <li v-if="$route.query.source==='ybxcx'"><collectBtn/></li>
  83. <li @click="copyUrl" class="copy" v-if="isShare"><i class="el-icon-share"/>分享</li>
  84. <li @click="refreshChart" v-if="chartInfo.UniqueCode&&$route.query.source!=='smartReportGetImg'"><i class="el-icon-refresh"/>刷新</li>
  85. </ul>
  86. </div>
  87. </div>
  88. </template>
  89. <script lang="ts">
  90. import { defineComponent, reactive, toRefs, onMounted, ref } from 'vue';
  91. import { ElMessage } from 'element-plus';
  92. import _ from 'lodash';
  93. import { useRoute } from 'vue-router';
  94. import chart from '@/components/chart.vue';
  95. import { IState } from './typing';
  96. import { ChartApi } from '@/request/api';
  97. import { IChartType, IDataProps, ILunarItem, IParams, ISeasonDataItemProps } from '@/types';
  98. import Highcharts from 'highcharts';
  99. import { defaultOpts, seasonOptions } from '@/utils/chartOptions';
  100. import moment from 'moment';
  101. import collectBtn from './components/collectBtn.vue'
  102. // 散点x
  103. const scatterXAxis = {
  104. tickPosition: 'inside',
  105. lineColor: '#bfbfbf',
  106. tickColor: '#bfbfbf',
  107. tickLength:5,
  108. ordinal: false,
  109. type: 'linear',
  110. }
  111. //基础y轴
  112. const basicYAxis = {
  113. tickLength:5,
  114. lineWidth: 1,
  115. lineColor: '#bfbfbf',
  116. tickColor: '#bfbfbf',
  117. // offset: 0,
  118. visible: true,
  119. gridLineWidth: 0,
  120. tickPosition: 'inside',
  121. endOnTick: false,
  122. startOnTick: false,
  123. showLastLabel: true
  124. }
  125. export default defineComponent({
  126. components: {
  127. chart,
  128. collectBtn
  129. },
  130. setup() {
  131. const route = useRoute();
  132. const loading = ref(false);
  133. const haveData = ref(true);
  134. const code = ref(route.query.code);
  135. const isShare = ref(route.query.fromType === 'share');
  136. // 语言 中英文 ch en 默认中文
  137. const language = ref(route.query.lang || 'ch');
  138. const screen = ref(document.body.clientWidth < 1200 ? 'phone' : 'pc');
  139. const state = reactive<IState>({
  140. options: {},
  141. chartInfo: {},
  142. dataList: [],
  143. sourceName: '',
  144. /* 奇怪柱形图 */
  145. barDateList: [],//柱形图的绘图数据
  146. barXIdData: [],//柱形图的x轴id
  147. barEdbData: [],//柱形图的表格数据 只用于取值
  148. chartLimit: {},
  149. /* 商品价格曲线 */
  150. commodityChartData: [],
  151. commodityXData: [],
  152. commodityEdbList: [],
  153. /* 统计频率图 */
  154. statisticFrequencyData: {}
  155. });
  156. onMounted((): void => {
  157. getChartInfo();
  158. });
  159. // 打开新窗口
  160. const openNew = (): void => {
  161. window.open(window.location.href,'_blank');
  162. }
  163. /* 获取图表数据信息 */
  164. const getChartInfo = async (type='') => {
  165. if(!code.value) {
  166. haveData.value = false;
  167. return
  168. }
  169. loading.value = true;
  170. try {
  171. const { Data } = await ChartApi.getChart({
  172. UniqueCode: code.value || '',
  173. });
  174. loading.value = false;
  175. state.chartInfo = Data.ChartInfo;
  176. state.dataList = Data.ChartInfo.Source === 1 ? Data.EdbInfoList : [Data.EdbInfoList[0]];
  177. //处理英文研报英文设置不全就展示中文
  178. setLangFromEnReport();
  179. document.title = language.value==='ch'?Data.ChartInfo.ChartName:Data.ChartInfo.ChartNameEn;
  180. //eta图
  181. if(Data.ChartInfo.Source === 1) {
  182. const typeMap = {
  183. 7: initBarData,
  184. 10: initSectionScatterData
  185. }
  186. typeMap[state.chartInfo.ChartType] ? typeMap[state.chartInfo.ChartType](Data) : setOptions();
  187. }
  188. const sourceTypeMap = {
  189. 2: initCommodityData,
  190. 3: initRelevanceChartData,
  191. 4: initStatisticChartData,
  192. 5: initCommodityData,
  193. 6: initFittingEquation,
  194. 7: initStatisticChartData,
  195. 8: initStatisticChartData,
  196. 9: initStatisticChartData,
  197. }
  198. sourceTypeMap[Data.ChartInfo.Source] && sourceTypeMap[Data.ChartInfo.Source](Data);
  199. haveData.value = true;
  200. type === 'refresh' && ElMessage.success('刷新成功');
  201. //水印配置
  202. const markDom = document.querySelector('.mark')
  203. Data.WaterMark&&(markDom.style.backgroundImage = `url(${Data.WaterMark})`)
  204. }catch (e) {
  205. loading.value = false;
  206. haveData.value = false;
  207. }
  208. };
  209. /* 相关性图表 */
  210. const initRelevanceChartData=(data)=>{
  211. /* 主题样式*/
  212. const chartTheme = state.chartInfo.ChartThemeStyle ? JSON.parse(state.chartInfo.ChartThemeStyle) : null;
  213. const relevanceUnitEnMap={
  214. '年': 'Year',
  215. '季': 'Season',
  216. '月': 'Month',
  217. '周': 'Week',
  218. '天': 'Day',
  219. }
  220. // 处理X轴
  221. let xAxis={
  222. categories: data.ChartInfo.Source===3 ? data.XEdbIdValue : data.DataResp.XDateTimeValue,
  223. tickWidth: 1,
  224. title: {
  225. text: data.ChartInfo.Source===3 ? (language.value=='ch'?`期数(${data.CorrelationChartInfo.LeadUnit})`:`stage(${relevanceUnitEnMap[data.CorrelationChartInfo.LeadUnit]})`):null,
  226. align: 'high',
  227. rotation: 0,
  228. style: {
  229. ...chartTheme&&chartTheme.xAxisOptions.style
  230. }
  231. },
  232. labels: {
  233. style: {
  234. ...chartTheme&&chartTheme.xAxisOptions.style
  235. }
  236. },
  237. tickInterval: 1,
  238. offset:0,
  239. tickmarkPlacement:'on'
  240. }
  241. // 处理Y轴
  242. let yAxis={
  243. ...basicYAxis,
  244. title: {
  245. text: language.value=='ch'?'相关性系数':'Correlation coefficient',
  246. align: 'high',
  247. rotation: 0,
  248. y: -15,
  249. textAlign: 'left',
  250. reserveSpace: false,
  251. style:{
  252. ...chartTheme&&chartTheme.yAxisOptions.style
  253. },
  254. },
  255. labels: {
  256. formatter: function (ctx) {
  257. let val = ctx.value;
  258. return val;
  259. },
  260. align: 'center',
  261. style:{
  262. ...chartTheme&&chartTheme.yAxisOptions.style
  263. },
  264. },
  265. opposite: false,
  266. tickWidth: 1,
  267. }
  268. //处理series
  269. let seriesData:any[]=[]
  270. data.YDataList.forEach(item=>{
  271. let serie_item = {
  272. data: item.Value,
  273. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  274. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  275. yAxis: 0,
  276. name: language.value=='ch'?data.ChartInfo.ChartName:data.ChartInfo.ChartNameEn,
  277. color: item.Color,
  278. chartType: 'linear',
  279. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 3,
  280. marker: {
  281. enabled: false
  282. }
  283. };
  284. seriesData.push(serie_item)
  285. })
  286. const { LeadValue,LeadUnit } = data.CorrelationChartInfo;
  287. let tooltip = {
  288. formatter: function() {
  289. const that:any = this;
  290. let str='';
  291. if(language.value=='ch'){
  292. str = `<p>相关性系数:${that.y.toFixed(4)}</p><br><p>领先${ data.ChartInfo.Source===3 ?that.x+'期' : LeadValue+LeadUnit}</p>`
  293. }else{
  294. str = `<p>Correlation coefficient:${that.y.toFixed(4)}</p><br><p>lead${ data.ChartInfo.Source===3 ? that.x+'stage' : LeadValue+relevanceUnitEnMap[LeadUnit]}</p>`
  295. }
  296. return str
  297. },
  298. }
  299. state.options = {
  300. isRelevanceChart:data.ChartInfo.Source===3,
  301. title: {
  302. text:'',
  303. },
  304. series: seriesData,
  305. yAxis: [yAxis] ,
  306. xAxis:xAxis,
  307. tooltip
  308. }
  309. }
  310. /* 拟合方程曲线 */
  311. const initFittingEquation = (data) => {
  312. state.dataList = [data.DataResp];
  313. setDefaultLineOptions();
  314. }
  315. /* 散点截面图 */
  316. const sectionScatterData = ref<any>({});
  317. const initSectionScatterData = (data: { DataResp:any,EdbInfoList:any[] }) => {
  318. sectionScatterData.value = data.DataResp;
  319. state.chartLimit.min=Number(data.DataResp.YMinValue)
  320. state.chartLimit.max=Number(data.DataResp.YMaxValue)
  321. state.chartLimit.x_min=Number(data.DataResp.XMinValue)
  322. state.chartLimit.x_max=Number(data.DataResp.XMaxValue)
  323. //校验英文名称是否完整
  324. if(route.query.fromPage === 'en' && language.value==='en') {
  325. const { XNameEn,XUnitNameEn,YNameEn,YUnitNameEn } = data.DataResp;
  326. // DataList可能是null
  327. const DataList = data.DataResp.DataList||[]
  328. let isAllEn = true;
  329. if(!XNameEn || !XUnitNameEn || !YNameEn || !YUnitNameEn) isAllEn = false;
  330. if(DataList.some(_ => !_.NameEn)) isAllEn = false;
  331. console.log(isAllEn)
  332. language.value = isAllEn ? 'en' : 'ch';
  333. }
  334. setSectionScatterChart()
  335. }
  336. /* 截面散点图设置 sectionScatterData */
  337. const setSectionScatterChart = () => {
  338. const { chartLimit } = state;
  339. const { DataList,XName,XNameEn,XUnitName,XUnitNameEn,YName,YNameEn,YUnitName,YUnitNameEn } = sectionScatterData.value;
  340. /* 主题样式*/
  341. const chartTheme = state.chartInfo.ChartThemeStyle ? JSON.parse(state.chartInfo.ChartThemeStyle) : null;
  342. if(!DataList){
  343. return
  344. }
  345. const { min,max,x_min,x_max } = chartLimit;
  346. //y轴
  347. let yAxis = {
  348. ...basicYAxis,
  349. title: {
  350. text: language.value === 'ch' ? YName : YNameEn,
  351. align: 'middle',
  352. style:{
  353. ...chartTheme&&chartTheme.yAxisOptions.style
  354. },
  355. },
  356. labels: {
  357. style:{
  358. ...chartTheme&&chartTheme.yAxisOptions.style
  359. },
  360. },
  361. opposite: false,
  362. reversed: false,
  363. min: Number(min),
  364. max: Number(max),
  365. tickWidth: 1,
  366. plotBands: setAxisPlotAreas(1),
  367. plotLines: setAxisPlotLines(1)
  368. }
  369. //x轴
  370. let xAxis = {
  371. ...scatterXAxis,
  372. title: {
  373. text: language.value === 'ch' ? XName : XNameEn,
  374. align: 'middle',
  375. style: {
  376. ...chartTheme&&chartTheme.xAxisOptions.style
  377. },
  378. },
  379. labels: {
  380. style:{
  381. ...chartTheme&&chartTheme.xAxisOptions.style
  382. },
  383. },
  384. min: Number(x_min),
  385. max: Number(x_max),
  386. plotBands: setAxisPlotAreas(3),
  387. plotLines: setAxisPlotLines(3)
  388. }
  389. //数据列
  390. let series: any[] = [];
  391. DataList.forEach(item => {
  392. //数据列
  393. let series_item = {
  394. data: [] as any[],
  395. type: 'scatter',
  396. name: language.value === 'ch' ? item.Name : item.NameEn,
  397. color: item.Color,
  398. lineWidth: 0,
  399. chartType: 'linear',
  400. zIndex:1,
  401. marker: {
  402. radius: (chartTheme&&chartTheme.lineOptions.radius)||5,
  403. },
  404. visible: true
  405. }
  406. item.EdbInfoList.forEach(_ => {
  407. series_item.data.push({
  408. x: _.XValue,
  409. y: _.YValue,
  410. dataLabels: {
  411. enabled: _.IsShow,
  412. allowOverlap: true,
  413. align: 'left',
  414. format: language.value === 'ch' ? _.Name : _.NameEn
  415. }
  416. })
  417. })
  418. series.push(series_item);
  419. //趋势线
  420. if(item.ShowTrendLine) {
  421. let trend_data = item.TrendLimitData.map((_,_index) => (
  422. _index === item.TrendLimitData.length-1 ? {
  423. x: _.X,
  424. y: _.Y,
  425. dataLabels: {
  426. enabled: item.ShowRSquare || item.ShowFitEquation,
  427. align: 'left',
  428. color: '#666',
  429. x: 20,
  430. y: 30,
  431. zIndex: 9,
  432. allowOverlap: true,
  433. formatter: function(){
  434. let tag = '';
  435. item.ShowRSquare && item.ShowFitEquation
  436. ? tag =`<span>${item.TrendLine}</span><br><span>R²=${item.RSquare}</span>`
  437. : item.ShowRSquare && !item.ShowFitEquation
  438. ? tag =`<span>R²=${item.RSquare}</span>`
  439. : item.ShowFitEquation && !item.ShowRSquare
  440. ? tag =`<span>${item.TrendLine}</span>`
  441. : ''
  442. return tag
  443. }
  444. }
  445. } : {
  446. x: _.X,
  447. y: _.Y,
  448. }
  449. ))
  450. let trend_item = {
  451. data: trend_data,
  452. type: 'spline',
  453. linkedTo: ':previous',
  454. color: item.Color,
  455. lineWidth: 1,
  456. chartType: 'linear',
  457. enableMouseTracking: false,
  458. dashStyle:'Dash',
  459. zIndex: 2,
  460. visible: true,
  461. marker: {
  462. enabled: false
  463. }
  464. }
  465. series.push(trend_item)
  466. }
  467. })
  468. let tooltip = {
  469. formatter: function() {
  470. const that:any = this;
  471. let str = '';
  472. if(language.value === 'ch') {
  473. let series_obj = DataList.find(_ => _.Name === that.series.name);
  474. let ponit_obj = series_obj.EdbInfoList.find(_ => _.XValue ===that.x && _.YValue===that.y);
  475. str=`<b>${ ponit_obj.Name }</b>`;
  476. str += `<br><span style="color:${that.color}">\u25CF</span>${ponit_obj.XName}: ${that.x} ${ponit_obj.XDate}<br>`;
  477. str += `<span style="color:${that.color}">\u25CF</span>${ponit_obj.YName}: ${that.y} ${ponit_obj.YDate}`;
  478. }else {
  479. let series_obj = DataList.find(_ => _.NameEn === that.series.name);
  480. let ponit_obj = series_obj.EdbInfoList.find(_ => _.XValue ===that.x && _.YValue===that.y);
  481. str=`<b>${ ponit_obj.NameEn }</b>`;
  482. str += `<br><span style="color:${that.color}">\u25CF</span>${ponit_obj.XNameEn}: ${that.x} ${ponit_obj.XDate}<br>`;
  483. str += `<span style="color:${that.color}">\u25CF</span>${ponit_obj.YNameEn}: ${that.y} ${ponit_obj.YDate}`;
  484. }
  485. return str
  486. }
  487. }
  488. state.options = {
  489. title: {
  490. text:''
  491. },
  492. series,
  493. yAxis: [yAxis],
  494. xAxis,
  495. tooltip
  496. }
  497. }
  498. const correlationChartInfo = ref<any>({});
  499. /* 统计特征 标准差 百分比 频率 滚动相关性*/
  500. const initStatisticChartData = (data: { DataResp:any,ChartInfo:any,CorrelationChartInfo:any }) => {
  501. if(data.ChartInfo.Source === 9) { //频率图
  502. state.statisticFrequencyData = data.DataResp;
  503. setStatisticFrequency();
  504. }else {
  505. state.dataList = [data.DataResp];
  506. if(data.ChartInfo.Source === 4) correlationChartInfo.value = data.CorrelationChartInfo;
  507. setDefaultLineOptions();
  508. }
  509. }
  510. /* 统计频率图 */
  511. const setStatisticFrequency = () => {
  512. const { DataList,LeftMaxValue,LeftMinValue,RightMaxValue,RightMinValue } = state.statisticFrequencyData;
  513. if(!DataList){
  514. return
  515. }
  516. /* 主题样式*/
  517. const chartTheme = state.chartInfo.ChartThemeStyle ? JSON.parse(state.chartInfo.ChartThemeStyle) : null;
  518. let xAxis = {
  519. ...scatterXAxis,
  520. tickWidth: 1,
  521. labels: {
  522. style: {
  523. ...chartTheme&&chartTheme.xAxisOptions.style
  524. }
  525. }
  526. }
  527. //y和系列
  528. let yAxis:any[] = [],series:any[] = [];
  529. DataList.forEach((item,index) => {
  530. let y_item = {
  531. ...basicYAxis,
  532. title: {
  533. text: language.value === 'ch' ? item.Unit : item.UnitEn,
  534. align: 'high',
  535. rotation: 0,
  536. y: -15,
  537. reserveSpace: false,
  538. style:{
  539. ...chartTheme&&chartTheme.yAxisOptions.style
  540. },
  541. },
  542. labels: {
  543. style:{
  544. ...chartTheme&&chartTheme.yAxisOptions.style
  545. },
  546. },
  547. opposite: item.IsAxis===1?false:true,
  548. min: index===0? Number(LeftMinValue):Number(RightMinValue),
  549. max: index===0? Number(LeftMaxValue):Number(RightMaxValue),
  550. tickWidth: 1,
  551. }
  552. let series_item = {
  553. data: item.Value.map(_ =>[_.X,_.Y]),
  554. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  555. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  556. yAxis: index,
  557. name: language.value === 'ch' ? item.Name : item.NameEn,
  558. color: item.Color,
  559. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth)||3,
  560. chartType: 'linear',
  561. zIndex:1
  562. }
  563. series.push(series_item);
  564. yAxis.push(y_item)
  565. })
  566. let tooltip = {
  567. formatter: function() {
  568. let that:any = this;
  569. let xList = DataList[0].Value.map(_ =>_.X);
  570. let step = xList[1]-xList[0];
  571. let data_interval = `[${Number(that.x).toFixed(2)},${Number(that.x+step).toFixed(2)}]`;
  572. let str=`<b>${ data_interval }</b>`;
  573. that.points.forEach(item => {
  574. str += `<br><span style="color:${item.color}">\u25CF</span>${item.series.name}: ${item.y}%`
  575. })
  576. return str
  577. },
  578. shared: true
  579. }
  580. state.options = {
  581. title: {
  582. text:''
  583. },
  584. tooltip,
  585. series,
  586. yAxis,
  587. xAxis
  588. }
  589. }
  590. //处理英文研报的图表英文设置不全的情况
  591. const setLangFromEnReport = () => {
  592. //来源于英文研报
  593. if(route.query.fromPage !== 'en') return
  594. let is_name_en = state.chartInfo.ChartNameEn ? true : false;//名称是否有英文
  595. let is_target_en = [2,9].includes(state.chartInfo.ChartType) ? true : state.dataList.every(_ => _.EdbNameEn);//指标是否有英文
  596. console.log(is_name_en,is_target_en)
  597. language.value = (is_name_en && is_target_en) ? 'en' : 'ch';
  598. }
  599. // 曲线图x轴显示计算年限差 >1年 显示年/月 <=1 显示月/日
  600. const xTimeDiffer = () => {
  601. const end_date = state.chartInfo.DateType === 5
  602. ? state.chartInfo.EndDate
  603. : state.chartInfo.DateType === 6
  604. ? new Date()
  605. : '';
  606. //年限差
  607. const year_differ = moment(end_date).diff(
  608. moment(state.chartInfo.StartDate),
  609. 'years',
  610. true
  611. );
  612. // console.log(year_differ)
  613. if ([5, 6].includes(state.chartInfo.DateType) && year_differ <= 1) {
  614. return true
  615. } else {
  616. return false
  617. }
  618. }
  619. /* 曲线图 */
  620. const setDefaultLineOptions = () => {
  621. const { dataList,chartInfo } = state;
  622. /* 主题样式*/
  623. const chartTheme = chartInfo.ChartThemeStyle ? JSON.parse(chartInfo.ChartThemeStyle) : null;
  624. //拼接标题 数据列
  625. let data = [] as any[],ydata = [] as any[],minTimeArr: number[] = [],maxTimeArr: number[] = [];
  626. let rightTwoIndex = dataList.findIndex(item => item.IsAxis ===2);
  627. // const chartData = _.cloneDeep(dataList);
  628. //有右二轴时排个序 按照左 右 右2的顺序
  629. let chartData = dataList.some(_ =>_.IsAxis===2) ? changeEdbOrder(dataList) : _.cloneDeep(dataList);
  630. chartData.forEach((item:IDataProps ,index:number) => {
  631. //轴位置值相同的下标
  632. let sameSideIndex = chartData.findIndex(
  633. (i:IDataProps) => i.IsAxis === item.IsAxis
  634. );
  635. //y轴
  636. let textEn = item.Unit?item.UnitEn:''
  637. let yItem = {
  638. ...basicYAxis,
  639. labels: {
  640. formatter: function (ctx: any) {
  641. let val = ctx.value;
  642. return sameSideIndex !== index ? '' : val;
  643. },
  644. align: 'center',
  645. x: [0,2].includes(item.IsAxis) ? 5 : -5,
  646. style: {
  647. ...chartTheme&&chartTheme.yAxisOptions.style,
  648. }
  649. },
  650. title: {
  651. text:language.value=='ch'?sameSideIndex !== index ? '' : `${item.Unit}`:textEn,
  652. style:{
  653. ...chartTheme&&chartTheme.yAxisOptions.style
  654. },
  655. align: 'high',
  656. rotation: 0,
  657. y: -15,
  658. x: (item.IsAxis===0 && rightTwoIndex>-1) ? -chartData[rightTwoIndex].Unit.length*12 : 0,
  659. textAlign: item.IsAxis===1 ? 'left' : 'right',
  660. reserveSpace: false
  661. },
  662. opposite: [0,2].includes(item.IsAxis),
  663. reversed: item.IsOrder,
  664. min: Number(chartData[sameSideIndex].MinData),
  665. max: Number(chartData[sameSideIndex].MaxData),
  666. tickWidth: 1,
  667. visible: sameSideIndex === index,
  668. plotBands: setAxisPlotAreas(item.IsAxis),
  669. plotLines: setAxisPlotLines(item.IsAxis)
  670. }
  671. // //拼接标题 判断相同指标名称拼接来源
  672. let dynamic_title = item.EdbName;
  673. let dynamic_arr = chartData.filter(
  674. (item: IDataProps) => dynamic_title === item.EdbName
  675. );
  676. // 拼接配置 IsAxis左轴1 右轴0 IsOrder正序false 逆序true EdbInfoType是否是领先指标
  677. let dynamic_tag = concatDynamicTag(item)
  678. // 英文后缀
  679. let dynamic_tag_en = concatDynamicTag(item,'en')
  680. //数据列
  681. let nameCh:String = dynamic_arr.length > 1
  682. ? `${item.EdbName}(${item.SourceName})${dynamic_tag}`
  683. : `${item.EdbName}${dynamic_tag}`
  684. let nameEn:String=item.EdbNameEn?`${item.EdbNameEn}${dynamic_tag_en}`:''
  685. let name :String=language.value == 'ch'?nameCh:nameEn
  686. //预测指标配置
  687. let predict_params = item.EdbInfoCategoryType === 1 ? getPredictParams(item) : {};
  688. let obj = {
  689. data: [] as any[],
  690. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  691. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  692. yAxis: sameSideIndex,
  693. name,
  694. color: item.ChartColor,
  695. lineWidth: Number(item.ChartWidth),
  696. ...predict_params
  697. };
  698. item.DataList = item.DataList || []
  699. for (let i of item.DataList) {
  700. obj.data.push([i.DataTimestamp, i.Value]);
  701. }
  702. if(item.DataList.length){
  703. minTimeArr.push(item.DataList[0].DataTimestamp)
  704. maxTimeArr.push(item.DataList[item.DataList.length-1].DataTimestamp)
  705. }
  706. data.push(obj);
  707. ydata.push(yItem);
  708. })
  709. // 范围为1年内 x轴显示为月/日 否则默认年/月
  710. let xAxis:any = {};
  711. const isLessThanOneYear:boolean = xTimeDiffer();
  712. let minTime: number=Math.min(...minTimeArr);
  713. let maxTime=Math.max(...maxTimeArr);
  714. let step = setXaxisStep(maxTime-minTime);
  715. xAxis = {
  716. ...defaultOpts.xAxis,
  717. tickInterval: screen.value === 'phone' ? step : undefined,
  718. labels: {
  719. formatter: function (ctx: any) {
  720. return isLessThanOneYear
  721. ? Highcharts.dateFormat('%m/%d', ctx.value)
  722. : Highcharts.dateFormat('%y/%m', ctx.value);
  723. },
  724. style: {
  725. ...chartTheme&&chartTheme.xAxisOptions.style
  726. }
  727. },
  728. plotBands: setAxisPlotAreas(3,'datetime'),
  729. plotLines: setAxisPlotLines(3,'datetime')
  730. }
  731. state.options = {
  732. series: data,
  733. yAxis: ydata,
  734. xAxis
  735. };
  736. //滚动相关性独立tooltip
  737. if(chartInfo.Source === 4) {
  738. const relevanceUnitEnMap={
  739. '年': 'Year',
  740. '季': 'Season',
  741. '月': 'Month',
  742. '周': 'Week',
  743. '天': 'Day',
  744. }
  745. const { LeadValue,LeadUnit } = correlationChartInfo.value;
  746. state.options.tooltip = {
  747. formatter: function() {
  748. let that:any = this;
  749. let str = '';
  750. if(language.value=='ch'){
  751. str = `${Highcharts.dateFormat('%Y/%m/%d',that.x)}<br><p>相关性系数:${that.y.toFixed(4)}</p><br><p>领先${LeadValue+LeadUnit}</p>`
  752. }else{
  753. str = `${Highcharts.dateFormat('%Y/%m/%d',that.x)}<br><p>Correlation coefficient:${that.y.toFixed(4)}</p><br><p>lead${LeadValue+relevanceUnitEnMap[LeadUnit]}</p>`
  754. }
  755. return str
  756. },
  757. }
  758. }
  759. }
  760. /* 设置x轴步长 刻度数量多一点 */
  761. const setXaxisStep = (timestamp: number) => {
  762. // return timestamp / 6 < 24 * 3600 * 1000 * 30 ? 24 * 3600 * 1000 * 30 : timestamp / 6;
  763. return timestamp / 6;
  764. }
  765. /* 指标顺序调整 IsAxis: 0右轴 1左轴 2右2*/
  766. const changeEdbOrder = (data: any[]) => {
  767. // 左轴指标
  768. let left_edbs = data.filter(_ => _.IsAxis===1);
  769. //右轴指标
  770. let right_edbs = data.filter(_ => !_.IsAxis);
  771. // 右2轴指标
  772. let right_two_edbs = data.filter(_ => _.IsAxis === 2);
  773. // 按 左 右 右2顺序排列
  774. return [left_edbs,right_edbs,right_two_edbs].flat(Infinity);
  775. }
  776. /* 堆叠图/组合图设置
  777. 本来和曲线图逻辑基本一致兼容下即可 为了以后便于维护和阅读还是拆开写吧
  778. */
  779. const setStackOrCombinChart = () => {
  780. const { dataList,chartInfo } = state;
  781. /* 主题样式*/
  782. const chartTheme = state.chartInfo.ChartThemeStyle ? JSON.parse(state.chartInfo.ChartThemeStyle) : null;
  783. //拼接标题 数据列
  784. let data = [] as any[],ydata = [] as any[],minTimeArr: number[] = [],maxTimeArr: number[] = [];
  785. // const chartData = _.cloneDeep(dataList);
  786. //有右二轴时排个序 按照左 右 右2的顺序
  787. let chartData = dataList.some(_ =>_.IsAxis===2) ? changeEdbOrder(dataList) : _.cloneDeep(dataList);
  788. //支持的图表类型
  789. const chartTypeMap: IChartType = {
  790. 3: 'areaspline',
  791. 4: 'column',
  792. 6: ''
  793. };
  794. let chartStyle = chartTypeMap[chartInfo.ChartType];
  795. chartData.forEach((item:IDataProps ,index:number) => {
  796. //轴位置值相同的下标
  797. let sameSideIndex = chartData.findIndex(
  798. (i:IDataProps) => i.IsAxis === item.IsAxis
  799. );
  800. //堆叠图的yAxis必须一致 数据列所对应的y轴
  801. let serie_yIndex = index;
  802. if([3,4].includes(chartInfo.ChartType)) {
  803. // 类型为堆叠图时公用第一个指标y轴
  804. serie_yIndex = 0;
  805. } else if(chartInfo.ChartType ===6 && ['areaspline','column'].includes(item.ChartStyle)) {
  806. // 组合图找第一个堆叠柱状或面积的作为公用
  807. serie_yIndex = chartData.findIndex((i:IDataProps) => i.ChartStyle === item.ChartStyle);
  808. }
  809. //数据对应的y轴是公用轴则配置也共享
  810. item.IsAxis = serie_yIndex === index ? item.IsAxis : chartData[serie_yIndex].IsAxis;
  811. item.IsOrder = serie_yIndex === index ? item.IsOrder : chartData[serie_yIndex].IsOrder;
  812. // 右2轴下标
  813. let rightTwoIndex = [3,4].includes(chartInfo.ChartType)
  814. ? -1
  815. : dataList.findIndex(item => item.IsAxis===2);
  816. //y轴
  817. let textEn = item.Unit?item.UnitEn:''
  818. let yItem = {
  819. ...basicYAxis,
  820. labels: {
  821. formatter: function (ctx: any) {
  822. let val = ctx.value;
  823. return sameSideIndex !== index ? '' : val;
  824. },
  825. align: 'center',
  826. x: [0,2].includes(item.IsAxis) ? 5 : -5,
  827. style:{
  828. ...chartTheme&&chartTheme.yAxisOptions.style
  829. },
  830. },
  831. title: {
  832. text:language.value=='ch'?sameSideIndex !== index ? '' : `${item.Unit}`:textEn,
  833. // text: null,
  834. align: 'high',
  835. rotation: 0,
  836. y: -15,
  837. x: (item.IsAxis===0 && rightTwoIndex>-1) ? -chartData[rightTwoIndex].Unit.length*12 : 0,
  838. textAlign: item.IsAxis===1 ? 'left' : 'right',
  839. reserveSpace: false,
  840. style:{
  841. ...chartTheme&&chartTheme.yAxisOptions.style
  842. },
  843. },
  844. opposite: [0,2].includes(item.IsAxis),
  845. reversed: item.IsOrder,
  846. min: Number(chartData[sameSideIndex].MinData),
  847. max: Number(chartData[sameSideIndex].MaxData),
  848. tickWidth: sameSideIndex !== index ? 0 : 1,
  849. visible: serie_yIndex === index && sameSideIndex ===index,
  850. plotBands: setAxisPlotAreas(item.IsAxis),
  851. plotLines: setAxisPlotLines(item.IsAxis)
  852. }
  853. // //拼接标题 判断相同指标名称拼接来源
  854. let dynamic_title = item.EdbName;
  855. let dynamic_arr = chartData.filter(
  856. (item: IDataProps) => dynamic_title === item.EdbName
  857. );
  858. // 拼接配置 IsAxis左轴1 右轴0 IsOrder正序false 逆序true EdbInfoType是否是领先指标
  859. let dynamic_tag = concatDynamicTag(item)
  860. // 英文后缀
  861. let dynamic_tag_en = concatDynamicTag(item,'en')
  862. //数据列
  863. let nameCh:String = dynamic_arr.length > 1
  864. ? `${item.EdbName}(${item.SourceName})${dynamic_tag}`
  865. : `${item.EdbName}${dynamic_tag}`
  866. let nameEn:String=item.EdbNameEn?`${item.EdbNameEn}${dynamic_tag_en}`:''
  867. let name :String=language.value == 'ch'?nameCh:nameEn
  868. //预测指标配置
  869. let predict_params = item.EdbInfoCategoryType === 1 ? getPredictParams(item,chartStyle) : {};
  870. let obj = {
  871. data: [] as any[],
  872. type: chartStyle || item.ChartStyle,
  873. yAxis: serie_yIndex,
  874. name,
  875. color: item.ChartColor,
  876. lineWidth: Number(item.ChartWidth),
  877. fillColor: (chartInfo.ChartType === 3 || (chartInfo.ChartType === 6 && item.ChartStyle === 'areaspline')) ? item.ChartColor : undefined,
  878. zIndex: (chartInfo.ChartType === 6 && ['line','spline'].includes(item.ChartStyle)) ? 1 : 0, //防止组合图曲线被遮住
  879. borderWidth: 1,
  880. borderColor: item.ChartColor,
  881. ...predict_params
  882. };
  883. item.DataList = item.DataList || []
  884. for (let i of item.DataList) {
  885. obj.data.push([i.DataTimestamp, i.Value]);
  886. }
  887. if(item.DataList.length){
  888. minTimeArr.push(item.DataList[0].DataTimestamp)
  889. maxTimeArr.push(item.DataList[item.DataList.length-1].DataTimestamp)
  890. }
  891. data.push(obj);
  892. ydata.push(yItem);
  893. })
  894. // 范围为1年内 x轴显示为月/日 否则默认年/月
  895. let xAxis:any = {};
  896. const isLessThanOneYear:boolean = xTimeDiffer();
  897. let minTime: number=Math.min(...minTimeArr);
  898. let maxTime=Math.max(...maxTimeArr);
  899. let step = setXaxisStep(maxTime-minTime);
  900. xAxis = {
  901. ...defaultOpts.xAxis,
  902. tickInterval: screen.value === 'phone' ? step : undefined,
  903. labels: {
  904. formatter: function (ctx: any) {
  905. return isLessThanOneYear
  906. ? Highcharts.dateFormat('%m/%d', ctx.value)
  907. : Highcharts.dateFormat('%y/%m', ctx.value);
  908. },
  909. style: {
  910. ...chartTheme&&chartTheme.xAxisOptions.style
  911. }
  912. },
  913. plotBands: setAxisPlotAreas(3,'datetime'),
  914. plotLines: setAxisPlotLines(3,'datetime')
  915. }
  916. state.options = {
  917. series: data,
  918. yAxis: ydata,
  919. xAxis
  920. };
  921. }
  922. /* 季节图 */
  923. const setSeasonOptions = () => {
  924. const chartData = state.dataList[0];
  925. // 农历数据需要去除第一项 在ETA1.0.5之后,除了这里 农历和公历处理逻辑一样
  926. if(!chartData.DataList){
  927. return
  928. }
  929. const chartDataHandle=state.chartInfo.Calendar === '农历'?
  930. chartData.DataList.filter((item:ISeasonDataItemProps, index:number) => index > 0):
  931. chartData.DataList
  932. let seasonYdata:any[] = [],
  933. seasonData:any[] = [];
  934. /* 主题样式*/
  935. const chartTheme = state.chartInfo.ChartThemeStyle ? JSON.parse(state.chartInfo.ChartThemeStyle) : null;
  936. /*处理数据列*/
  937. for (let j of chartDataHandle) {
  938. //预测指标配置
  939. let predict_params = chartData.EdbInfoCategoryType === 1 ? getSeasonPredictParams(j.CuttingDataTimestamp) : {};
  940. let serie_item = {
  941. data: [] as any[],
  942. type: (chartTheme&&chartTheme.lineOptions.lineType) || chartData.ChartStyle,
  943. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  944. yAxis: 0,
  945. name: j.ChartLegend,
  946. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 1,
  947. ...predict_params
  948. };
  949. const data_array = _.cloneDeep(j.DataList);
  950. data_array &&
  951. data_array.forEach((item: IParams) => {
  952. serie_item.data.push([item.DataTimestamp, item.Value]);
  953. });
  954. seasonData.push(serie_item);
  955. }
  956. //y轴
  957. seasonYdata = [{
  958. ...defaultOpts.yAxis,
  959. labels: {
  960. formatter: function (ctx: any) {
  961. let val = ctx.value;
  962. return val;
  963. },
  964. align: 'center',
  965. style: {
  966. ...chartTheme&&chartTheme.yAxisOptions.style
  967. }
  968. },
  969. title: {
  970. text:language.value=='ch'?`${chartData.Unit}`:chartData.UnitEn,
  971. align: 'high',
  972. rotation: 0,
  973. y: -15,
  974. x: 0,
  975. textAlign: 'left',
  976. reserveSpace: false,
  977. style:{
  978. ...chartTheme&&chartTheme.yAxisOptions.style
  979. },
  980. },
  981. max: Number(chartData.MaxData),
  982. min: Number(chartData.MinData),
  983. plotBands: setAxisPlotAreas(1),
  984. plotLines: setAxisPlotLines(1)
  985. }];
  986. // 季节图x轴显示月/日
  987. const xAxis = {
  988. ...defaultOpts.xAxis,
  989. tickInterval: screen.value === 'phone' ? 24 * 3600 * 1000 * 60 : undefined,
  990. labels: {
  991. formatter: function (ctx: any) {
  992. return Highcharts.dateFormat('%m/%d', ctx.value);
  993. },
  994. style: {
  995. ...chartTheme&&chartTheme.xAxisOptions.style
  996. }
  997. },
  998. plotBands: setAxisPlotAreas(3,'datetime'),
  999. plotLines: setAxisPlotLines(3,'datetime')
  1000. };
  1001. const tooltip = {
  1002. ...defaultOpts.tooltip,
  1003. dateTimeLabelFormats: {
  1004. // 时间格式化字符
  1005. day: '%m/%d',
  1006. week: '%m/%d',
  1007. month: '%m/%d',
  1008. year: '%m/%d',
  1009. },
  1010. xDateFormat: '%m/%d',
  1011. }
  1012. state.options = {
  1013. colors:seasonOptions.colors.slice(-chartDataHandle.length),
  1014. series: seasonData,
  1015. yAxis: seasonYdata,
  1016. xAxis,
  1017. tooltip
  1018. };
  1019. }
  1020. /* 散点图 第一个指标值为x轴 第二个指标为y轴*/
  1021. const setScatterChartOptions = () => {
  1022. const { dataList,chartInfo } = state;
  1023. /* 主题样式*/
  1024. const chartTheme = state.chartInfo.ChartThemeStyle ? JSON.parse(state.chartInfo.ChartThemeStyle) : null;
  1025. // 取2个指标中日期相同的数据
  1026. const real_data: any[] = [];
  1027. let tmpData_date: any = {};//用来取点对应的日期
  1028. let data1 = _.cloneDeep(dataList)[0].DataList || [];
  1029. let data2 = _.cloneDeep(dataList)[1].DataList || [];
  1030. data1.forEach((_item: IParams) => {
  1031. data2.forEach((_item2: IParams) => {
  1032. if(_item.DataTimestamp === _item2.DataTimestamp) {
  1033. //日期
  1034. let itemIndex =_item.Value + "_" +_item2.Value
  1035. if(tmpData_date[itemIndex]) {
  1036. tmpData_date[itemIndex].push( moment(_item.DataTimestamp).format('YYYY/MM/DD'))
  1037. } else {
  1038. tmpData_date[itemIndex] = [moment(_item.DataTimestamp).format('YYYY/MM/DD')]
  1039. }
  1040. //值
  1041. real_data.push({
  1042. x: _item.Value,
  1043. y: _item2.Value
  1044. })
  1045. }
  1046. })
  1047. })
  1048. real_data.sort((x,y) => x-y);
  1049. //悬浮窗 拼接日期 原始指标名称
  1050. let tooltip = {
  1051. formatter: function() {
  1052. const that: any = this;
  1053. return language.value=='ch'?
  1054. `<strong>${ tmpData_date[that.x+'_'+that.y].length > 4 ? tmpData_date[that.x+'_'+that.y].slice(0,4).join()+'...' : tmpData_date[that.x+'_'+that.y].join() }</strong><br>
  1055. ${dataList[0].EdbName}: <span style="font-weight: 600"> ${that.x}</span><br>
  1056. ${dataList[1].EdbName}: <span style="font-weight: 600"> ${that.y}</span>
  1057. `:
  1058. `<strong>${ tmpData_date[that.x+'_'+that.y].length > 4 ? tmpData_date[that.x+'_'+that.y].slice(0,4).join()+'...' : tmpData_date[that.x+'_'+that.y].join() }</strong><br>
  1059. ${dataList[0].EdbNameEn}: <span style="font-weight: 600"> ${that.x}</span><br>
  1060. ${dataList[1].EdbNameEn}: <span style="font-weight: 600"> ${that.y}</span>
  1061. `
  1062. }
  1063. }
  1064. const { IsOrder,ChartColor } = dataList[0];
  1065. //y轴
  1066. let yAxis = {
  1067. title: {
  1068. text: language.value=='ch'?`${dataList[1].Unit}`:dataList[1].Unit?dataList[1].UnitEn:'',
  1069. // text: null,
  1070. align: 'high',
  1071. rotation: 0,
  1072. y: -5,
  1073. x:0,
  1074. textAlign: 'left',
  1075. reserveSpace: false,
  1076. style:{
  1077. ...chartTheme&&chartTheme.yAxisOptions.style
  1078. },
  1079. },
  1080. labels: {
  1081. formatter: function (ctx: any) {
  1082. return ctx.value;
  1083. },
  1084. align: 'center',
  1085. style:{
  1086. ...chartTheme&&chartTheme.yAxisOptions.style
  1087. },
  1088. },
  1089. opposite: false,
  1090. reversed: IsOrder,
  1091. min: Number(dataList[0].MinData),
  1092. max: Number(dataList[0].MaxData),
  1093. tickWidth: 1,
  1094. tickLength: 5,
  1095. lineWidth: 1,
  1096. lineColor: '#bfbfbf',
  1097. tickColor: '#bfbfbf',
  1098. offset: 0,
  1099. visible: true,
  1100. gridLineWidth: 0,
  1101. tickPosition: 'inside',
  1102. endOnTick: false,
  1103. startOnTick: false,
  1104. showLastLabel: true,
  1105. tickPixelInterval: 50,
  1106. plotBands: setAxisPlotAreas(1),
  1107. plotLines: setAxisPlotLines(1)
  1108. }
  1109. //数据列
  1110. let series: any = {
  1111. data: [],
  1112. type: 'scatter',
  1113. name: language.value == 'ch'?`${chartInfo.ChartName}${IsOrder ? '(逆序)' : ''}`:`${chartInfo.ChartNameEn}${IsOrder ? '(reserve)' : ''}`,
  1114. color: ChartColor,
  1115. chartType: 'linear',
  1116. marker: {
  1117. radius: (chartTheme&&chartTheme.lineOptions.radius)||5,
  1118. },
  1119. }
  1120. real_data.forEach(_ => {
  1121. series.data.push([_.x,_.y])
  1122. })
  1123. state.options = {
  1124. title: {
  1125. text:''
  1126. },
  1127. series: [ series ],
  1128. yAxis,
  1129. xAxis: {
  1130. ...scatterXAxis,
  1131. title: {
  1132. text: language.value=='ch'?`${dataList[0].Unit}`:dataList[0].Unit?dataList[0].UnitEn:'',
  1133. // text: null,
  1134. align: 'high',
  1135. rotation: 0,
  1136. x: 0,
  1137. offset: 20,
  1138. style: {
  1139. ...chartTheme&&chartTheme.xAxisOptions.style
  1140. },
  1141. },
  1142. labels: {
  1143. style: {
  1144. ...chartTheme&&chartTheme.xAxisOptions.style
  1145. }
  1146. },
  1147. plotBands: setAxisPlotAreas(3),
  1148. plotLines: setAxisPlotLines(3)
  1149. },
  1150. tooltip
  1151. }
  1152. console.log(state.options)
  1153. }
  1154. /* 获取图表详情后赋值柱状图数据 */
  1155. const initBarData = (data: { XEdbIdValue: number[]; YDataList: any; EdbInfoList: any; ChartInfo: any; }) => {
  1156. const { XEdbIdValue,YDataList,EdbInfoList,ChartInfo } = data;
  1157. state.barDateList = YDataList;
  1158. state.barXIdData = XEdbIdValue;
  1159. state.barEdbData = EdbInfoList;
  1160. state.chartLimit = {
  1161. min: Number(ChartInfo.LeftMin),
  1162. max: Number(ChartInfo.LeftMax),
  1163. }
  1164. setBarChart();
  1165. }
  1166. /* 奇怪柱状图 和以上逻辑无公用点 依赖数据为单独的数据
  1167. x轴为指标名称的柱形图 以日期作为series
  1168. */
  1169. const setBarChart = () => {
  1170. const {barDateList,barXIdData,chartLimit,barEdbData,chartInfo} = state;
  1171. let seriesData: { data: any; type: string; yAxis: number; name: any; color: any; chartType: string; }[] = [];
  1172. const data = _.cloneDeep(barDateList);
  1173. /* 主题样式*/
  1174. const chartTheme = chartInfo.ChartThemeStyle ? JSON.parse(chartInfo.ChartThemeStyle) : null;
  1175. let categories = language.value==='ch'
  1176. ? barXIdData.map((_:number) => barEdbData.find((edb: { EdbInfoId: number; }) => edb.EdbInfoId===_).EdbAliasName)
  1177. : barXIdData.map((_:number) => barEdbData.find((edb: { EdbInfoId: number; }) => edb.EdbInfoId===_).EdbNameEn)
  1178. //x轴
  1179. let xAxis = {
  1180. ...scatterXAxis,
  1181. categories,
  1182. tickWidth: 1,
  1183. labels: {
  1184. style: {
  1185. ...chartTheme&&chartTheme.xAxisOptions.style
  1186. }
  1187. }
  1188. }
  1189. const { max,min } = chartLimit;
  1190. console.log(max,min)
  1191. //y轴
  1192. let yAxis = {
  1193. ...basicYAxis,
  1194. title: {
  1195. text: language.value==='ch' ? chartInfo.Unit : chartInfo.UnitEn,
  1196. align: 'high',
  1197. rotation: 0,
  1198. y: -15,
  1199. x:0,
  1200. textAlign: 'left',
  1201. reserveSpace: false,
  1202. style:{
  1203. ...chartTheme&&chartTheme.yAxisOptions.style
  1204. },
  1205. },
  1206. labels: {
  1207. formatter: function (ctx:any) {
  1208. let val = ctx.value;
  1209. return val;
  1210. },
  1211. align: 'center',
  1212. style:{
  1213. ...chartTheme&&chartTheme.yAxisOptions.style
  1214. },
  1215. },
  1216. min: Number(min),
  1217. max: Number(max),
  1218. opposite: false,
  1219. tickWidth: 1,
  1220. plotBands: setAxisPlotAreas(1),
  1221. plotLines: setAxisPlotLines(1)
  1222. }
  1223. //数据列
  1224. data.forEach((item: { Value: number; Name: string; Date: string; Color: string; }) => {
  1225. let serie_item = {
  1226. data: item.Value,
  1227. type: 'column',
  1228. yAxis: 0,
  1229. name: language.value==='ch' ? (item.Name || item.Date) : item.Date,
  1230. color: item.Color,
  1231. chartType: 'linear'
  1232. };
  1233. seriesData.push(serie_item)
  1234. })
  1235. state.options = {
  1236. title: {
  1237. text:''
  1238. },
  1239. plotOptions: {
  1240. column:{
  1241. stacking: null,
  1242. },
  1243. },
  1244. series: seriesData,
  1245. yAxis: [ yAxis ],
  1246. xAxis
  1247. }
  1248. }
  1249. /* 商品价格曲线获取详情赋值 */
  1250. const initCommodityData = (data: { XDataList: any[]; YDataList: any; EdbInfoList: any; ChartInfo: any; DataResp?:any }) => {
  1251. const { XDataList,YDataList,EdbInfoList,ChartInfo,DataResp } = data;
  1252. state.commodityEdbList = EdbInfoList;
  1253. state.commodityChartData = ChartInfo.Source===5?DataResp.YDataList:YDataList;
  1254. state.commodityXData = ChartInfo.Source===5?DataResp.XDataList:XDataList;
  1255. if(ChartInfo.Source===5) {
  1256. state.chartInfo = {
  1257. ...state.chartInfo,
  1258. ProfitName: DataResp.ProfitName,
  1259. ProfitNameEn: DataResp.ProfitNameEn
  1260. }
  1261. }
  1262. state.chartLimit = {
  1263. min: Number(ChartInfo.LeftMin),
  1264. max: Number(ChartInfo.LeftMax)
  1265. }
  1266. setCommodityChart();
  1267. }
  1268. /* 商品价格曲线设置 绘图逻辑同奇怪柱形图*/
  1269. const setCommodityChart = () => {
  1270. const {chartLimit,commodityChartData,commodityXData,commodityEdbList,chartInfo} = state;
  1271. let seriesData:any[] = [];
  1272. const data = _.cloneDeep(commodityChartData);
  1273. /* 主题样式*/
  1274. const chartTheme = chartInfo.ChartThemeStyle ? JSON.parse(chartInfo.ChartThemeStyle) : null;
  1275. //x轴
  1276. let xAxis = {
  1277. ...scatterXAxis,
  1278. categories: commodityXData.map(_ => language.value === 'ch' ? _.Name:_.NameEn),
  1279. tickWidth: 1,
  1280. labels: {
  1281. style: {
  1282. ...chartTheme&&chartTheme.xAxisOptions.style
  1283. }
  1284. }
  1285. }
  1286. const { max,min } = chartLimit;
  1287. //y轴
  1288. let yAxis = {
  1289. ...basicYAxis,
  1290. title: {
  1291. text: language.value === 'ch' ? commodityEdbList[0].Unit : commodityEdbList[0].UnitEn,
  1292. align: 'high',
  1293. rotation: 0,
  1294. y: -15,
  1295. textAlign: 'left',
  1296. reserveSpace: false,
  1297. style:{
  1298. ...chartTheme&&chartTheme.yAxisOptions.style
  1299. },
  1300. },
  1301. labels: {
  1302. formatter: function (ctx:any) {
  1303. let val = ctx.value;
  1304. return val;
  1305. },
  1306. align: 'center',
  1307. style:{
  1308. ...chartTheme&&chartTheme.yAxisOptions.style
  1309. },
  1310. },
  1311. min: Number(min),
  1312. max: Number(max),
  1313. opposite: false,
  1314. tickWidth: 1,
  1315. }
  1316. //数据列
  1317. data.forEach((item: { Value: number[]; Name: string; Date: string; Color: string;NameEn: string,XEdbInfoIdList: number[],NoDataEdbList: number[] }) => {
  1318. //处理首或/尾全是无效数据的以null填充
  1319. let filterData = filterInvalidData(item)
  1320. let serie_item = {
  1321. data: filterData,
  1322. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  1323. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  1324. yAxis: 0,
  1325. name: language.value === 'ch' ? item.Name : item.NameEn,
  1326. color: item.Color,
  1327. chartType: 'linear',
  1328. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 3,
  1329. marker: {
  1330. enabled: false
  1331. }
  1332. };
  1333. seriesData.push(serie_item)
  1334. })
  1335. //tooltip
  1336. let tooltip = {
  1337. formatter: function() {
  1338. const ctx: any = this;
  1339. let str: string = '';
  1340. if(language.value === 'ch') {
  1341. ctx.points.forEach((item: { series:{name: string},y: number,color: string }) => {
  1342. let obj_item = data.find((_:any) => _.Name === item.series.name);
  1343. let index = commodityXData.findIndex(_ => _.Name === ctx.x);
  1344. //合约显示
  1345. let haveContract = obj_item.XEdbInfoIdList[index];
  1346. if(haveContract) {
  1347. // 利润曲线指标名
  1348. let edb_name = chartInfo.Source === 5
  1349. ? (index === 0 ? obj_item.NameList[index] : `${chartInfo.ProfitName}(${obj_item.NameList[index]})`)
  1350. : commodityEdbList.find(_ => _.EdbInfoId === obj_item.XEdbInfoIdList[index]).EdbName;
  1351. str+=`<b>${ edb_name }</b>`
  1352. if(!obj_item.NoDataEdbList.includes(obj_item.XEdbInfoIdList[index])) {
  1353. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: ${item.y}<br>`
  1354. }else {
  1355. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: 无<br>`
  1356. }
  1357. }
  1358. })
  1359. }else {
  1360. ctx.points.forEach((item: { series:{name: string},y: number,color: string }) => {
  1361. let obj_item = data.find((_: any) => _.NameEn === item.series.name);
  1362. let index = commodityXData.findIndex(_ => _.NameEn === ctx.x);
  1363. let haveContract = obj_item.XEdbInfoIdList[index];
  1364. if(haveContract) {
  1365. let edb_name = chartInfo.Source === 5
  1366. ? (index === 0 ? obj_item.NameList[index] : `${chartInfo.ProfitNameEn}(${obj_item.NameList[index]})`)
  1367. : commodityEdbList.find(_ => _.EdbInfoId === obj_item.XEdbInfoIdList[index]).EdbNameEn;
  1368. str+=`<b>${ edb_name }</b>`
  1369. if(!obj_item.NoDataEdbList.includes(obj_item.XEdbInfoIdList[index])) {
  1370. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: ${item.y}<br>`
  1371. }else {
  1372. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: 无<br>`
  1373. }
  1374. }
  1375. })
  1376. }
  1377. return str || '无合约'
  1378. },
  1379. shared: true
  1380. }
  1381. state.options = {
  1382. title: {
  1383. text:''
  1384. },
  1385. series: seriesData,
  1386. yAxis: [ yAxis ],
  1387. xAxis,
  1388. tooltip
  1389. }
  1390. };
  1391. /* 处理无效数据为null */
  1392. const filterInvalidData = (item:{ Value: number[]; Name: string; Date: string; Color: string;NameEn: string,XEdbInfoIdList: number[],NoDataEdbList: number[] })=> {
  1393. let validateArr = item.XEdbInfoIdList.filter(_ => _&&!item.NoDataEdbList.includes(_));
  1394. let first_index = item.XEdbInfoIdList.findIndex(_ => _ === validateArr[0]);
  1395. let last_index = item.XEdbInfoIdList.findIndex(_ => _ === validateArr[validateArr.length-1]);
  1396. console.log('first_index',first_index)
  1397. console.log('last_index',last_index)
  1398. let arr = item.Value.map((item,index:number) => {
  1399. if(index < first_index || index > last_index) {
  1400. return null
  1401. }else {
  1402. return item
  1403. }
  1404. })
  1405. return arr;
  1406. }
  1407. /* 设置options */
  1408. const setOptions = () => {
  1409. // ChartType: 1曲线图 2季节图 3面积图 4柱状图 5散点图 6组合图 季节图中公历和农历数据结构不同
  1410. const { chartInfo } = state;
  1411. const chartSetMap: any = {
  1412. 1: setDefaultLineOptions,
  1413. 2: setSeasonOptions,
  1414. 3: setStackOrCombinChart,
  1415. 4: setStackOrCombinChart,
  1416. 5: setScatterChartOptions,
  1417. 6: setStackOrCombinChart
  1418. };
  1419. chartSetMap[chartInfo.ChartType]()
  1420. };
  1421. /* 拼接动态的指标名称小标签 */
  1422. const concatDynamicTag = ({ IsAxis,IsOrder,EdbInfoType,LeadValue,LeadUnit }: IDataProps,lang:String='ch'): string => {
  1423. // IsAxis左轴1 右轴0 2右2轴
  1424. //IsOrder正序false 逆序true
  1425. //EdbInfoType是否是领先指标
  1426. const axisLabelMap:any = lang=='ch'?{
  1427. 0: '右轴',
  1428. 2: '右2轴'
  1429. }:{
  1430. 0: 'RHS',
  1431. 2: '2-RHS'
  1432. }
  1433. const orderLabelMap:any = lang=='ch'?{
  1434. 1: '逆序'
  1435. }:{
  1436. 1: 'REV'
  1437. }
  1438. const edbInfoMap:any = lang=='ch'?{
  1439. 0: '领先'
  1440. }:{
  1441. 0: 'Lead'
  1442. }
  1443. const leadUnitEnMap:any = {
  1444. '年': 'Y',
  1445. '季': 'Q',
  1446. '月': 'M',
  1447. '周': 'W',
  1448. '天': 'D',
  1449. }
  1450. //英文领先单位转换
  1451. const edbLeadUnit = lang=='ch' ? LeadUnit : leadUnitEnMap[LeadUnit];
  1452. let axis_tag = axisLabelMap[IsAxis] || '';
  1453. //逆序拼接
  1454. let order_tag = orderLabelMap[Number(IsOrder)] ? `${axis_tag ? ',': ''}${orderLabelMap[Number(IsOrder)]}` : ''
  1455. //领先拼接
  1456. let edb_tag = edbInfoMap[EdbInfoType] ? `${(axis_tag||order_tag) ? ',' : '' }${edbInfoMap[EdbInfoType]} ${LeadValue}${edbLeadUnit}` : '';
  1457. return (axis_tag || order_tag || edb_tag) ? `(${axis_tag}${order_tag}${edb_tag})` : ''
  1458. }
  1459. /* 预测配置 分区 */
  1460. const getPredictParams = ({LatestDate,MoveLatestDate,PredictChartColor,ChartStyle}: IDataProps,chartStyle='') => {
  1461. return {
  1462. zoneAxis: 'x',
  1463. zones: [{
  1464. value: new Date(MoveLatestDate||LatestDate).getTime()+1
  1465. }, {
  1466. dashStyle: 'ShortDot',
  1467. color:(ChartStyle==='column' || chartStyle==='column') ? 'transparent' : PredictChartColor
  1468. }]
  1469. }
  1470. }
  1471. /* 季节图预测数据 年份=分割点年份做分割 年份>分割点年份全为预测 */
  1472. const getSeasonPredictParams = (timestamp: number) => {
  1473. return timestamp
  1474. ? {
  1475. zoneAxis: 'x',
  1476. zones: [{
  1477. value: new Date(timestamp).getTime()+1
  1478. }, {
  1479. dashStyle: 'ShortDot',
  1480. }]
  1481. }
  1482. : {}
  1483. }
  1484. /* 处理轴的标识线结构 在指定轴位置上拼接标识线
  1485. 0:右轴 1:左轴 2:右2轴 x轴固定3
  1486. axisType表示x轴类型 处理时间轴的值 datetime/null
  1487. */
  1488. const setAxisPlotLines = (axis:number,axisType:any=null) => {
  1489. const { MarkersLines,ChartType } = state.chartInfo;
  1490. if(!MarkersLines) return []
  1491. let markerLines = JSON.parse(MarkersLines);
  1492. let arr = markerLines.filter(_ => _.isShow&&_.axis===axis)
  1493. let plotLines = arr.map(_ => {
  1494. //是否是x时间轴
  1495. let isXDateAxis = axis===3&&axisType==='datetime';
  1496. let markerValue;
  1497. if(isXDateAxis) {
  1498. //季节图x轴额外拼个年份
  1499. let nowYear = new Date().getFullYear();
  1500. markerValue = ChartType===2
  1501. ? new Date(`${nowYear}-${_.value}`).getTime()
  1502. : new Date(_.value).getTime()
  1503. }else {
  1504. markerValue = Number(_.value)
  1505. }
  1506. return {
  1507. value: markerValue,
  1508. dashStyle: _.dashStyle,
  1509. width: Number(_.lineWidth),
  1510. color: _.color,
  1511. label: {
  1512. text: _.text||'',
  1513. verticalAlign: _.textPosition,
  1514. style: {
  1515. color: _.textColor,
  1516. fontSize: _.textFontSize
  1517. }
  1518. }
  1519. }
  1520. })
  1521. return plotLines
  1522. }
  1523. /* 处理标识区拼接 axisType表示x轴类型处理时间轴的值 datetime/null */
  1524. const setAxisPlotAreas = (axis:number,axisType:any=null) => {
  1525. const { MarkersAreas,ChartType } = state.chartInfo;
  1526. if(!MarkersAreas) return []
  1527. let markerAreas = JSON.parse(MarkersAreas);
  1528. let arr = markerAreas.filter(_ => _.isShow&&_.axis===axis)
  1529. let plotBands = arr.map(_ => {
  1530. //是否是x时间轴
  1531. let isXDateAxis = axis===3&&axisType==='datetime';
  1532. let fromMarkerValue,toMarkerValue;
  1533. if(isXDateAxis) {
  1534. //季节图x轴额外拼个年份
  1535. let nowYear = new Date().getFullYear();
  1536. fromMarkerValue = ChartType===2
  1537. ? new Date(`${nowYear}-${_.fromValue}`).getTime()
  1538. : new Date(_.fromValue).getTime()
  1539. toMarkerValue = ChartType===2
  1540. ? new Date(`${nowYear}-${_.toValue}`).getTime()
  1541. : new Date(_.toValue).getTime()
  1542. }else {
  1543. fromMarkerValue = Number(_.fromValue);
  1544. toMarkerValue = Number(_.toValue);
  1545. }
  1546. return {
  1547. from: fromMarkerValue,
  1548. to: toMarkerValue,
  1549. color: _.color,
  1550. label: {
  1551. text: _.text||'',
  1552. verticalAlign: _.textPosition,
  1553. style: {
  1554. color: _.textColor,
  1555. fontSize: _.textFontSize
  1556. }
  1557. }
  1558. }
  1559. })
  1560. return plotBands
  1561. }
  1562. /* 复制标题 */
  1563. const copyText = () => {
  1564. const { chartInfo } = state;
  1565. let input = document.createElement("input");
  1566. input.value = chartInfo.ChartName;
  1567. document.body.appendChild(input)
  1568. input.select();
  1569. document.execCommand('copy');
  1570. document.body.removeChild(input);
  1571. ElMessage.success('复制标题成功')
  1572. }
  1573. /* 分享链接 */
  1574. const copyUrl = () => {
  1575. let input = document.createElement("input");
  1576. input.value = location.href;
  1577. document.body.appendChild(input)
  1578. input.select();
  1579. document.execCommand('copy');
  1580. document.body.removeChild(input);
  1581. ElMessage.success('复制链接成功')
  1582. }
  1583. const refreshChart = _.debounce(async () => {
  1584. loading.value = true;
  1585. let res: any=null
  1586. if([1,6,7,8,9].includes(state.chartInfo.Source)){
  1587. res=await ChartApi.refreshChart({UniqueCode: state.chartInfo.UniqueCode})
  1588. }else if([2,5].includes(state.chartInfo.Source)){
  1589. res=await ChartApi.refreshCommordityChart({UniqueCode: state.chartInfo.UniqueCode});
  1590. }else if([3,4].includes(state.chartInfo.Source)){
  1591. res=await ChartApi.refreshRelevanceChart({UniqueCode: state.chartInfo.UniqueCode});
  1592. }
  1593. loading.value = false;
  1594. res.Ret === 200 && getChartInfo('refresh');
  1595. },400)
  1596. return {
  1597. ...toRefs(state),
  1598. loading,
  1599. haveData,
  1600. getChartInfo,
  1601. openNew,
  1602. copyUrl,
  1603. isShare,
  1604. language,
  1605. refreshChart,
  1606. copyText
  1607. };
  1608. },
  1609. });
  1610. </script>
  1611. <style scoped lang="less">
  1612. @import './index.less';
  1613. </style>