chartPublic.js 103 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030
  1. import {
  2. defaultOpts,
  3. seasonOptions,
  4. yearSelector,
  5. } from '@/utils/defaultOptions';
  6. import Highcharts from 'highcharts';
  7. import { dataBaseInterface } from '@/api/api';
  8. import * as preDictEdbInterface from '@/api/modules/predictEdbApi.js';
  9. import futuresInterface from '@/api/modules/futuresBaseApi';
  10. import chartRelevanceApi from '@/api/modules/chartRelevanceApi';
  11. import { fittingEquationInterface,statisticFeatureInterface,crossVarietyInterface } from '@/api/modules/chartRelevanceApi';
  12. /* 散点x轴 */
  13. const scatterXAxis = {
  14. tickPosition: 'inside',
  15. lineColor: '#bfbfbf',
  16. tickColor: '#bfbfbf',
  17. tickLength:5,
  18. ordinal: false,
  19. type: 'linear'
  20. }
  21. /* 曲线基础y轴静态配置 */
  22. const basicYAxis = {
  23. tickLength: 5,
  24. lineWidth: 1,
  25. lineColor: '#bfbfbf',
  26. tickColor: '#bfbfbf',
  27. // offset: 0,
  28. visible: true,
  29. gridLineWidth: 0,
  30. tickPosition: 'inside',
  31. endOnTick: false,
  32. startOnTick: false,
  33. showLastLabel: true,
  34. tickPixelInterval: 50,
  35. }
  36. /**备注一下 越多越乱
  37. * @params
  38. * Source 1 ; chartType 1曲线 2季节 3面积 4堆积柱 5散点 6组合 7柱形 10截面散点 11雷达图
  39. * 2 商品价格
  40. * 3 相关性
  41. * 4 滚动相关性
  42. * 5 商品利润
  43. * 6 拟合方程
  44. * 7 统计特征/标准差
  45. * 8 统计特征/百分位
  46. * 9 统计特征/频率
  47. * 10 跨品种分析
  48. */
  49. export const chartSetMixin = {
  50. data() {
  51. return {
  52. /**
  53. * 默认区间15年至今 值等于 5、6、20 是自定义时间段
  54. * 20代表最近几年 季节性图默认 最近5年
  55. * 6代表至今 只需要选择开始日期
  56. * 5代表范围 需要选择开始日期和结束日期
  57. */
  58. year_select: yearSelector[0].value,
  59. sameOptionType:[1,3,4,5,6],// 头部一样 配置一样的图表类型 曲线 散点 柱状 面积 组合
  60. // yearSelector, //年份按钮组
  61. select_date: '', //自定义时间段
  62. count_year: 0, //最近年数
  63. isDateDia: false, // 时间段弹窗
  64. earliestDate:'', // 最早的日期 - 起始日期
  65. latestDate:'', //最晚日期 - 最新日期
  66. dateForm: {},
  67. dateTip: /* '请选择时间段' */ this.$t('Chart.choose_time'),
  68. predefineColors: defaultOpts.colors.slice(0, 2), //定义颜色蓝,红 默认颜色
  69. fre_options: ['年', '季', '月', '周', '天'], //领先指标频度配置
  70. //领先频度对应英文
  71. leadUnitEnMap: {
  72. '年': 'Y',
  73. '季': 'Q',
  74. '月': 'M',
  75. '周': 'W',
  76. '天': 'D',
  77. },
  78. tableData: [],//表格指标数据
  79. options: {}, //配置options
  80. leftIndex: -1, //左侧上下限对应下标
  81. rightIndex: -1, //右侧上下限对应下标
  82. rightTwoIndex: -1,//右2上下限对应下标
  83. left_extreme: [],
  84. right_extreme: [],
  85. chartItemStyleArr:[
  86. { label: '曲线图', key: 1 ,value: 'spline'},
  87. { label: '折线图', key: 2 ,value: 'line'},
  88. // { label: '堆积面积图', key: 3 ,value: 'areaspline'},
  89. { label: '堆积柱状图', key: 4 ,value: 'column'},
  90. ],//组合图可选样式
  91. isSetExtremeValue: false,//控制添加散点图时是否重置上下限
  92. isShowSaveOther: false,//图表另存为弹窗
  93. // -------------切换图表中英文
  94. // currentLang:'zh', // ch(中文) en(英文)
  95. setEnName:false,
  96. // 传入的formItem所需内容
  97. formItemArray:[],
  98. /* 奇怪柱形图 */
  99. barDateList: [],//柱形图的绘图数据
  100. barXData: [],//柱形图的x轴
  101. barXIdList: [],//x轴id数组 用于切换英文遍历用
  102. barEdbData: [],//柱形图的表格数据 只用于取值
  103. chartLimit: {
  104. min:'', //左轴上下限
  105. max:'',
  106. rightMin:'',//右轴上下限
  107. rightMax:'',
  108. rightTwoMin:'',//右二轴上下限
  109. rightTwoMax:'',
  110. },
  111. /* 商品价格曲线 本来逻辑分开写 但需求又得加入myeta和eta图库共存 还是写一块算了 */
  112. commodityChartData: [],
  113. commodityXData: [],
  114. commodityEdbList: [],
  115. /* 时间截面散点图 */
  116. sectionScatterData: {},
  117. /* 相关性图表 */
  118. relevanceChartData:null,
  119. relevanceUnitEnMap:{
  120. '年': 'Year',
  121. '季': 'Season',
  122. '月': 'Month',
  123. '周': 'Week',
  124. '天': 'Day',
  125. },
  126. /* 统计频率图 */
  127. statisticFrequencyData: {},
  128. // 季节性图和额外配置 这里为了预览
  129. SeasonExtraConfig:{
  130. ChartLegend:[], // 图例名称数组
  131. XStartDate:"01-01", // 横坐标显示范围 - 开始
  132. XEndDate:"12-31", // 横坐标显示范围-结束
  133. JumpYear:0, //是否跨年
  134. },
  135. /* 跨品种分析图 */
  136. crossVarietyChartData: {},
  137. /* 雷达图 */
  138. radarChartData: {},
  139. /* 修改对应版本信息弹窗 替换原有设置英文*/
  140. isLangInfoDia: false
  141. }
  142. },
  143. computed:{
  144. tableColums(){
  145. return [
  146. {
  147. label: this.$t('Edb.Detail.e_name'),
  148. key: 'EdbName',
  149. enKey:'EdbNameEn',
  150. inputTip:'点击输入英文指标名称',
  151. minwidthsty: '150px',
  152. },
  153. {
  154. label: this.$t('Edb.Detail.e_id'),
  155. key: 'EdbCode',
  156. widthsty: '120px',
  157. },
  158. {
  159. label: this.$t('Edb.Detail.e_fre'),
  160. key: 'Frequency',
  161. enKey:'FrequencyEn',
  162. minwidthsty: '60px',
  163. },
  164. {
  165. label: this.$t('Edb.Detail.e_unit'),
  166. key: 'Unit',
  167. enKey:'UnitEn',
  168. inputTip:'英文单位',
  169. minwidthsty: '50px',
  170. },
  171. {
  172. label: this.$t('Edb.Detail.e_start_time'),
  173. key: 'StartDate',
  174. minwidthsty: '100px',
  175. },
  176. {
  177. label: this.$t('Edb.Detail.e_latest_date'),
  178. key: 'LatestDate',
  179. minwidthsty: '90px',
  180. },
  181. {
  182. label: this.$t('Edb.Detail.e_latest_value'),
  183. key: 'LatestValue',
  184. minwidthsty: '90px',
  185. },
  186. {
  187. label: this.$t('Edb.Detail.e_recent_time'),
  188. key: 'ModifyTime',
  189. minwidthsty: '100px',
  190. },
  191. {
  192. label: this.$t('Edb.Detail.source'),
  193. key: 'SourceName',
  194. },
  195. ]
  196. },
  197. yearSelector() {
  198. return yearSelector
  199. },
  200. currentLang() {
  201. return this.$store.state.lang
  202. }
  203. },
  204. watch: {
  205. /* 奇怪柱状图数据 */
  206. barDateList: {
  207. handler(newval) {
  208. console.log('bar')
  209. newval.length && this.barXData.length && this.setBarChart();
  210. },
  211. deep: true
  212. },
  213. sectionScatterData: {
  214. handler(newval) {
  215. newval.DataList && this.setSectionScatterChart();
  216. },
  217. deep: true
  218. },
  219. currentLang() {
  220. this.changeLanguage()
  221. }
  222. },
  223. methods: {
  224. /* --------------------------------------------切换中英文 */
  225. async changeLanguage(){
  226. this.search_txt = '';
  227. if(!this.$refs.chartRef) return
  228. //不同图表来源 1eta图 2商品价格图 3相关性图表
  229. const sourceMap = {
  230. 1: this.changeOptions,
  231. 2: this.changeCommodityLang,
  232. 3: this.changeRelevanceLang,
  233. 4: this.changeRelevanceLang,
  234. 5: this.changeCommodityLang
  235. }
  236. sourceMap[this.chartInfo.Source]&&sourceMap[this.chartInfo.Source]()
  237. },
  238. // // 打开设置英文信息弹窗
  239. // async openEnNameDia(){
  240. // this.formItemArray={}
  241. // this.formItemArray.chartInfo=[]
  242. // this.formItemArray.chartsList=[]
  243. // this.formItemArray.chartInfo.push({
  244. // label:/* '图表名称' */this.$t('Chart.Detail.chart_name'),
  245. // value:this.chartInfo.ChartName,
  246. // key:'ChartName',
  247. // id:this.chartInfo.ChartInfoId,
  248. // source: this.chartInfo.Source,
  249. // notEdit:true
  250. // },
  251. // {
  252. // label:/* '英文图表名称' */this.$t('Chart.Detail.chart_en_name'),
  253. // value:this.chartInfo.ChartNameEn,
  254. // key:'ChartNameEn',
  255. // id:this.chartInfo.ChartInfoId,
  256. // placeholder:/* '请输入英文图表名称' */this.$t('Chart.InputHolderAll.input_en_name')
  257. // })
  258. // if([1,2,5].includes(this.chartInfo.Source)){ //需要设置指标的
  259. // this.tableData.map(item =>{
  260. // if(item.Unit){
  261. // this.formItemArray.chartsList.push([
  262. // {
  263. // label:/* '指标名称' */this.$t('Edb.Detail.e_name'),
  264. // value:item.EdbName,
  265. // key:'EdbName',
  266. // id:item.EdbInfoId,
  267. // notEdit:true
  268. // },
  269. // {
  270. // label:/* '单位' */this.$t('Edb.Detail.e_unit'),
  271. // value:item.Unit,
  272. // key:'Unit',
  273. // id:item.EdbInfoId,
  274. // notEdit:true
  275. // },
  276. // {
  277. // label:/* '英文指标名称' */this.$t('Edb.Detail.e_en_name'),
  278. // value:item.EdbNameEn,
  279. // key:'EdbNameEn',
  280. // id:item.EdbInfoId,
  281. // placeholder:/* '请输入英文指标名称' */ this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_en_name')})
  282. // },
  283. // {
  284. // label:/* '英文单位' */this.$t('Edb.Detail.e_en_unit'),
  285. // value:item.UnitEn,
  286. // key:'UnitEn',
  287. // id:item.EdbInfoId,
  288. // placeholder:/* '请输入英文单位' */this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_en_unit')})
  289. // }
  290. // ])
  291. // }else{
  292. // this.formItemArray.chartsList.push([
  293. // {
  294. // label:/* '指标名称' */this.$t('Edb.Detail.e_name'),
  295. // value:item.EdbName,
  296. // key:'EdbName',
  297. // id:item.EdbInfoId,
  298. // notEdit:true
  299. // },
  300. // {
  301. // label:/* '英文指标名称' */this.$t('Edb.Detail.e_en_name'),
  302. // value:item.EdbNameEn,
  303. // key:'EdbNameEn',
  304. // id:item.EdbInfoId,
  305. // placeholder:this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_en_name')})
  306. // }
  307. // ])
  308. // }
  309. // })
  310. // }
  311. // //价格曲线
  312. // if(this.chartInfo.Source===2) {
  313. // this.formItemArray.chartInfo.push({
  314. // label:/* '期货名称' */this.$t('Chart.Detail.good_name'),
  315. // value:this.commodityEdbList[1].EdbName,
  316. // key:'FutureGoodName',
  317. // id:this.chartInfo.ChartInfoId,
  318. // notEdit:true
  319. // },
  320. // {
  321. // label:/* '英文期货名称' */this.$t('Chart.Detail.good_en_name'),
  322. // value:this.commodityEdbList[1].EdbNameEn,
  323. // key:'FutureGoodNameEn',
  324. // id:this.chartInfo.ChartInfoId,
  325. // placeholder:/* '请输入英文期货名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.good_en_name')})
  326. // })
  327. // }
  328. // //利润曲线
  329. // else if(this.chartInfo.Source===5) {
  330. // this.formItemArray.chartInfo.push({
  331. // label:/* '盘面利润名称' */this.$t('Chart.Detail.profit_name'),
  332. // value:this.chartInfo.ProfitName,
  333. // key:'ProfitName',
  334. // id:this.chartInfo.ChartInfoId,
  335. // notEdit:true
  336. // },
  337. // {
  338. // label:/* '英文盘面利润名称' */this.$t('Chart.Detail.profit_en_name'),
  339. // value:this.chartInfo.ProfitNameEn,
  340. // key:'ProfitNameEn',
  341. // id:this.chartInfo.ChartInfoId,
  342. // placeholder:/* '请输入英文盘面利润名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.profit_en_name')})
  343. // })
  344. // }
  345. // //跨品种分析
  346. // else if(this.chartInfo.Source===10) {
  347. // let { Data } = await crossVarietyInterface.chartLangOption({ChartInfoId: this.chartInfo.ChartInfoId})
  348. // const { TagList,VarietyList } = Data;
  349. // this.formItemArray.chartInfo.push({
  350. // label:/* 'X轴名称' */this.$t('Chart.Detail.x_name'),
  351. // value:this.crossVarietyChartData.XName,
  352. // key:'XName',
  353. // id:TagList[0].ChartTagId,
  354. // notEdit:true
  355. // },
  356. // {
  357. // label:/* '英文X轴名称' */this.$t('Chart.Detail.x_en_name'),
  358. // value:this.crossVarietyChartData.XNameEn,
  359. // key:'XNameEn',
  360. // id:TagList[0].ChartTagId,
  361. // placeholder:/* '请输入英文X轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.x_en_name')})
  362. // },{
  363. // label:/* 'Y轴名称' */this.$t('Chart.Detail.y_name'),
  364. // value:this.crossVarietyChartData.YName,
  365. // key:'YName',
  366. // id:TagList[1].ChartTagId,
  367. // notEdit:true
  368. // },
  369. // {
  370. // label:/* '英文Y轴名称' */this.$t('Chart.Detail.y_en_name'),
  371. // value:this.crossVarietyChartData.YNameEn,
  372. // key:'YNameEn',
  373. // id:TagList[1].ChartTagId,
  374. // placeholder:/* '请输入英文Y轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.y_en_name')})
  375. // })
  376. // VarietyList.forEach(item => {
  377. // this.formItemArray.chartsList.push([
  378. // {
  379. // label:/* '品种名称' */this.$t('Chart.Detail.variety_name'),
  380. // value:item.ChartVarietyName,
  381. // key:'ChartVarietyName',
  382. // id:item.ChartVarietyId,
  383. // notEdit:true
  384. // },
  385. // {
  386. // label:/* '英文品种名称' */this.$t('Chart.Detail.variety_en_name'),
  387. // value:item.ChartVarietyNameEn,
  388. // key:'ChartVarietyNameEn',
  389. // id:item.ChartVarietyId,
  390. // placeholder:/* '请输入英文品种名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.variety_en_name')})
  391. // }
  392. // ])
  393. // })
  394. // }
  395. // this.setEnName = true
  396. // },
  397. // // 更新英文信息
  398. // async updateEnName(enNameData){
  399. // // console.log(enNameData)
  400. // let res=null
  401. // if([2,5].includes(this.chartInfo.Source)){//商品价格
  402. // res=await futuresInterface.editChartEn({
  403. // ChartInfoId: enNameData.ChartInfoId,
  404. // ChartNameEn: enNameData.ChartNameEn,
  405. // UnitEn: enNameData.ChartEdbInfoList[0].UnitEn || '',
  406. // EdbNameEn: enNameData.ChartEdbInfoList[0].EdbNameEn || '',
  407. // FutureGoodNameEn: enNameData.FutureGoodNameEn || '',
  408. // ProfitNameEn: enNameData.ProfitNameEn || ''
  409. // })
  410. // }else if(this.chartInfo.Source===3){//相关性
  411. // res=await chartRelevanceApi.editChartEn({
  412. // ChartInfoId: enNameData.ChartInfoId,
  413. // ChartNameEn: enNameData.ChartNameEn
  414. // })
  415. // }else if(this.chartInfo.Source===6){//拟合方程
  416. // res=await fittingEquationInterface.editChartEn({
  417. // ChartInfoId: enNameData.ChartInfoId,
  418. // ChartNameEn: enNameData.ChartNameEn
  419. // })
  420. // }else if([7,8,9].includes(this.chartInfo.Source)){//统计特征
  421. // res=await statisticFeatureInterface.editChartEn({
  422. // ChartInfoId: enNameData.ChartInfoId,
  423. // ChartNameEn: enNameData.ChartNameEn
  424. // })
  425. // }else if(this.chartInfo.Source === 10) {//跨品种分析
  426. // res=await crossVarietyInterface.editChartEn(enNameData)
  427. // }else{
  428. // res = await dataBaseInterface.chartInfoEditEn(enNameData)
  429. // }
  430. // if(res.Ret !==200) return
  431. // this.$message.success(this.$t('MsgPrompt.edit_msg'))
  432. // this.getChartInfo()
  433. // this.setEnName = false
  434. // },
  435. /* 打开编辑信息弹窗 */
  436. async openLangInfoDia() {
  437. this.formItemArray={}
  438. this.formItemArray.chartInfo=[]
  439. this.formItemArray.chartsList=[]
  440. this.formItemArray.chartInfo.push({
  441. label:/* '图表名称' */this.$t('Chart.Detail.chart_name'),
  442. value: this.currentLang==='en'?this.chartInfo.ChartNameEn:this.chartInfo.ChartName,
  443. key:'ChartName',
  444. id:this.chartInfo.ChartInfoId,
  445. source: this.chartInfo.Source,
  446. placeholder:/* '请输入图表名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.chart_name')})
  447. })
  448. if([1,2,5].includes(this.chartInfo.Source)){ //需要设置指标的
  449. this.tableData.map(item =>{
  450. this.formItemArray.chartsList.push([
  451. {
  452. label:/* '指标名称' */this.$t('Edb.Detail.e_name'),
  453. value:this.currentLang==='en'?item.EdbNameEn:item.EdbName,
  454. key:'EdbName',
  455. id:item.EdbInfoId,
  456. placeholder:/* '请输入指标名称' */ this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_name')})
  457. },
  458. {
  459. label:/* '单位' */this.$t('Edb.Detail.e_unit'),
  460. value:this.currentLang==='en'?item.UnitEn:item.Unit,
  461. key:'Unit',
  462. id:item.EdbInfoId,
  463. placeholder:/* '请输入单位' */ this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_unit')})
  464. }
  465. ])
  466. })
  467. }
  468. //价格曲线
  469. if(this.chartInfo.Source===2) {
  470. this.formItemArray.chartInfo.push({
  471. label:/* '期货名称' */this.$t('Chart.Detail.good_name'),
  472. value:this.currentLang==='en'?this.commodityEdbList[1].EdbNameEn:this.commodityEdbList[1].EdbName,
  473. key:'FutureGoodName',
  474. id:this.chartInfo.ChartInfoId,
  475. placeholder:/* '请输入期货名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.good_name')})
  476. })
  477. }
  478. //利润曲线
  479. else if(this.chartInfo.Source===5) {
  480. this.formItemArray.chartInfo.push({
  481. label:/* '盘面利润名称' */this.$t('Chart.Detail.profit_name'),
  482. value: this.currentLang==='en'?this.chartInfo.ProfitNameEn:this.chartInfo.ProfitName,
  483. key:'ProfitName',
  484. id:this.chartInfo.ChartInfoId,
  485. placeholder:/* '请输入盘面利润名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.profit_en_name')})
  486. })
  487. }
  488. //跨品种分析
  489. else if(this.chartInfo.Source===10) {
  490. let { Data } = await crossVarietyInterface.chartLangOption({ChartInfoId: this.chartInfo.ChartInfoId})
  491. const { TagList,VarietyList } = Data;
  492. this.formItemArray.chartInfo.push({
  493. label:/* 'X轴名称' */this.$t('Chart.Detail.x_name'),
  494. value:this.currentLang==='en'?this.crossVarietyChartData.XNameEn:this.crossVarietyChartData.XName,
  495. key:'XName',
  496. id:TagList[0].ChartTagId,
  497. placeholder:/* '请输入X轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.x_name')})
  498. },
  499. {
  500. label:/* 'Y轴名称' */this.$t('Chart.Detail.y_name'),
  501. value:this.currentLang==='en'?this.crossVarietyChartData.YNameEn:this.crossVarietyChartData.YName,
  502. key:'YName',
  503. id:TagList[1].ChartTagId,
  504. placeholder:/* '请输入Y轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.y_name')})
  505. })
  506. VarietyList.forEach(item => {
  507. this.formItemArray.chartsList.push(
  508. {
  509. label:/* '品种名称' */this.$t('Chart.Detail.variety_name'),
  510. value:this.currentLang==='en'?item.ChartVarietyNameEn:item.ChartVarietyName,
  511. key:'ChartVarietyName',
  512. id:item.ChartVarietyId,
  513. placeholder:/* '请输入品种名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.variety_name')})
  514. }
  515. )
  516. })
  517. }
  518. this.isLangInfoDia = true
  519. },
  520. /* 更新版本图表信息 */
  521. async updateLang(paramsData) {
  522. let res=null
  523. if([2,5].includes(this.chartInfo.Source)){//商品价格
  524. res=await futuresInterface.setChartLangInfo({
  525. ChartInfoId: paramsData.ChartInfoId,
  526. ChartName: paramsData.ChartName,
  527. Unit: paramsData.ChartEdbInfoList[0].Unit || '',
  528. EdbName: paramsData.ChartEdbInfoList[0].EdbName || '',
  529. FutureGoodName: paramsData.FutureGoodName || '',
  530. ProfitName: paramsData.ProfitName || ''
  531. })
  532. }else if(this.chartInfo.Source===3){//相关性
  533. res=await chartRelevanceApi.setChartLangInfo({
  534. ChartInfoId: paramsData.ChartInfoId,
  535. ChartName: paramsData.ChartName
  536. })
  537. }else if(this.chartInfo.Source===6){//拟合方程
  538. res=await fittingEquationInterface.setChartLangInfo({
  539. ChartInfoId: paramsData.ChartInfoId,
  540. ChartNameEn: paramsData.ChartName
  541. })
  542. }else if([7,8,9].includes(this.chartInfo.Source)){//统计特征
  543. res=await statisticFeatureInterface.setChartLangInfo({
  544. ChartInfoId: paramsData.ChartInfoId,
  545. ChartName: paramsData.ChartName
  546. })
  547. }else if(this.chartInfo.Source === 10) {//跨品种分析
  548. res=await crossVarietyInterface.setChartLangInfo(paramsData)
  549. }else{
  550. res = await dataBaseInterface.setChartLangInfo(paramsData)
  551. }
  552. if(res.Ret !==200) return
  553. this.$message.success(this.$t('MsgPrompt.edit_msg'))
  554. this.getChartInfo()
  555. this.isLangInfoDia = false
  556. },
  557. /* 切换柱形图中英文 */
  558. changeBarOptions() {
  559. //x轴
  560. this.options.xAxis.categories = this.barXIdList.map(_ => this.currentLang === 'zh'
  561. ? this.barEdbData.find(edb => edb.EdbInfoId===_).EdbAliasName
  562. : this.barEdbData.find(edb => edb.EdbInfoId===_).EdbNameEn)
  563. // 单位
  564. this.options.yAxis.forEach(item => {
  565. item.title.text = this.currentLang === 'zh' ? this.chartInfo.Unit : this.chartInfo.Unit;
  566. });
  567. },
  568. // 切换中英文时,更改图表配置
  569. changeOptions(){
  570. // console.log(this.tableData,'tableData');
  571. // console.log(this.currentLang,this.chartInfo.ChartType);
  572. console.log(this.options,'options');
  573. // 散点图
  574. if(this.chartInfo.ChartType == 5){
  575. this.options.yAxis.title.text = this.currentLang == 'zh' ? this.options.yAxis.title.textCh : this.options.yAxis.title.textEn;
  576. this.options.xAxis.title.text = this.currentLang == 'zh' ? this.options.xAxis.title.textCh : this.options.xAxis.title.textEn
  577. this.options.series.forEach(item => {
  578. item.name = this.currentLang == 'zh' ? item.nameCh : item.nameEn;
  579. });
  580. this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn;
  581. }else{
  582. // 单位
  583. this.options.yAxis.forEach(item => {
  584. item.title.text = this.currentLang == 'zh' ? item.title.textCh : item.title.textEn;
  585. });
  586. // 图例 名称
  587. if(this.chartInfo.ChartType != 2){
  588. // 季节图 不更改图例名称
  589. this.options.series.forEach(item => {
  590. item.name = this.currentLang == 'zh'
  591. ? item.nameCh
  592. : (item.nameEn||item.nameCh)
  593. });
  594. }
  595. //截面散点 x轴标题
  596. if(this.chartInfo.ChartType === 10){
  597. this.options.xAxis.title.text = this.currentLang == 'zh' ? this.options.xAxis.title.textCh : this.options.xAxis.title.textEn;
  598. this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn;
  599. this.options.series.forEach(item => {
  600. if(!item.linkedTo) {
  601. item.data.forEach(point => {
  602. point.dataLabels.format = this.currentLang == 'zh' ? point.dataLabels.formatCh : (point.dataLabels.formatEn||point.dataLabels.formatCh);
  603. point.dataLabels.color = (this.currentLang==='en' && !point.dataLabels.formatEn) ? '#999' : '#333'
  604. })
  605. }
  606. });
  607. }
  608. //柱形图额外设置x轴中英文
  609. this.chartInfo.ChartType ===7 && this.changeBarOptions();
  610. }
  611. },
  612. /* 切换商品价格图中英文 */
  613. changeCommodityLang() {
  614. console.log(this.options)
  615. this.options.yAxis.forEach(item => {
  616. item.title.text = this.currentLang == 'zh' ? item.title.textCh : item.title.textEn
  617. });
  618. //图例
  619. this.options.series.forEach(item => {
  620. item.name = this.currentLang == 'zh' ? item.nameCh : item.nameEn
  621. });
  622. //tooltip
  623. this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn
  624. //x轴
  625. this.options.xAxis.categories = this.commodityXData.map(_ => this.currentLang == 'zh' ? _.Name:_.NameEn);
  626. },
  627. /* 切换相关性图中英文 */
  628. changeRelevanceLang(){
  629. this.options.yAxis.forEach(item => {
  630. item.title.text = this.currentLang == 'zh' ? item.title.textCh : item.title.textEn
  631. });
  632. //图例
  633. this.options.series.forEach(item => {
  634. item.name = this.currentLang == 'zh' ? item.nameCh : item.nameEn
  635. });
  636. //tooltip
  637. this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn
  638. this.options.xAxis.title.text=this.currentLang == 'zh' ? this.options.xAxis.title.textCh : this.options.xAxis.title.textEn
  639. },
  640. /* ================================================================ */
  641. /* 图表的配置项*/
  642. setChartOptionHandle(newval) {
  643. /* 显示类型 数据结构单独处理 */
  644. const chartSetMap = {
  645. 1: this.setDefaultChart,
  646. 2: this.setSeasonChart,
  647. 3: this.setStackOrCombinChart,
  648. 4: this.setStackOrCombinChart,
  649. 5: this.setScatterChart,
  650. 6: this.setStackOrCombinChart,
  651. };
  652. chartSetMap[this.chartInfo.ChartType]&&chartSetMap[this.chartInfo.ChartType](newval)
  653. //myeta内或者数据指标库内
  654. if(this.$route.path === '/mychart'||this.$route.path==='/database') this.showData = true;
  655. },
  656. /* 曲线图设置 */
  657. setDefaultChart(newval) {
  658. /* 上下限显示和值的设置 */
  659. if([4,6,7,8].includes(this.chartInfo.Source)) {
  660. this.leftIndex = -1;
  661. this.rightIndex = -1;
  662. this.rightTwoIndex = -1;
  663. this.chartLimit = {
  664. min:'', //左轴上下限
  665. max:'',
  666. rightMin:'',//右轴上下限
  667. rightMax:'',
  668. rightTwoMin:'',//右二轴上下限
  669. rightTwoMax:'',
  670. };
  671. }else {
  672. this.leftIndex = newval.findIndex((item) => item.IsAxis===1);
  673. this.rightIndex = newval.findIndex((item) => !item.IsAxis);
  674. this.rightTwoIndex = newval.findIndex((item) => item.IsAxis ===2);
  675. }
  676. /* 主题样式*/
  677. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  678. //拼接标题 数据列
  679. let data = [];
  680. let ydata = [];
  681. //有右二轴时排个序 按照左 右 右2的顺序
  682. let chartData = newval.some(_ =>_.IsAxis===2) ? this.changeEdbOrder(newval) : _.cloneDeep(newval);
  683. chartData.forEach((item, index) => {
  684. //轴位置值相同的下标
  685. let sameSideIndex = chartData.findIndex(
  686. (i) => i.IsAxis === item.IsAxis
  687. );
  688. //获取对应轴的上下限
  689. //预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除
  690. const useTableLimit = ['/predictEdb','/addpredictEdb','/editpredictEdb','/chartThemeSet'].includes(this.$route.path)
  691. //非ETA图库图表也不设置自定义上下限,相关性和统计特征也会用到曲线图
  692. //若chartInfo.Source为1,需在之前调用setLimitData
  693. const isETASource = this.chartInfo.Source===1
  694. let minLimit = 0,maxLimit = 0
  695. if(useTableLimit||!isETASource){
  696. minLimit = newval[sameSideIndex].MinData
  697. maxLimit = newval[sameSideIndex].MaxData
  698. }
  699. if(!useTableLimit&&isETASource){
  700. const limitMap = {
  701. 0:['rightMin','rightMax'],
  702. 1:['min','max'],
  703. 2:['rightTwoMin','rightTwoMax']
  704. }
  705. if(limitMap[item.IsAxis]){
  706. minLimit = this.chartLimit[`${limitMap[item.IsAxis][0]}`]||0
  707. maxLimit = this.chartLimit[`${limitMap[item.IsAxis][1]}`]||0
  708. }
  709. }
  710. //y轴
  711. const textZh = item.ConvertUnit||item.Unit
  712. const textEn = item.ConvertEnUnit||item.UnitEn||item.ConvertUnit||item.Unit
  713. let yItem = {
  714. ...basicYAxis,
  715. title: {
  716. text: textZh,
  717. textCh:textZh, // 中文
  718. // 中文不存在,无论英文有无都显示空
  719. textEn:textZh?textEn:'', // 英文
  720. style:{
  721. ...chartTheme&&chartTheme.yAxisOptions.style
  722. },
  723. align: 'high',
  724. rotation: 0,
  725. y: -12,
  726. x: (item.IsAxis===0 && this.rightTwoIndex>-1) ? -chartData[this.rightTwoIndex].Unit.length*12 : 0,
  727. textAlign: item.IsAxis===1 ? 'left' : 'right',
  728. reserveSpace: false
  729. },
  730. labels: {
  731. formatter: function (ctx) {
  732. return ctx.value;
  733. },
  734. align: 'center',
  735. x: [0,2].includes(item.IsAxis) ? 5 : -5,
  736. style: {
  737. ...chartTheme&&chartTheme.yAxisOptions.style,
  738. }
  739. },
  740. opposite: [0,2].includes(item.IsAxis),
  741. reversed: item.IsOrder,
  742. min: Number(minLimit),
  743. max: Number(maxLimit),
  744. tickWidth: 1,
  745. visible: sameSideIndex === index,
  746. plotBands: this.setAxisPlotAreas(item.IsAxis),
  747. plotLines: this.setAxisPlotLines(item.IsAxis)
  748. };
  749. //拼接标题 判断相同指标名称拼接来源
  750. let dynamic_title = item.EdbName;
  751. let dynamic_arr = chartData.filter(
  752. (item) => dynamic_title === item.EdbName
  753. );
  754. // 拼接配置 IsAxis左轴1 右轴0 右2轴2 IsOrder正序false 逆序true EdbInfoType是否是领先指标
  755. let dynamic_tag = this.concatDynamicTag(item);
  756. let dynamic_tag_en = this.concatDynamicTag(item,'en');
  757. //预测指标配置
  758. let predict_params = item.EdbInfoCategoryType === 1 ? this.getPredictParams(item) : {};
  759. //数据列
  760. let obj = {
  761. data: [],
  762. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  763. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  764. yAxis: sameSideIndex,
  765. name:
  766. dynamic_arr.length > 1
  767. ? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}`
  768. : `${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
  769. nameCh:dynamic_arr.length > 1
  770. ? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}`
  771. : `${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
  772. nameEn:item.EdbNameEn?`${item.EdbNameEn}${dynamic_tag_en}`:`${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
  773. color: item.ChartColor,
  774. lineWidth: Number(item.ChartWidth)||(chartTheme&&chartTheme.lineOptions.lineWidth),
  775. ...predict_params
  776. };
  777. item.DataList = item.DataList || [];
  778. for (let i of item.DataList) {
  779. obj.data.push([i.DataTimestamp, i.Value]);
  780. }
  781. data.push(obj);
  782. ydata.push(yItem);
  783. });
  784. /* x轴处理 */
  785. let isLessThanOneYear = this.xLabelDealHandle();
  786. let xAxis = {
  787. ...defaultOpts.xAxis,
  788. labels: {
  789. formatter: function (ctx) {
  790. return isLessThanOneYear
  791. ? Highcharts.dateFormat('%m/%d', ctx.value)
  792. : Highcharts.dateFormat('%y/%m', ctx.value);
  793. },
  794. style: {
  795. ...chartTheme&&chartTheme.xAxisOptions.style
  796. }
  797. },
  798. plotBands: this.setAxisPlotAreas(3,'datetime'),
  799. plotLines: this.setAxisPlotLines(3,'datetime')
  800. }
  801. this.options = {
  802. series: data,
  803. yAxis: ydata,
  804. xAxis
  805. };
  806. //滚动相关性独立tooltip
  807. if(this.chartInfo.Source === 4) {
  808. const { LeadValue,LeadUnit } = this.relevanceChartData.CorrelationChartInfo;
  809. let relevanceUnitEnMap = this.relevanceUnitEnMap;
  810. this.options.tooltip = {
  811. formatter: function() {
  812. let str = `${Highcharts.dateFormat('%Y/%m/%d',this.x)}<br><p>相关性系数:${this.y.toFixed(4)}</p><br><p>领先${LeadValue+LeadUnit}</p>`
  813. return str
  814. },
  815. formatterCh: function() {
  816. let str = `${Highcharts.dateFormat('%Y/%m/%d',this.x)}<br><p>相关性系数:${this.y.toFixed(4)}</p><br><p>领先${LeadValue+LeadUnit}</p>`
  817. return str
  818. },
  819. formatterEn: function() {
  820. let str = `${Highcharts.dateFormat('%Y/%m/%d',this.x)}<br><p>Correlation coefficient:${this.y.toFixed(4)}</p><br><p>lead${LeadValue+relevanceUnitEnMap[LeadUnit]}</p>`
  821. return str
  822. }
  823. }
  824. }
  825. if(this.currentLang=='en') this.changeOptions()
  826. },
  827. /* 堆叠图/组合图设置
  828. 本来和曲线图逻辑基本一致兼容下即可 为了以后便于维护和阅读还是拆开写吧
  829. */
  830. setStackOrCombinChart(newval) {
  831. const chartTypeMap = {
  832. 3: 'areaspline',
  833. 4: 'column',
  834. 6: ''
  835. };
  836. let chartStyle = chartTypeMap[this.chartInfo.ChartType];
  837. /* 主题样式*/
  838. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  839. //拼接标题 数据列
  840. let data = [];
  841. let ydata = [];
  842. //有右二轴时排个序 按照左 右 右2的顺序
  843. let chartData = newval.some(_ =>_.IsAxis===2) ? this.changeEdbOrder(newval) : _.cloneDeep(newval);
  844. chartData.forEach((item, index) => {
  845. //轴位置值相同的下标
  846. let sameSideIndex = chartData.findIndex(
  847. (i) => i.IsAxis === item.IsAxis
  848. );
  849. //堆叠图的yAxis必须一致 数据列所对应的y轴
  850. let serie_yIndex = index;
  851. if([3,4].includes(this.chartInfo.ChartType)) {
  852. // 类型为堆叠图时公用第一个指标y轴
  853. serie_yIndex = 0;
  854. } else if(this.chartInfo.ChartType ===6 && ['areaspline','column'].includes(item.ChartStyle)) {
  855. // 组合图找第一个堆叠柱状或面积的作为公用
  856. serie_yIndex = chartData.findIndex(i => i.ChartStyle === item.ChartStyle);
  857. }
  858. //数据对应的y轴是公用轴则配置也共享
  859. item.IsAxis = serie_yIndex === index ? item.IsAxis : chartData[serie_yIndex].IsAxis;
  860. item.IsOrder = serie_yIndex === index ? item.IsOrder : chartData[serie_yIndex].IsOrder;
  861. /* 上下限显示的设置 */
  862. this.leftIndex = [3,4].includes(this.chartInfo.ChartType)
  863. ? (chartData[serie_yIndex].IsAxis ? serie_yIndex : -1)
  864. : newval.findIndex((item) => item.IsAxis===1);
  865. this.rightIndex = [3,4].includes(this.chartInfo.ChartType)
  866. ? (chartData[serie_yIndex].IsAxis ? -1 : serie_yIndex)
  867. : newval.findIndex((item) => !item.IsAxis);
  868. this.rightTwoIndex = [3,4].includes(this.chartInfo.ChartType)
  869. ? -1
  870. : newval.findIndex((item) => item.IsAxis===2);
  871. //获取对应轴的上下限
  872. //预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除
  873. const useTableLimit = ['/predictEdb','/chartThemeSet'].includes(this.$route.path)
  874. let minLimit = 0,maxLimit = 0
  875. if(useTableLimit){
  876. minLimit = chartData[sameSideIndex].MinData
  877. maxLimit = chartData[sameSideIndex].MaxData
  878. }else{
  879. const limitMap = {
  880. 0:['rightMin','rightMax'],
  881. 1:['min','max'],
  882. 2:['rightTwoMin','rightTwoMax']
  883. }
  884. if(limitMap[item.IsAxis]){
  885. minLimit = this.chartLimit[`${limitMap[item.IsAxis][0]}`]||0
  886. maxLimit = this.chartLimit[`${limitMap[item.IsAxis][1]}`]||0
  887. }
  888. }
  889. //y轴
  890. const textZh = item.ConvertUnit||item.Unit
  891. const textEn = item.ConvertEnUnit||item.UnitEn||item.ConvertUnit||item.Unit
  892. let yItem = {
  893. ...basicYAxis,
  894. title: {
  895. text: textZh,
  896. textCh:textZh, // 中文
  897. // 中文不存在,无论英文有无都显示空
  898. textEn:textZh?textEn:'', // 英文
  899. style:{
  900. ...chartTheme&&chartTheme.yAxisOptions.style
  901. },
  902. align: 'high',
  903. rotation: 0,
  904. y: -12,
  905. x: (item.IsAxis===0 && this.rightTwoIndex>-1) ? -chartData[this.rightTwoIndex].Unit.length*12 : 0,
  906. textAlign: item.IsAxis===1 ? 'left' : 'right',
  907. reserveSpace: false
  908. },
  909. labels: {
  910. formatter: function (ctx) {
  911. return ctx.value;
  912. },
  913. align: 'center',
  914. x: [0,2].includes(item.IsAxis) ? 5 : -5,
  915. style: {
  916. ...chartTheme&&chartTheme.yAxisOptions.style
  917. }
  918. },
  919. opposite: [0,2].includes(item.IsAxis),
  920. reversed: item.IsOrder,
  921. min: Number(minLimit),
  922. max: Number(maxLimit),
  923. tickWidth: 1,
  924. visible: serie_yIndex === index && sameSideIndex ===index,
  925. plotBands: this.setAxisPlotAreas(item.IsAxis),
  926. plotLines: this.setAxisPlotLines(item.IsAxis)
  927. };
  928. //拼接标题 判断相同指标名称拼接来源
  929. let dynamic_title = item.EdbName;
  930. let dynamic_arr = chartData.filter(
  931. (item) => dynamic_title === item.EdbName
  932. );
  933. // 拼接配置 IsAxis左轴1 右轴0 IsOrder正序false 逆序true EdbInfoType是否是领先指标
  934. let dynamic_tag = this.concatDynamicTag(item);
  935. let dynamic_tag_en = this.concatDynamicTag(item,'en');
  936. //预测指标配置
  937. let predict_params = item.EdbInfoCategoryType === 1 ? this.getPredictParams(item,chartStyle) : {};
  938. //数据列
  939. let obj = {
  940. data: [],
  941. type: chartStyle || item.ChartStyle,
  942. yAxis: serie_yIndex,
  943. name:
  944. dynamic_arr.length > 1
  945. ? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}`
  946. : `${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
  947. nameCh:dynamic_arr.length > 1
  948. ? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}`
  949. : `${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
  950. nameEn:item.EdbNameEn?`${item.EdbNameEn}${dynamic_tag_en}`:`${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
  951. color: item.ChartColor,
  952. lineWidth: Number(item.ChartWidth)||(chartTheme&&chartTheme.lineOptions.lineWidth),
  953. fillColor: (this.chartInfo.ChartType === 3 || (this.chartInfo.ChartType === 6 && item.ChartStyle === 'areaspline')) ? item.ChartColor : undefined,
  954. borderWidth: 1,
  955. borderColor: item.ChartColor,
  956. zIndex: (this.chartInfo.ChartType === 6 && ['line','spline'].includes(item.ChartStyle)) ? 1 : 0, //防止组合图曲线被遮住
  957. ...predict_params
  958. };
  959. item.DataList = item.DataList || [];
  960. for (let i of item.DataList) {
  961. obj.data.push([i.DataTimestamp, i.Value]);
  962. }
  963. data.push(obj);
  964. ydata.push(yItem);
  965. });
  966. let isLessThanOneYear = this.xLabelDealHandle();
  967. let xAxis = {
  968. ...defaultOpts.xAxis,
  969. labels: {
  970. formatter: function (ctx) {
  971. return isLessThanOneYear
  972. ? Highcharts.dateFormat('%m/%d', ctx.value)
  973. : Highcharts.dateFormat('%y/%m', ctx.value);
  974. },
  975. style: {
  976. ...chartTheme&&chartTheme.xAxisOptions.style
  977. }
  978. },
  979. plotBands: this.setAxisPlotAreas(3,'datetime'),
  980. plotLines: this.setAxisPlotLines(3,'datetime')
  981. }
  982. this.options = {
  983. series: data,
  984. yAxis: ydata,
  985. xAxis,
  986. };
  987. if(this.currentLang=='en') this.changeOptions()
  988. },
  989. /* 季节图设置 农历 公历 */
  990. setSeasonChart(newval) {
  991. /* 季节性图的图表配置 */
  992. this.leftIndex = 0;
  993. this.rightIndex = -1;
  994. this.rightTwoIndex = -1;
  995. const chartData = newval[0];
  996. // 农历数据需要去除第一项 农历和公历处理逻辑一样
  997. /**
  998. * isPredictorChart - 预测指标的chartInfo.vue组件内定义的变量
  999. * 预测指标成图 还是之前农历图的逻辑,有空再拆吧,做个兼容
  1000. */
  1001. const chartDataHandle=this.calendar_type === '农历'?
  1002. this.isPredictorChart?chartData.DataList.List.filter((item, index) => index > 0):
  1003. chartData.DataList.filter((item, index) => index > 0):
  1004. chartData.DataList
  1005. /* 主题样式*/
  1006. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1007. let seasonYdata = [],
  1008. seasonData = [];
  1009. //获取对应轴的上下限
  1010. //预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除
  1011. const useTableLimit = ['/predictEdb','/chartThemeSet','/addpredictEdb','/editpredictEdb'].includes(this.$route.path)
  1012. let minLimit = 0,maxLimit = 0
  1013. if(useTableLimit){
  1014. minLimit = chartData.MinData
  1015. maxLimit = chartData.MaxData
  1016. }else{
  1017. minLimit = this.chartLimit.min||0
  1018. maxLimit = this.chartLimit.max||0
  1019. }
  1020. //数据列
  1021. for (let j of chartDataHandle) {
  1022. //预测指标配置
  1023. let predict_params = chartData.EdbInfoCategoryType === 1 ? this.getSeasonPredictParams(j.CuttingDataTimestamp) : {};
  1024. let serie_item = {
  1025. data: [],
  1026. type: (chartTheme&&chartTheme.lineOptions.lineType) || chartData.ChartStyle,
  1027. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  1028. yAxis: 0,
  1029. name: this.isPredictorChart?j.Year:j.ChartLegend,
  1030. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 1,
  1031. ...predict_params
  1032. };
  1033. const data_array = this.calendar_type === '农历' && this.isPredictorChart?_.cloneDeep(j.Items):_.cloneDeep(j.DataList);
  1034. data_array && data_array.forEach((item) => {
  1035. serie_item.data.push([item.DataTimestamp, item.Value]);
  1036. });
  1037. seasonData.push(serie_item);
  1038. }
  1039. //y轴
  1040. const textZh = chartData.ConvertUnit||chartData.Unit
  1041. const textEn = chartData.ConvertEnUnit||chartData.UnitEn||chartData.ConvertUnit||chartData.Unit
  1042. seasonYdata = [{
  1043. ...seasonOptions.yAxis,
  1044. labels: {
  1045. formatter: function () {
  1046. let val = this.value;
  1047. return val;
  1048. },
  1049. align: 'center',
  1050. style: {
  1051. ...chartTheme&&chartTheme.yAxisOptions.style
  1052. }
  1053. },
  1054. title: {
  1055. text: `${textZh}`,
  1056. textCh:textZh, // 中文
  1057. // 中文不存在,无论英文有无都显示空
  1058. textEn:textZh?textEn:'', // 英文
  1059. style:{
  1060. ...chartTheme&&chartTheme.yAxisOptions.style
  1061. },
  1062. // text: null,
  1063. align: 'high',
  1064. rotation: 0,
  1065. y: -12,
  1066. x: 0,
  1067. textAlign: 'left',
  1068. reserveSpace: false
  1069. },
  1070. max: Number(maxLimit),
  1071. min: Number(minLimit),
  1072. plotBands: this.setAxisPlotAreas(1),
  1073. plotLines: this.setAxisPlotLines(1)
  1074. }];
  1075. /* x轴显示月日 */
  1076. const xAxis = {
  1077. ...defaultOpts.xAxis,
  1078. labels: {
  1079. formatter: function () {
  1080. return Highcharts.dateFormat('%m/%d', this.value);
  1081. },
  1082. style: {
  1083. ...chartTheme&&chartTheme.xAxisOptions.style
  1084. }
  1085. },
  1086. plotBands: this.setAxisPlotAreas(3,'datetime'),
  1087. plotLines: this.setAxisPlotLines(3,'datetime')
  1088. }
  1089. const tooltip = {
  1090. ...defaultOpts.tooltip,
  1091. dateTimeLabelFormats: {
  1092. // 时间格式化字符
  1093. day: '%m/%d',
  1094. week: '%m/%d',
  1095. month: '%m/%d',
  1096. year: '%m/%d',
  1097. },
  1098. xDateFormat: '%m/%d',
  1099. }
  1100. let colors = chartTheme&&chartTheme.colorsOptions.reverse()||seasonOptions.colors;
  1101. // 季节性 农历成图老逻辑,需要截取 指标成图还在用,后面需要统一或者拆分
  1102. let rangeSelector =
  1103. this.calendar_type === '农历' && this.isPredictorChart
  1104. ? {
  1105. enabled: true,
  1106. selected: 0,
  1107. inputStyle: {
  1108. display: 'none',
  1109. },
  1110. labelStyle: {
  1111. display: 'none',
  1112. },
  1113. buttonTheme: {
  1114. style: {
  1115. display: 'none',
  1116. },
  1117. },
  1118. buttons: [
  1119. {
  1120. type: 'month',
  1121. count: 12,
  1122. text: '12月',
  1123. },
  1124. {
  1125. type: 'month',
  1126. count: 15,
  1127. text: '15月',
  1128. },
  1129. {
  1130. type: 'all',
  1131. text: '全部',
  1132. type: 'all',
  1133. },
  1134. ],
  1135. }
  1136. : {
  1137. enabled: false,
  1138. };
  1139. this.options = {
  1140. colors: colors.slice(-chartDataHandle.length),
  1141. series: seasonData,
  1142. yAxis: seasonYdata,
  1143. xAxis,
  1144. rangeSelector,
  1145. tooltip
  1146. };
  1147. if(this.currentLang=='en') this.changeOptions()
  1148. },
  1149. /* 散点图设置 只允许2指标画图第一个指标值为x轴 第二个指标为y轴 */
  1150. setScatterChart(newval) {
  1151. console.log(newval);
  1152. this.leftIndex = 1;
  1153. this.rightIndex = -1;
  1154. this.rightTwoIndex = -1;
  1155. if(newval.length !== 2) {
  1156. this.leftIndex = -1
  1157. this.options = {}
  1158. return
  1159. };
  1160. console.log('===========:',this.isSetExtremeValue)
  1161. /* 主题样式*/
  1162. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1163. // 取2个指标中日期相同的数据
  1164. let real_data = [];
  1165. let tmpData_date = {};//用来取点对应的日期
  1166. let data1 = _.cloneDeep(newval)[0].DataList || [];
  1167. let data2 = _.cloneDeep(newval)[1].DataList || [];
  1168. data1.forEach(_item => {
  1169. data2.forEach(_item2 => {
  1170. if(_item.DataTimestamp === _item2.DataTimestamp) {
  1171. _item.DataTime = _item.DataTime.replace(/-/g,'/');
  1172. //日期
  1173. let itemIndex =_item.Value + "_" +_item2.Value
  1174. if(tmpData_date[itemIndex]) {
  1175. tmpData_date[itemIndex].push(_item.DataTime)
  1176. } else {
  1177. tmpData_date[itemIndex] = [_item.DataTime]
  1178. }
  1179. //值
  1180. real_data.push({
  1181. x: _item.Value,
  1182. y: _item2.Value
  1183. })
  1184. }
  1185. })
  1186. })
  1187. real_data.sort((x,y) => x-y);
  1188. //悬浮窗 拼接日期 原始指标名称
  1189. let tooltip = {
  1190. formatter: function() {
  1191. return `<strong>${ tmpData_date[this.x+'_'+this.y].length > 4 ? tmpData_date[this.x+'_'+this.y].slice(0,4).join()+'...' : tmpData_date[this.x+'_'+this.y].join() }</strong><br>
  1192. ${newval[0].EdbName}: <span style="font-weight: 600"> ${this.x}</span><br>
  1193. ${newval[1].EdbName}: <span style="font-weight: 600"> ${this.y}</span>
  1194. `
  1195. },
  1196. // 中文
  1197. formatterCh: function() {
  1198. return `<strong>${ tmpData_date[this.x+'_'+this.y].length > 4 ? tmpData_date[this.x+'_'+this.y].slice(0,4).join()+'...' : tmpData_date[this.x+'_'+this.y].join() }</strong><br>
  1199. ${newval[0].EdbName}: <span style="font-weight: 600"> ${this.x}</span><br>
  1200. ${newval[1].EdbName}: <span style="font-weight: 600"> ${this.y}</span>
  1201. `
  1202. },
  1203. // 英文
  1204. formatterEn: function() {
  1205. let str1 = newval[0].EdbNameEn||newval[0].EdbName;
  1206. let str2 = newval[1].EdbNameEn||newval[1].EdbName;
  1207. return `<strong>${ tmpData_date[this.x+'_'+this.y].length > 4 ? tmpData_date[this.x+'_'+this.y].slice(0,4).join()+'...' : tmpData_date[this.x+'_'+this.y].join() }</strong><br>
  1208. ${str1}: <span style="font-weight: 600"> ${this.x}</span><br>
  1209. ${str2}: <span style="font-weight: 600"> ${this.y}</span>
  1210. `
  1211. }
  1212. }
  1213. // 目前指标限制死2个 为了简单 配置项直接关联第一个指标算了 自定义极值只在新增时用
  1214. //ETA1.5.8 增加数据转换功能,不用第一个指标了
  1215. /* if(!this.chartInfo.ChartInfoId) {
  1216. let maxYData = Math.max(...real_data.map(_ => _.y));
  1217. let minYData = Math.min(...real_data.map(_ => _.y));
  1218. newval[0].MaxData = maxYData;
  1219. newval[0].MinData = minYData;
  1220. } */
  1221. //获取对应轴的上下限
  1222. //预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除
  1223. const useTableLimit = ['/predictEdb','/chartThemeSet'].includes(this.$route.path)
  1224. let minLimit = 0,maxLimit = 0
  1225. if(useTableLimit){
  1226. minLimit = Math.min(...real_data.map(_ => _.y));
  1227. maxLimit = Math.max(...real_data.map(_ => _.y));
  1228. }else{
  1229. minLimit = this.chartLimit.min||0
  1230. maxLimit = this.chartLimit.max||0
  1231. }
  1232. const { IsOrder,ChartColor } = newval[0];
  1233. //y轴
  1234. const textYZh = newval[1].ConvertUnit||newval[1].Unit
  1235. const textYEn = newval[1].ConvertEnUnit||newval[1].UnitEn||newval[1].ConvertUnit||newval[1].Unit
  1236. let yAxis = {
  1237. ...basicYAxis,
  1238. title: {
  1239. text: `${textYZh}`,
  1240. textCh:textYZh,// 中文
  1241. // 中文不存在,无论英文有无都显示空
  1242. textEn:textYZh?textYEn:'', // 英文
  1243. style:{
  1244. ...chartTheme&&chartTheme.yAxisOptions.style
  1245. },
  1246. align: 'high',
  1247. rotation: 0,
  1248. y: -12,
  1249. x:0,
  1250. textAlign: 'left',
  1251. reserveSpace: false
  1252. },
  1253. labels: {
  1254. formatter: function (ctx) {
  1255. let val = ctx.value;
  1256. return val;
  1257. },
  1258. align: 'center',
  1259. style:{
  1260. ...chartTheme&&chartTheme.yAxisOptions.style
  1261. },
  1262. },
  1263. opposite: false,
  1264. reversed: IsOrder,
  1265. min: Number(minLimit),
  1266. max: Number(maxLimit),
  1267. tickWidth: 1,
  1268. plotBands: this.setAxisPlotAreas(1),
  1269. plotLines: this.setAxisPlotLines(1)
  1270. }
  1271. //x轴
  1272. const textXZh = newval[0].ConvertUnit||newval[0].Unit
  1273. const textXEn = newval[0].ConvertEnUnit||newval[0].UnitEn||newval[0].ConvertUnit||newval[0].Unit
  1274. const xAxis = {
  1275. ...scatterXAxis,
  1276. title: {
  1277. text: `${textXZh}`,
  1278. textCh:textXZh, // 中文
  1279. // 中文不存在,无论英文有无都显示空
  1280. textEn:textXZh?textXEn:'', // 英文
  1281. style: {
  1282. ...chartTheme&&chartTheme.xAxisOptions.style
  1283. },
  1284. align: 'high',
  1285. rotation: 0,
  1286. x: 0,
  1287. offset: 20,
  1288. },
  1289. labels: {
  1290. style: {
  1291. ...chartTheme&&chartTheme.xAxisOptions.style
  1292. }
  1293. },
  1294. plotBands: this.setAxisPlotAreas(3),
  1295. plotLines: this.setAxisPlotLines(3)
  1296. }
  1297. //数据列
  1298. let series = {
  1299. data: [],
  1300. type: 'scatter',
  1301. name: `${this.chartInfo.ChartName}${IsOrder ? '(逆序)' : ''}`,
  1302. nameCh: `${this.chartInfo.ChartName}${IsOrder ? '(逆序)' : ''}`,
  1303. nameEn: `${this.chartInfo.ChartNameEn||this.chartInfo.ChartName}${IsOrder ? '(reverse)' : ''}`,
  1304. color: ChartColor,
  1305. chartType: 'linear',
  1306. marker: {
  1307. radius: (chartTheme&&chartTheme.lineOptions.radius)||5,
  1308. },
  1309. }
  1310. real_data.forEach(_ => {
  1311. series.data.push([_.x,_.y])
  1312. })
  1313. this.options = {
  1314. title: {
  1315. text:''
  1316. },
  1317. series: [ series ],
  1318. yAxis,
  1319. xAxis,
  1320. tooltip
  1321. }
  1322. if(this.currentLang=='en') this.changeOptions()
  1323. },
  1324. /* 奇怪柱状图 和以上逻辑无公用点 依赖数据为单独的数据
  1325. x轴为指标名称的柱形图 以日期作为series
  1326. */
  1327. setBarChart() {
  1328. this.leftIndex = -1;
  1329. this.rightIndex = -1;
  1330. this.rightTwoIndex = -1;
  1331. let seriesData = [];
  1332. const data = _.cloneDeep(this.barDateList);
  1333. /* 主题样式*/
  1334. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1335. //x轴
  1336. let xAxis = {
  1337. ...scatterXAxis,
  1338. categories: this.barXData,
  1339. tickWidth: 1,
  1340. labels: {
  1341. style: {
  1342. ...chartTheme&&chartTheme.xAxisOptions.style
  1343. }
  1344. }
  1345. }
  1346. const { max,min } = this.chartLimit;
  1347. console.log(max,min)
  1348. //y轴
  1349. let yAxis = {
  1350. ...basicYAxis,
  1351. title: {
  1352. text: this.chartInfo.Unit,
  1353. textCh: this.chartInfo.Unit,
  1354. textEn: this.chartInfo.UnitEn,
  1355. align: 'high',
  1356. rotation: 0,
  1357. y: -12,
  1358. x:0,
  1359. textAlign: 'left',
  1360. reserveSpace: false,
  1361. style:{
  1362. ...chartTheme&&chartTheme.yAxisOptions.style
  1363. },
  1364. },
  1365. labels: {
  1366. formatter: function (ctx) {
  1367. let val = ctx.value;
  1368. return val;
  1369. },
  1370. align: 'center',
  1371. style:{
  1372. ...chartTheme&&chartTheme.yAxisOptions.style
  1373. },
  1374. },
  1375. min: Number(min),
  1376. max: Number(max),
  1377. opposite: false,
  1378. tickWidth: 1,
  1379. plotBands: this.setAxisPlotAreas(1),
  1380. plotLines: this.setAxisPlotLines(1)
  1381. }
  1382. //数据列
  1383. data.forEach(item => {
  1384. let serie_item = {
  1385. data: item.Value,
  1386. type: 'column',
  1387. yAxis: 0,
  1388. name: item.Name || item.Date,
  1389. nameCh: item.Name || item.Date,
  1390. nameEn: item.Date,
  1391. color: item.Color,
  1392. chartType: 'linear'
  1393. };
  1394. seriesData.push(serie_item)
  1395. })
  1396. this.options = {
  1397. title: {
  1398. text:''
  1399. },
  1400. plotOptions: {
  1401. column:{
  1402. stacking: null,
  1403. },
  1404. },
  1405. series: seriesData,
  1406. yAxis: [ yAxis ],
  1407. xAxis
  1408. }
  1409. this.currentLang=='en' && this.changeOptions();
  1410. },
  1411. /* 商品价格曲线设置 绘图逻辑同奇怪柱形图*/
  1412. setCommodityChart() {
  1413. //上下限都不通用
  1414. this.leftIndex = -1;
  1415. this.rightIndex = -1;
  1416. this.rightTwoIndex = -1;
  1417. let seriesData = [];
  1418. const data = _.cloneDeep(this.commodityChartData);
  1419. /* 主题样式*/
  1420. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1421. //x轴
  1422. let xAxis = {
  1423. ...scatterXAxis,
  1424. categories: this.commodityXData.map(_ =>_.Name),
  1425. tickWidth: 1,
  1426. labels: {
  1427. style: {
  1428. ...chartTheme&&chartTheme.xAxisOptions.style
  1429. }
  1430. }
  1431. }
  1432. const { max,min } = this.chartLimit;
  1433. //y轴
  1434. let yAxis = {
  1435. ...basicYAxis,
  1436. title: {
  1437. text: this.commodityEdbList[0].Unit,
  1438. textCh: this.commodityEdbList[0].Unit,
  1439. textEn: this.commodityEdbList[0].UnitEn||this.commodityEdbList[0].Unit,
  1440. style:{
  1441. ...chartTheme&&chartTheme.yAxisOptions.style
  1442. },
  1443. align: 'high',
  1444. rotation: 0,
  1445. y: -12,
  1446. textAlign: 'left',
  1447. reserveSpace: false
  1448. },
  1449. labels: {
  1450. formatter: function (ctx) {
  1451. let val = ctx.value;
  1452. return val;
  1453. },
  1454. align: 'center',
  1455. style:{
  1456. ...chartTheme&&chartTheme.yAxisOptions.style
  1457. },
  1458. },
  1459. min: Number(min),
  1460. max: Number(max),
  1461. opposite: false,
  1462. tickWidth: 1,
  1463. }
  1464. //数据列
  1465. data.forEach(item => {
  1466. //处理首或/尾全是无效数据的以null填充
  1467. let filterData = this.filterInvalidData(item)
  1468. // console.log(filterData)
  1469. let serie_item = {
  1470. data: filterData,
  1471. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  1472. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  1473. yAxis: 0,
  1474. name: item.Name,
  1475. nameCh: item.Name,
  1476. nameEn: item.NameEn,
  1477. color: item.Color,
  1478. chartType: 'linear',
  1479. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 3,
  1480. marker: {
  1481. enabled: false
  1482. }
  1483. };
  1484. seriesData.push(serie_item)
  1485. })
  1486. //tooltip
  1487. let commodityEdbList = this.commodityEdbList;
  1488. let commodityXData = this.commodityXData;
  1489. let chartInfo = this.chartInfo;
  1490. let tooltip = {
  1491. formatter: function() {
  1492. let str = '';
  1493. this.points.forEach(item => {
  1494. let obj_item = data.find(_ => _.Name === item.series.name);
  1495. let index = commodityXData.findIndex(_ => _.Name === this.x);
  1496. //合约显示
  1497. let haveContract = obj_item.XEdbInfoIdList[index];
  1498. if(haveContract) {
  1499. // 利润曲线指标名
  1500. let edb_name = chartInfo.Source === 5
  1501. ? (index === 0 ? obj_item.NameList[index] : `${chartInfo.ProfitName}(${obj_item.NameList[index]})`)
  1502. : commodityEdbList.find(_ => _.EdbInfoId === obj_item.XEdbInfoIdList[index]).EdbName;
  1503. str+=`<b>${ edb_name }</b>`
  1504. if(!obj_item.NoDataEdbList.includes(obj_item.XEdbInfoIdList[index])) {
  1505. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: ${item.y}<br>`
  1506. }else {
  1507. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: 无<br>`
  1508. }
  1509. }
  1510. })
  1511. return str||'无合约'
  1512. },
  1513. formatterCh: function() {
  1514. let str = '';
  1515. this.points.forEach(item => {
  1516. let obj_item = data.find(_ => _.Name === item.series.name);
  1517. let index = commodityXData.findIndex(_ => _.Name === this.x);
  1518. //合约显示
  1519. let haveContract = obj_item.XEdbInfoIdList[index];
  1520. if(haveContract) {
  1521. // 利润曲线指标名
  1522. let edb_name = chartInfo.Source === 5
  1523. ? (index === 0 ? obj_item.NameList[index] : `${chartInfo.ProfitName}(${obj_item.NameList[index]})`)
  1524. : commodityEdbList.find(_ => _.EdbInfoId === obj_item.XEdbInfoIdList[index]).EdbName;
  1525. str+=`<b>${ edb_name }</b>`
  1526. if(!obj_item.NoDataEdbList.includes(obj_item.XEdbInfoIdList[index])) {
  1527. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: ${item.y}<br>`
  1528. }else {
  1529. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: 无<br>`
  1530. }
  1531. }
  1532. })
  1533. return str||'无合约'
  1534. },
  1535. formatterEn: function() {
  1536. let str = '';
  1537. this.points.forEach(item => {
  1538. let obj_item = data.find(_ => _.NameEn === item.series.name);
  1539. let index = commodityXData.findIndex(_ => _.NameEn === this.x);
  1540. //合约显示
  1541. let haveContract = obj_item.XEdbInfoIdList[index];
  1542. if(haveContract) {
  1543. // 利润曲线指标名
  1544. let edb_name = chartInfo.Source === 5
  1545. ? (index === 0 ? obj_item.NameList[index] : `${chartInfo.ProfitNameEn}(${obj_item.NameList[index]})`)
  1546. : commodityEdbList.find(_ => _.EdbInfoId === obj_item.XEdbInfoIdList[index]).EdbNameEn;
  1547. str+=`<b>${ edb_name }</b>`
  1548. if(!obj_item.NoDataEdbList.includes(obj_item.XEdbInfoIdList[index])) {
  1549. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: ${item.y}<br>`
  1550. }else {
  1551. str += `<br><span style="color:${item.color}">\u25CF</span>${obj_item.Date}: 无<br>`
  1552. }
  1553. }
  1554. })
  1555. return str||'无合约'
  1556. },
  1557. shared: true,
  1558. }
  1559. this.options = {
  1560. title: {
  1561. text:''
  1562. },
  1563. series: seriesData,
  1564. yAxis: [ yAxis ],
  1565. xAxis,
  1566. tooltip
  1567. }
  1568. this.currentLang==='en' && this.changeCommodityLang();
  1569. },
  1570. /* 处理无效数据为null */
  1571. filterInvalidData(item) {
  1572. //找出第一个有效数据和最后一个有效数据的index index1 index2
  1573. //0到index1全填null index2到最后一个全为null
  1574. let validateArr = item.XEdbInfoIdList.filter(_ =>_&&!item.NoDataEdbList.includes(_));
  1575. let first_index = item.XEdbInfoIdList.findIndex(_ => _ === validateArr[0]);
  1576. let last_index = item.XEdbInfoIdList.findIndex(_ => _ === validateArr[validateArr.length-1]);
  1577. console.log('first_index',first_index)
  1578. console.log('last_index',last_index)
  1579. let arr = item.Value.map((item,index) => {
  1580. if(index < first_index || index > last_index) {
  1581. return null
  1582. }else {
  1583. return item
  1584. }
  1585. })
  1586. return arr;
  1587. },
  1588. /* 获取图表详情后赋值柱状图数据 */
  1589. initBarData(data) {
  1590. const { XEdbIdValue,YDataList,EdbInfoList,ChartInfo } = data;
  1591. let xData = XEdbIdValue.map(_ => EdbInfoList.find(edb => edb.EdbInfoId===_).EdbAliasName)
  1592. this.leftIndex = -1;
  1593. this.rightIndex = -1;
  1594. this.rightTwoIndex = -1;
  1595. this.barXIdList = XEdbIdValue;
  1596. this.barDateList = YDataList;
  1597. this.barXData = xData;
  1598. this.barEdbData = EdbInfoList;
  1599. this.chartLimit = {
  1600. min: Number(ChartInfo.LeftMin),
  1601. max: Number(ChartInfo.LeftMax)
  1602. }
  1603. },
  1604. /* 商品价格曲线获取详情赋值 */
  1605. initCommodityData(data) {
  1606. const { XDataList,YDataList,EdbInfoList,ChartInfo,DataResp } = data;
  1607. this.commodityEdbList = EdbInfoList;
  1608. this.commodityChartData = ChartInfo.Source===5?DataResp.YDataList:YDataList;
  1609. this.commodityXData = ChartInfo.Source===5?DataResp.XDataList:XDataList;
  1610. this.chartLimit = {
  1611. min: Number(ChartInfo.LeftMin),
  1612. max: Number(ChartInfo.LeftMax)
  1613. }
  1614. this.setCommodityChart()
  1615. },
  1616. /* 相关性图表初始化 source4 滚动相关性*/
  1617. initRelevanceChartData(){
  1618. this.leftIndex = -1;
  1619. this.rightIndex = -1;
  1620. this.rightTwoIndex = -1;
  1621. /* 主题样式*/
  1622. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1623. // 处理X轴
  1624. let xAxis={
  1625. categories: this.relevanceChartData.XEdbIdValue,
  1626. tickWidth: 1,
  1627. title: {
  1628. text: this.relevanceChartData.ChartInfo.Source===3 ?`期数(${this.relevanceChartData.CorrelationChartInfo.LeadUnit})` : null,
  1629. textCh:this.relevanceChartData.ChartInfo.Source===3 ? `期数(${this.relevanceChartData.CorrelationChartInfo.LeadUnit})`:null,
  1630. textEn:this.relevanceChartData.ChartInfo.Source===3 ? `stage(${this.relevanceUnitEnMap[this.relevanceChartData.CorrelationChartInfo.LeadUnit]})`:null,
  1631. align: 'high',
  1632. rotation: 0,
  1633. style: {
  1634. ...chartTheme&&chartTheme.xAxisOptions.style
  1635. }
  1636. },
  1637. labels: {
  1638. style: {
  1639. ...chartTheme&&chartTheme.xAxisOptions.style
  1640. }
  1641. },
  1642. tickInterval: 1,
  1643. offset:0,
  1644. tickmarkPlacement:'on'
  1645. }
  1646. // 处理Y轴
  1647. let yAxis={
  1648. ...basicYAxis,
  1649. title: {
  1650. text: '相关性系数',
  1651. textCh: '相关性系数',
  1652. textEn: 'Correlation coefficient',
  1653. align: 'high',
  1654. rotation: 0,
  1655. y: -12,
  1656. textAlign: 'left',
  1657. reserveSpace: false,
  1658. style:{
  1659. ...chartTheme&&chartTheme.yAxisOptions.style
  1660. },
  1661. },
  1662. labels: {
  1663. formatter: function (ctx) {
  1664. let val = ctx.value;
  1665. return val;
  1666. },
  1667. align: 'center',
  1668. style:{
  1669. ...chartTheme&&chartTheme.yAxisOptions.style
  1670. },
  1671. },
  1672. opposite: false,
  1673. tickWidth: 1,
  1674. }
  1675. //处理series
  1676. let seriesData=[]
  1677. this.relevanceChartData.YDataList.forEach(item=>{
  1678. let serie_item = {
  1679. data: item.Value,
  1680. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  1681. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  1682. yAxis: 0,
  1683. name: item.Name,
  1684. nameCh: item.Name,
  1685. nameEn: item.NameEn,
  1686. color: item.Color,
  1687. chartType: 'linear',
  1688. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 3,
  1689. marker: {
  1690. enabled: false
  1691. }
  1692. };
  1693. seriesData.push(serie_item)
  1694. })
  1695. const { LeadValue,LeadUnit } = this.relevanceChartData.CorrelationChartInfo;
  1696. const { Source } = this.relevanceChartData.ChartInfo;
  1697. let relevanceUnitEnMap = this.relevanceUnitEnMap;
  1698. let tooltip = {
  1699. formatter: function() {
  1700. let str = `<p>相关性系数:${this.y.toFixed(4)}</p><br><p>领先${ Source===3 ?this.x+'期' : LeadValue+LeadUnit}</p>`
  1701. return str
  1702. },
  1703. formatterCh: function() {
  1704. let str = `<p>相关性系数:${this.y.toFixed(4)}</p><br><p>领先${ Source===3 ?this.x+'期' : LeadValue+LeadUnit}</p>`
  1705. return str
  1706. },
  1707. formatterEn: function() {
  1708. let str = `<p>Correlation coefficient:${this.y.toFixed(4)}</p><br><p>lead${ Source===3 ? this.x+'stage' : LeadValue+relevanceUnitEnMap[LeadUnit]}</p>`
  1709. return str
  1710. }
  1711. }
  1712. this.options = {
  1713. isRelevanceChart: Source===3,
  1714. title: {
  1715. text:''
  1716. },
  1717. series: seriesData,
  1718. yAxis: [yAxis] ,
  1719. xAxis:xAxis,
  1720. tooltip
  1721. }
  1722. console.log('处理结束相关性图表',this.options);
  1723. },
  1724. /* 截面散点图获取详情赋值 */
  1725. initSectionScatterData({ DataResp }) {
  1726. this.chartLimit = {
  1727. min: Number(DataResp.YMinValue),
  1728. max: Number(DataResp.YMaxValue),
  1729. x_min: Number(DataResp.XMinValue),
  1730. x_max: Number(DataResp.XMaxValue)
  1731. }
  1732. this.sectionScatterData = DataResp;
  1733. },
  1734. /* 截面散点图设置 sectionScatterData */
  1735. setSectionScatterChart() {
  1736. this.leftIndex = -1;
  1737. this.rightIndex = -1;
  1738. this.rightTwoIndex = -1;
  1739. const { DataList,XName,XNameEn,YName,YNameEn } = this.sectionScatterData;
  1740. const { min,max,x_min,x_max } = this.chartLimit;
  1741. /* 主题样式*/
  1742. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1743. //y轴
  1744. let yAxis = {
  1745. ...basicYAxis,
  1746. title: {
  1747. text: YName,
  1748. textCh:YName,// 中文
  1749. textEn:YNameEn||YName,
  1750. style:{
  1751. ...chartTheme&&chartTheme.yAxisOptions.style
  1752. },
  1753. align: 'middle',
  1754. },
  1755. labels: {
  1756. style:{
  1757. ...chartTheme&&chartTheme.yAxisOptions.style
  1758. },
  1759. },
  1760. opposite: false,
  1761. reversed: false,
  1762. min: Number(min),
  1763. max: Number(max),
  1764. tickWidth: 1,
  1765. plotBands: this.setAxisPlotAreas(1),
  1766. plotLines: this.setAxisPlotLines(1)
  1767. }
  1768. //x轴
  1769. let xAxis = {
  1770. ...scatterXAxis,
  1771. title: {
  1772. text: XName,
  1773. textCh:XName,// 中文
  1774. textEn:XNameEn || XName,
  1775. style: {
  1776. ...chartTheme&&chartTheme.xAxisOptions.style
  1777. },
  1778. align: 'middle',
  1779. },
  1780. labels: {
  1781. style:{
  1782. ...chartTheme&&chartTheme.xAxisOptions.style
  1783. },
  1784. },
  1785. min: Number(x_min),
  1786. max: Number(x_max),
  1787. plotBands: this.setAxisPlotAreas(3),
  1788. plotLines: this.setAxisPlotLines(3)
  1789. }
  1790. //数据列
  1791. let series = [];
  1792. DataList.forEach(item => {
  1793. //数据列
  1794. let series_item = {
  1795. data: [],
  1796. type: 'scatter',
  1797. name: item.Name,
  1798. nameCh: item.Name,
  1799. nameEn: item.NameEn||item.Name,
  1800. color: item.Color,
  1801. chartType: 'linear',
  1802. zIndex:1,
  1803. marker: {
  1804. radius: (chartTheme&&chartTheme.lineOptions.radius)||5,
  1805. },
  1806. }
  1807. item.EdbInfoList.forEach(_ => {
  1808. series_item.data.push({
  1809. x: _.XValue,
  1810. y: _.YValue,
  1811. dataLabels: {
  1812. enabled: _.IsShow,
  1813. allowOverlap: true,
  1814. align: 'left',
  1815. format: _.Name,
  1816. formatCh: _.Name,
  1817. formatEn: _.NameEn||_.Name
  1818. }
  1819. })
  1820. })
  1821. series.push(series_item);
  1822. //趋势线
  1823. if(item.ShowTrendLine) {
  1824. let trend_data = item.TrendLimitData.map((_,_index) => (
  1825. _index === item.TrendLimitData.length-1 ? {
  1826. x: _.X,
  1827. y: _.Y,
  1828. dataLabels: {
  1829. enabled: item.ShowRSquare || item.ShowFitEquation,
  1830. align: 'left',
  1831. color: '#666',
  1832. x: 20,
  1833. y: 30,
  1834. zIndex: 9,
  1835. allowOverlap: true,
  1836. formatter: function(){
  1837. let tag = '';
  1838. item.ShowRSquare && item.ShowFitEquation
  1839. ? tag =`<span>${item.TrendLine}</span><br><span>R²=${item.RSquare}</span>`
  1840. : item.ShowRSquare && !item.ShowFitEquation
  1841. ? tag =`<span>R²=${item.RSquare}</span>`
  1842. : item.ShowFitEquation && !item.ShowRSquare
  1843. ? tag =`<span>${item.TrendLine}</span>`
  1844. : ''
  1845. return tag
  1846. }
  1847. }
  1848. } : {
  1849. x: _.X,
  1850. y: _.Y,
  1851. }
  1852. ))
  1853. let trend_item = {
  1854. data: trend_data,
  1855. type: 'spline',
  1856. linkedTo: ':previous',
  1857. color: item.Color,
  1858. lineWidth: 1,
  1859. chartType: 'linear',
  1860. enableMouseTracking: false,
  1861. dashStyle:'Dash',
  1862. zIndex: 2,
  1863. marker: {
  1864. enabled: false
  1865. }
  1866. }
  1867. series.push(trend_item)
  1868. }
  1869. })
  1870. let tooltip = {
  1871. formatter: function() {
  1872. let series_obj = DataList.find(_ => _.Name === this.series.name);
  1873. let ponit_obj = series_obj.EdbInfoList.find(_ => _.XValue ===this.x && _.YValue===this.y);
  1874. let str=`<b>${ ponit_obj.Name }</b>`;
  1875. str += `<br><span style="color:${this.color}">\u25CF</span>${ponit_obj.XName}: ${this.x} ${ponit_obj.XDate}<br>`;
  1876. str += `<span style="color:${this.color}">\u25CF</span>${ponit_obj.YName}: ${this.y} ${ponit_obj.YDate}`;
  1877. return str
  1878. },
  1879. formatterCh: function() {
  1880. let series_obj = DataList.find(_ => _.Name === this.series.name);
  1881. let ponit_obj = series_obj.EdbInfoList.find(_ => _.XValue ===this.x && _.YValue===this.y);
  1882. let str=`<b>${ ponit_obj.Name }</b>`;
  1883. str += `<br><span style="color:${this.color}">\u25CF</span>${ponit_obj.XName}: ${this.x} ${ponit_obj.XDate}<br>`;
  1884. str += `<span style="color:${this.color}">\u25CF</span>${ponit_obj.YName}: ${this.y} ${ponit_obj.YDate}`;
  1885. return str
  1886. },
  1887. formatterEn: function() {
  1888. let series_obj = DataList.find(_ => _.NameEn === this.series.name);
  1889. let ponit_obj = series_obj.EdbInfoList.find(_ => _.XValue ===this.x && _.YValue===this.y);
  1890. let str=`<b>${ ponit_obj.NameEn }</b>`;
  1891. str += `<br><span style="color:${this.color}">\u25CF</span>${ponit_obj.XNameEn}: ${this.x} ${ponit_obj.XDate}<br>`;
  1892. str += `<span style="color:${this.color}">\u25CF</span>${ponit_obj.YNameEn}: ${this.y} ${ponit_obj.YDate}`;
  1893. return str
  1894. }
  1895. }
  1896. this.options = {
  1897. title: {
  1898. text:''
  1899. },
  1900. series,
  1901. yAxis: [yAxis],
  1902. xAxis,
  1903. tooltip
  1904. }
  1905. this.currentLang=='en' && this.changeOptions()
  1906. },
  1907. //雷达图数据初始化
  1908. initRadarData(data) {
  1909. const { DataResp,EdbInfoList,ChartInfo } = data;
  1910. this.leftIndex = -1;
  1911. this.rightIndex = -1;
  1912. this.rightTwoIndex = -1;
  1913. this.radarChartData = {
  1914. YDataList: DataResp.YDataList,
  1915. XDataList: EdbInfoList.filter(_ => DataResp.XEdbIdValue.includes(_.EdbInfoId))
  1916. }
  1917. this.chartLimit = {
  1918. min: Number(ChartInfo.LeftMin),
  1919. max: Number(ChartInfo.LeftMax)
  1920. }
  1921. this.setRadarChart();
  1922. },
  1923. /*雷达图绘图*/
  1924. setRadarChart() {
  1925. const { YDataList,XDataList } = this.radarChartData;
  1926. /* 主题样式*/
  1927. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  1928. //x轴
  1929. let xAxis = {
  1930. lineWidth: 0,
  1931. tickLength: 0,
  1932. tickmarkPlacement: 'on',
  1933. categories: XDataList.map(_ => _.EdbAliasName||_.EdbName),
  1934. labels: {
  1935. allowOverlap: true,
  1936. autoRotationLimit: 40,
  1937. distance: 20,
  1938. style: {
  1939. ...chartTheme&&chartTheme.xAxisOptions.style
  1940. }
  1941. }
  1942. }
  1943. //y轴
  1944. const { max,min } = this.chartLimit;
  1945. let yAxis = [{
  1946. gridLineInterpolation: 'polygon',
  1947. gridLineWidth: 1,
  1948. lineWidth: 0,
  1949. endOnTick: false,
  1950. startOnTick: false,
  1951. showLastLabel: true,
  1952. // tickAmount:4,
  1953. title: {
  1954. text: this.chartInfo.Unit,
  1955. textCh: this.chartInfo.Unit,
  1956. textEn: this.chartInfo.UnitEn,
  1957. align: 'high',
  1958. rotation: 0,
  1959. y: 5,
  1960. x:10,
  1961. textAlign: 'left',
  1962. reserveSpace: false,
  1963. style:{
  1964. ...chartTheme&&chartTheme.yAxisOptions.style
  1965. },
  1966. },
  1967. labels: {
  1968. allowOverlap: true,
  1969. style:{
  1970. ...chartTheme&&chartTheme.yAxisOptions.style
  1971. }
  1972. },
  1973. min: Number(min),
  1974. max: Number(max),
  1975. }]
  1976. //系列
  1977. let series = [];
  1978. YDataList.forEach(item => {
  1979. let serie_item = {
  1980. data: item.Value,
  1981. pointPlacement: 'on',
  1982. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'line',
  1983. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  1984. yAxis: 0,
  1985. name: item.Name || item.Date,
  1986. nameCh: item.Name || item.Date,
  1987. nameEn: item.Date,
  1988. color: item.Color,
  1989. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 1,
  1990. chartType: 'linear'
  1991. };
  1992. series.push(serie_item)
  1993. })
  1994. this.options = {
  1995. chart: {
  1996. ...defaultOpts.chart,
  1997. ...chartTheme.drawOption,
  1998. spacing: [2,10,2,10],
  1999. polar:true,
  2000. },
  2001. title: {
  2002. text:''
  2003. },
  2004. pane: {
  2005. size: '80%'
  2006. },
  2007. series,
  2008. yAxis,
  2009. xAxis
  2010. }
  2011. this.currentLang=='en' && this.changeOptions();
  2012. },
  2013. /* 统计频率图 */
  2014. setStatisticFrequency() {
  2015. this.leftIndex = -1;
  2016. this.rightIndex = -1;
  2017. this.rightTwoIndex = -1;
  2018. /* 主题样式*/
  2019. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  2020. const { DataList,LeftMaxValue,LeftMinValue,RightMaxValue,RightMinValue } = this.statisticFrequencyData;
  2021. let xAxis = {
  2022. ...scatterXAxis,
  2023. tickWidth: 1,
  2024. labels: {
  2025. style: {
  2026. ...chartTheme&&chartTheme.xAxisOptions.style
  2027. }
  2028. }
  2029. }
  2030. //y和系列
  2031. let yAxis = [],series = [];
  2032. DataList.forEach((item,index) => {
  2033. let y_item = {
  2034. ...basicYAxis,
  2035. title: {
  2036. text: item.Unit,
  2037. textCh:item.Unit,// 中文
  2038. textEn:item.UnitEn||item.Unit,
  2039. align: 'high',
  2040. rotation: 0,
  2041. y: -12,
  2042. reserveSpace: false,
  2043. style:{
  2044. ...chartTheme&&chartTheme.yAxisOptions.style
  2045. },
  2046. },
  2047. labels: {
  2048. style:{
  2049. ...chartTheme&&chartTheme.yAxisOptions.style
  2050. },
  2051. },
  2052. opposite: item.IsAxis===1?false:true,
  2053. min: index===0? Number(LeftMinValue):Number(RightMinValue),
  2054. max: index===0? Number(LeftMaxValue):Number(RightMaxValue),
  2055. tickWidth: 1,
  2056. }
  2057. let series_item = {
  2058. data: item.Value.map(_ =>[_.X,_.Y]),
  2059. dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
  2060. type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
  2061. yAxis: index,
  2062. name: item.Name,
  2063. nameCh: item.Name,
  2064. nameEn: item.NameEn||item.Name,
  2065. color: item.Color,
  2066. lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth)||3,
  2067. chartType: 'linear',
  2068. zIndex:1
  2069. }
  2070. series.push(series_item);
  2071. yAxis.push(y_item)
  2072. })
  2073. let tooltip = {
  2074. formatter: function() {
  2075. let xList = DataList[0].Value.map(_ =>_.X);
  2076. let step = xList[1]-xList[0];
  2077. let data_interval = `[${this.x},${this.x+step}]`;
  2078. let str=`<b>${ data_interval }</b>`;
  2079. this.points.forEach(item => {
  2080. str += `<br><span style="color:${item.color}">\u25CF</span>${item.series.name}: ${item.y}%<br>`
  2081. })
  2082. return str
  2083. },
  2084. shared: true
  2085. }
  2086. this.options = {
  2087. title: {
  2088. text:''
  2089. },
  2090. tooltip,
  2091. series,
  2092. yAxis,
  2093. xAxis
  2094. }
  2095. this.currentLang=='en' && this.changeOptions()
  2096. },
  2097. /* 跨品种分析 */
  2098. setCrossVarietyChart() {
  2099. this.leftIndex = -1;
  2100. this.rightIndex = -1;
  2101. this.rightTwoIndex = -1;
  2102. /* 主题样式*/
  2103. const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
  2104. const { min,max,x_min,x_max } = this.chartLimit;
  2105. const { DataList,XName,YName,XNameEn,YNameEn } = this.crossVarietyChartData;
  2106. //y轴
  2107. let yAxis = {
  2108. ...basicYAxis,
  2109. title: {
  2110. text: YName,
  2111. textCh:YName,// 中文
  2112. textEn:YNameEn||YName,
  2113. align: 'middle',
  2114. style: {
  2115. ...chartTheme&&chartTheme.xAxisOptions.style
  2116. }
  2117. },
  2118. labels: {
  2119. style: {
  2120. ...chartTheme&&chartTheme.xAxisOptions.style
  2121. }
  2122. },
  2123. opposite: false,
  2124. reversed: false,
  2125. min: Number(min),
  2126. max: Number(max),
  2127. tickWidth: 1,
  2128. }
  2129. // x轴
  2130. let xAxis = {
  2131. ...scatterXAxis,
  2132. title: {
  2133. text: XName,
  2134. textCh:XName,// 中文
  2135. textEn:XNameEn || XName,
  2136. align: 'middle',
  2137. style: {
  2138. ...chartTheme&&chartTheme.xAxisOptions.style
  2139. }
  2140. },
  2141. labels: {
  2142. style: {
  2143. ...chartTheme&&chartTheme.xAxisOptions.style
  2144. }
  2145. },
  2146. min: Number(x_min),
  2147. max: Number(x_max),
  2148. }
  2149. //数据列
  2150. let series = [];
  2151. const tagMap = { //标签对应文字
  2152. 1: '最新',
  2153. 3: 'Fix'
  2154. }
  2155. DataList.forEach(item => {
  2156. //数据列
  2157. let series_item = {
  2158. data: [],
  2159. type: 'scatter',
  2160. name: item.Name,
  2161. nameCh: item.Name,
  2162. nameEn: item.NameEn||item.Name,
  2163. color: item.Color,
  2164. chartType: 'linear',
  2165. zIndex:1
  2166. }
  2167. item.CoordinatePointData.forEach(_ => {
  2168. series_item.data.push({
  2169. x: _.X,
  2170. y: _.Y,
  2171. dataLabels: {
  2172. enabled: _.ShowTips===1,
  2173. allowOverlap: true,
  2174. align: 'left',
  2175. format: tagMap[_.DateType] || `-${_.DaysAgo}D`,
  2176. }
  2177. })
  2178. })
  2179. series.push(series_item);
  2180. })
  2181. let edbInfoList = this.tableData;
  2182. let tooltip = {
  2183. formatter: function() {
  2184. let series_obj = DataList.find(_ => _.Name === this.series.name);
  2185. let ponit_obj = series_obj.CoordinatePointData.find(_ => _.X ===this.x && _.Y===this.y);
  2186. let xEdbInfo = edbInfoList.find(_ => _.EdbInfoId===ponit_obj.XEdbInfoId);
  2187. let yEdbInfo = edbInfoList.find(_ => _.EdbInfoId===ponit_obj.YEdbInfoId);
  2188. let str=`<b>${ this.series.name }</b>`;
  2189. str += `<br><span style="color:${this.color}">\u25CF</span>${xEdbInfo.EdbName}: ${this.x} ${ponit_obj.XDate}<br>`;
  2190. str += `<span style="color:${this.color}">\u25CF</span>${yEdbInfo.EdbName}: ${this.y} ${ponit_obj.YDate}`;
  2191. return str
  2192. },
  2193. formatterCh: function() {
  2194. let series_obj = DataList.find(_ => _.Name === this.series.name);
  2195. let ponit_obj = series_obj.CoordinatePointData.find(_ => _.X ===this.x && _.Y===this.y);
  2196. let xEdbInfo = edbInfoList.find(_ => _.EdbInfoId===ponit_obj.XEdbInfoId);
  2197. let yEdbInfo = edbInfoList.find(_ => _.EdbInfoId===ponit_obj.YEdbInfoId);
  2198. let str=`<b>${ this.series.name }</b>`;
  2199. str += `<br><span style="color:${this.color}">\u25CF</span>${xEdbInfo.EdbName}: ${this.x} ${ponit_obj.XDate}<br>`;
  2200. str += `<span style="color:${this.color}">\u25CF</span>${yEdbInfo.EdbName}: ${this.y} ${ponit_obj.YDate}`;
  2201. return str
  2202. },
  2203. formatterEn: function() {
  2204. let series_obj = DataList.find(_ => _.NameEn === this.series.name);
  2205. let ponit_obj = series_obj.CoordinatePointData.find(_ => _.X ===this.x && _.Y===this.y);
  2206. let xEdbInfo = edbInfoList.find(_ => _.EdbInfoId===ponit_obj.XEdbInfoId);
  2207. let yEdbInfo = edbInfoList.find(_ => _.EdbInfoId===ponit_obj.YEdbInfoId);
  2208. let str=`<b>${ this.series.name }</b>`;
  2209. str += `<br><span style="color:${this.color}">\u25CF</span>${xEdbInfo.EdbNameEn}: ${this.x} ${ponit_obj.XDate}<br>`;
  2210. str += `<span style="color:${this.color}">\u25CF</span>${yEdbInfo.EdbNameEn}: ${this.y} ${ponit_obj.YDate}`;
  2211. return str
  2212. }
  2213. }
  2214. this.options = {
  2215. title: {
  2216. text:''
  2217. },
  2218. series,
  2219. yAxis: [yAxis],
  2220. xAxis,
  2221. tooltip
  2222. }
  2223. this.currentLang=='en' && this.changeRelevanceLang()
  2224. },
  2225. /* 查询范围为1年内 x轴显示为月/日 否则默认年/月 */
  2226. xLabelDealHandle() {
  2227. const end =
  2228. this.year_select === 5
  2229. ? this.select_date[1]
  2230. : this.year_select === 6
  2231. ? new Date()
  2232. : '';
  2233. const year_differ = this.$moment(end).diff(
  2234. this.$moment(this.select_date[0]),
  2235. 'years',
  2236. true
  2237. );
  2238. // console.log(year_differ)
  2239. if ([5, 6].includes(this.year_select) && year_differ <= 1) {
  2240. return true
  2241. } else {
  2242. return false;
  2243. }
  2244. },
  2245. /* 拼接动态的指标名称小标签 */
  2246. concatDynamicTag({ IsAxis,IsOrder,EdbInfoType,LeadValue,LeadUnit },lang='zh') {
  2247. // IsAxis左轴1 右轴0 2右2轴
  2248. //IsOrder正序false 逆序true
  2249. //EdbInfoType是否是领先指标
  2250. // lang ch 中文 en 英文
  2251. const axisLabelMap = lang=='zh'?{
  2252. 0: '右轴',
  2253. 2: '右2轴'
  2254. }:{
  2255. 0: 'RHS',
  2256. 2: '2-RHS'
  2257. }
  2258. const orderLabelMap = lang=='zh'?{
  2259. 1: '逆序'
  2260. }:{
  2261. 1: 'REV'
  2262. }
  2263. const edbInfoMap = lang=='zh'?{
  2264. 0: '领先'
  2265. }:{
  2266. 0: 'Lead'
  2267. }
  2268. //英文领先单位转换
  2269. const leadUnit = lang==='zh' ? LeadUnit : this.leadUnitEnMap[LeadUnit];
  2270. let axis_tag = axisLabelMap[IsAxis] || '';
  2271. //逆序拼接
  2272. let order_tag = orderLabelMap[Number(IsOrder)] ? `${axis_tag ? ',': ''}${orderLabelMap[Number(IsOrder)]}` : ''
  2273. //领先拼接
  2274. let edb_tag = edbInfoMap[EdbInfoType] ? `${(axis_tag||order_tag) ? ',' : '' }${edbInfoMap[EdbInfoType]} ${LeadValue}${leadUnit}` : '';
  2275. return (axis_tag || order_tag || edb_tag) ? `(${axis_tag}${order_tag}${edb_tag})` : ''
  2276. },
  2277. /* 指标顺序调整 IsAxis: 0右轴 1左轴 2右2*/
  2278. changeEdbOrder(data) {
  2279. // 左轴指标
  2280. let left_edbs = data.filter(_ => _.IsAxis===1);
  2281. //右轴指标
  2282. let right_edbs = data.filter(_ => !_.IsAxis);
  2283. // 右2轴指标
  2284. let right_two_edbs = data.filter(_ => _.IsAxis === 2);
  2285. // 按 左 右 右2顺序排列
  2286. return [left_edbs,right_edbs,right_two_edbs].flat(Infinity);
  2287. },
  2288. /* 获取同侧极值 左右极值 */
  2289. getSameSideExtreme(newval) {
  2290. // // 左侧
  2291. let leftArr = newval.filter(item => item.IsAxis === 1);
  2292. let left_extreme_max = leftArr.length ? Math.max(...leftArr.map(item => Number(item.MaxData))) : undefined;
  2293. let left_extreme_min = leftArr.length ? Math.min(...leftArr.map(item => Number(item.MinData))) : undefined;
  2294. // // 右侧
  2295. let rightArr = newval.filter(item => !item.IsAxis);
  2296. let right_extreme_max = rightArr.length ? Math.max(...rightArr.map(item => Number(item.MaxData))) : undefined;
  2297. let right_extreme_min = rightArr.length ? Math.min(...rightArr.map(item => Number(item.MinData))) : undefined;
  2298. this.left_extreme = leftArr.length ? [left_extreme_max,left_extreme_min] : [];
  2299. this.right_extreme = rightArr.length ? [right_extreme_max,right_extreme_min] : [];
  2300. },
  2301. /* 用于复制指标 */
  2302. async copyCode({EdbInfoId, EdbInfoCategoryType}) {
  2303. let params = {
  2304. PageSize: 100000,
  2305. CurrentIndex: 1,
  2306. EdbInfoId: EdbInfoId,
  2307. }
  2308. const res = EdbInfoCategoryType === 1
  2309. ? await preDictEdbInterface.edbDataInfo(params)
  2310. : await dataBaseInterface.targetList(params);
  2311. if (res.Ret !== 200) return
  2312. let arr = res.Data.Item.DataList || [];
  2313. let str = '日期\t 值\n';
  2314. arr.forEach((item) => (str += `${item.DataTime}\t${item.Value}\n`));
  2315. this.$copyText(str).then(
  2316. (res) => {
  2317. this.$message.success(/* '已成功复制!' */this.$t('MsgPrompt.copy_success_msg'));
  2318. },
  2319. (err) => {
  2320. this.$message.error(/* '复制失败!' */this.$t('MsgPrompt.copy_fail_msg'));
  2321. }
  2322. );
  2323. },
  2324. /* 查看数据 跳转指标库展开具体指标 */
  2325. viewTarget({ UniqueCode,EdbInfoId,EdbInfoCategoryType,ClassifyId }) {
  2326. let path = EdbInfoCategoryType ? '/predictEdb' : '/database';
  2327. let {href} = this.$router.resolve({path, query: {
  2328. code: UniqueCode,
  2329. id: EdbInfoId,
  2330. classifyId:ClassifyId
  2331. }});
  2332. window.open(href,'_blank');
  2333. },
  2334. /* 领先指标 过滤负数 小数点*/
  2335. filterCode(item) {
  2336. item.LeadValue=item.LeadValue.replace(/[^\.\d]/g,'').replace('.','');
  2337. },
  2338. /* 线条粗细 过滤最小1 */
  2339. filterWidth(item) {
  2340. item.ChartWidth = Number(item.ChartWidth) >= 1 ? Number(item.ChartWidth) : 1;
  2341. },
  2342. /* 图标复制的 blob方法*/
  2343. copyBlobItem(widthNum,heightNum,svg,genre){
  2344. let bas64 = this.svgToBase64(svg)
  2345. const dynamic_condition = {
  2346. bigScreen:{
  2347. x:1.22,
  2348. zoomY:0.8,
  2349. zoomX:0.8,
  2350. },
  2351. smallScreen:{
  2352. x:1,
  2353. zoomY:1,
  2354. zoomX:1,
  2355. }
  2356. }
  2357. const {x,zoomY,zoomX} = dynamic_condition.smallScreen;
  2358. // const {x,zoomY,zoomX} = document.body.clientWidth > 1500 && genre === '微信' ? dynamic_condition.bigScreen : dynamic_condition.smallScreen;
  2359. const canvas = document.createElement("canvas");
  2360. const ctx = canvas.getContext("2d");
  2361. const img = new Image();
  2362. canvas.width = widthNum / x;
  2363. canvas.height = heightNum / x;
  2364. img.crossOrigin = "Anonymous";
  2365. img.src = bas64;
  2366. img.onload = () => {
  2367. ctx.clearRect(0, 0, ctx.canvas.width, ctx.canvas.height);
  2368. if(genre === '微信'){
  2369. ctx.fillStyle="#fff";
  2370. ctx.fillRect(0, 0, ctx.canvas.width, ctx.canvas.height);
  2371. }
  2372. ctx.scale(zoomX,zoomY)
  2373. ctx.drawImage(img, 0, 0); // 将canvas转为blob
  2374. if(window.ClipboardItem) {
  2375. canvas.toBlob(async (blob) => {
  2376. const data = [new ClipboardItem({ [blob.type]: blob })];
  2377. await navigator.clipboard.write(data).then(
  2378. () => {
  2379. // this.$message.success('复制成功!')
  2380. this.$message.success(this.$t('MsgPrompt.copy_success_msg'))
  2381. },
  2382. (error) => {
  2383. console.log('复制失败,稍后再试',error);
  2384. // this.$message.warning('复制失败,稍后再试')
  2385. this.$message.warning(this.$t('MsgPrompt.copy_fail_msg'))
  2386. }
  2387. );
  2388. });
  2389. }else {
  2390. // this.$message.warning('当前协议暂不支持,仅支持https协议')
  2391. this.$message.warning(this.$t('MsgPrompt.http_not_support'))
  2392. }
  2393. };
  2394. },
  2395. /* 图标复制的 区分微信 office 来动态的计算*/
  2396. dynamicWidthAndHeight(type, ChartType, ChartName, tableData) {
  2397. //动态计算高度 宽度
  2398. let heightNum = this.sameOptionType.includes(ChartType) ? 352 + parseInt((tableData - 1) / 2) * 20 : 352;
  2399. let widthNum = this.sameOptionType.includes(ChartType) ? parseFloat(heightNum * (591 / 352)) : 591;
  2400. widthNum = type === "微信" ? 1260 : widthNum;
  2401. heightNum = type === "微信" ? 669 : heightNum;
  2402. //动态 标题
  2403. let count = parseInt(widthNum / 21);
  2404. let reg = new RegExp("(.{" + count + "})", "g");
  2405. let newTitle = ''
  2406. if(this.currentLang == 'zh'){
  2407. newTitle = _.cloneDeep(ChartName).replace(/\s/g, "").replace(reg, "$1<br/>");
  2408. }else{
  2409. newTitle = _.cloneDeep(ChartName);
  2410. }
  2411. //动态配置
  2412. let dynamic_copyOptions = {
  2413. legend: {
  2414. labelFormatter: function () {
  2415. let count = Math.floor(widthNum/2/15);//多少字数换行;
  2416. let reg2 = new RegExp("(.{" + count + "})", "g");
  2417. let newName = this.name.replace(/-/g, "—").replace(/\s/g, "").replace(/(/g, "(").replace(/)/g, ")").replace(reg2, "$1<br/>");
  2418. return newName;
  2419. },
  2420. itemWidth: widthNum > 1000 ? 530 : 235,
  2421. itemStyle: {
  2422. fontSize: 10,
  2423. color: this.chartInfo.ChartThemeStyle&&JSON.parse(this.chartInfo.ChartThemeStyle).legendOptions.itemStyle.color,
  2424. textOverflow:undefined
  2425. },
  2426. },
  2427. seasonLegend: {},
  2428. };
  2429. return {
  2430. heightNum,
  2431. widthNum,
  2432. newTitle,
  2433. dynamic_copyOptions,
  2434. };
  2435. },
  2436. /* 控制y轴配置项是否显示 其他类型图|堆叠图第一个显示|组合图第一个堆叠的样式显示 */
  2437. showYOptionsHandle(item,index) {
  2438. let isShowOptions = true;
  2439. // 图表类型不是堆叠或组合 y轴配置显示
  2440. // 堆叠图类型第一个指标的y轴配置显示 其余不显示
  2441. // 组合图类型 组合样式不是堆叠 显示
  2442. //组合图类型 组合样式是堆叠 第一个堆叠的y轴配置显示 其余相同样式堆叠不显示
  2443. if([3,4].includes(this.chartInfo.ChartType) && index !== 0) {
  2444. isShowOptions = false
  2445. } else if(this.chartInfo.ChartType ===6
  2446. && ['areaspline','column'].includes(item.ChartStyle)
  2447. && index !== this.tableData.findIndex(_ => _.ChartStyle === item.ChartStyle)) {
  2448. isShowOptions = false
  2449. }
  2450. return isShowOptions;
  2451. },
  2452. /* 图表另存为 */
  2453. saveChartOtherHandle() {
  2454. this.isShowSaveOther = true;
  2455. },
  2456. /* 预测配置 分区 */
  2457. getPredictParams({LatestDate,MoveLatestDate,PredictChartColor,ChartStyle},chartStyle='') {
  2458. return {
  2459. zoneAxis: 'x',
  2460. zones: [{
  2461. value: new Date(MoveLatestDate||LatestDate).getTime()+1
  2462. }, {
  2463. dashStyle: 'ShortDot',
  2464. color:(ChartStyle==='column' || chartStyle==='column') ? 'transparent' : PredictChartColor
  2465. }]
  2466. }
  2467. },
  2468. /* 季节图预测数据 年份=分割点年份做分割 年份>分割点年份全为预测 */
  2469. getSeasonPredictParams(timestamp) {
  2470. return timestamp
  2471. ? {
  2472. zoneAxis: 'x',
  2473. zones: [{
  2474. value: new Date(timestamp).getTime()+1
  2475. }, {
  2476. dashStyle: 'ShortDot',
  2477. }]
  2478. }
  2479. : {}
  2480. },
  2481. /* 转base64 */
  2482. svgToBase64(svg) {
  2483. const base64img = `data:image/svg+xml;base64,${window.btoa(
  2484. unescape(encodeURI(svg))
  2485. )}`;
  2486. // console.log(base64img)
  2487. return base64img;
  2488. },
  2489. /* 改变上下限时重绘图 依赖于chartLimit的柱 商品价格 散点截面 */
  2490. changeLimit() {
  2491. //source1 eta图库的类型对应
  2492. const typeMap = {
  2493. 1: this.setDefaultChart,
  2494. 2: this.setSeasonChart,
  2495. 3: this.setStackOrCombinChart,
  2496. 4: this.setStackOrCombinChart,
  2497. 5: this.setScatterChart,
  2498. 6: this.setStackOrCombinChart,
  2499. 7: this.setBarChart,
  2500. 10: this.setSectionScatterChart,
  2501. 11: this.setRadarChart
  2502. }
  2503. //其他source
  2504. const sourceMap = {
  2505. 2: this.setCommodityChart,
  2506. 10: this.setCrossVarietyChart
  2507. // 3:
  2508. }
  2509. if(this.chartInfo.Source === 1) {
  2510. if([7,10,11].includes(this.chartInfo.ChartType)){
  2511. typeMap[this.chartInfo.ChartType]()
  2512. }else{
  2513. this.setAddChartDefault&&this.setAddChartDefault();
  2514. typeMap[this.chartInfo.ChartType](this.tableData)
  2515. }
  2516. }else {
  2517. sourceMap[this.chartInfo.Source]();
  2518. }
  2519. },
  2520. // 设置 起始日期 和 最新日期
  2521. setExtremumDate(){
  2522. //过滤、排序 拿到起始日期
  2523. let startDateList = this.tableData.map(it => it.StartDate)
  2524. .filter(Boolean)
  2525. .sort((a,b)=> new Date(a).getTime() - new Date(b).getTime())
  2526. this.earliestDate = startDateList[0]
  2527. //过滤、排序 拿到最新日期
  2528. let endDateList = this.tableData.map(it => it.LatestDate)
  2529. .filter(Boolean)
  2530. .sort((a,b)=> new Date(b).getTime() - new Date(a).getTime())
  2531. this.latestDate = endDateList[0]
  2532. console.log(this.earliestDate,this.latestDate,'this.latestDate');
  2533. },
  2534. /* 处理轴的标识线结构 在指定轴位置上拼接标识线
  2535. 0:右轴 1:左轴 2:右2轴 x轴固定3
  2536. axisType表示x轴类型 处理时间轴的值 datetime/null
  2537. */
  2538. setAxisPlotLines(axis,axisType=null) {
  2539. const { MarkersLines,ChartType } = this.chartInfo;
  2540. if(!MarkersLines) return []
  2541. let markerLines = JSON.parse(MarkersLines);
  2542. let arr = markerLines.filter(_ => _.isShow&&_.axis===axis)
  2543. let plotLines = arr.map(_ => {
  2544. //是否是x时间轴
  2545. let isXDateAxis = axis===3&&axisType==='datetime';
  2546. let markerValue='';
  2547. if(isXDateAxis) {
  2548. //季节图x轴额外拼个年份
  2549. let nowYear = ChartType===2 ? new Date(this.tableData[0].DataList[1].DataList
  2550. [0].DataTimestamp).getFullYear() : '';
  2551. console.log(nowYear)
  2552. markerValue = ChartType===2
  2553. ? new Date(`${nowYear}-${_.value}`).getTime()
  2554. : new Date(_.value).getTime()
  2555. }else {
  2556. markerValue = Number(_.value)
  2557. }
  2558. return {
  2559. value: markerValue,
  2560. dashStyle: _.dashStyle,
  2561. width: Number(_.lineWidth),
  2562. color: _.color,
  2563. label: {
  2564. text: _.text||'',
  2565. verticalAlign: _.textPosition,
  2566. style: {
  2567. color: _.textColor,
  2568. fontSize: _.textFontSize
  2569. }
  2570. }
  2571. }
  2572. })
  2573. return plotLines
  2574. },
  2575. /* 处理标识区拼接 axisType表示x轴类型处理时间轴的值 datetime/null */
  2576. setAxisPlotAreas(axis,axisType=null) {
  2577. const { MarkersAreas,ChartType } = this.chartInfo;
  2578. if(!MarkersAreas) return []
  2579. let markerAreas = JSON.parse(MarkersAreas);
  2580. let arr = markerAreas.filter(_ => _.isShow&&_.axis===axis)
  2581. let plotBands = arr.map(_ => {
  2582. //是否是x时间轴
  2583. let isXDateAxis = axis===3&&axisType==='datetime';
  2584. let fromMarkerValue='',toMarkerValue='';
  2585. if(isXDateAxis) {
  2586. //季节图x轴额外拼个年份
  2587. let nowYear = ChartType===2 ? new Date(this.tableData[0].DataList[1].DataList
  2588. [0].DataTimestamp).getFullYear() : '';
  2589. fromMarkerValue = ChartType===2
  2590. ? new Date(`${nowYear}-${_.fromValue}`).getTime()
  2591. : new Date(_.fromValue).getTime()
  2592. toMarkerValue = ChartType===2
  2593. ? new Date(`${nowYear}-${_.toValue}`).getTime()
  2594. : new Date(_.toValue).getTime()
  2595. }else {
  2596. fromMarkerValue = Number(_.fromValue);
  2597. toMarkerValue = Number(_.toValue);
  2598. }
  2599. //默认label有些偏移 重新归正下
  2600. let positionMapValue = {
  2601. 'top': 12,
  2602. 'middle': 0,
  2603. 'bottom': -10
  2604. }
  2605. return {
  2606. from: fromMarkerValue,
  2607. to: toMarkerValue,
  2608. color: _.color,
  2609. label: {
  2610. text: _.text||'',
  2611. verticalAlign: _.textPosition,
  2612. style: {
  2613. color: _.textColor,
  2614. fontSize: _.textFontSize
  2615. },
  2616. y: positionMapValue[_.textPosition]
  2617. }
  2618. }
  2619. })
  2620. return plotBands
  2621. },
  2622. /* 历史图表默认显示图表来源 d毛后端不修复只能自己每次详情处理下*/
  2623. setDefaultSourceFrom() {
  2624. if(!this.chartInfo.SourcesFrom) {
  2625. let themeOpt = JSON.parse(this.chartInfo.ChartThemeStyle);
  2626. this.chartInfo.SourcesFrom = JSON.stringify({
  2627. isShow: true,
  2628. text: this.chartInfo.ChartSource,
  2629. color: themeOpt&&themeOpt.markerOptions.style.color,
  2630. fontSize: themeOpt&&themeOpt.markerOptions.style.fontSize
  2631. });
  2632. }
  2633. },
  2634. /* ----自定义上下限相关--- */
  2635. /* 计算y轴上下限 */
  2636. calcYAxislimit(tableData=[]) {
  2637. //散点图单独处理
  2638. if(this.chartInfo.ChartType===5){
  2639. if(tableData[1]){
  2640. this.chartLimit.min = tableData[1].MinData
  2641. this.chartLimit.max = tableData[1].MaxData
  2642. }
  2643. return
  2644. }
  2645. //分组
  2646. const leftData = tableData.filter(i => i.IsAxis === 1).map(i => [Number(i.MinData), Number(i.MaxData)])
  2647. const rightData = tableData.filter(i => !i.IsAxis).map(i => [Number(i.MinData), Number(i.MaxData)])
  2648. const rightTwoData = tableData.filter(i => i.IsAxis === 2).map(i => [Number(i.MinData), Number(i.MaxData)])
  2649. //计算最大最小值
  2650. if (leftData.length) {
  2651. const {
  2652. Max,
  2653. Min
  2654. } = this.calcLimit(leftData.flat())
  2655. this.chartLimit.min=Min
  2656. this.chartLimit.max=Max
  2657. } else {
  2658. this.leftIndex = -1
  2659. this.chartLimit.min=0
  2660. this.chartLimit.max=0
  2661. }
  2662. if (rightData.length) {
  2663. const {
  2664. Max,
  2665. Min
  2666. } = this.calcLimit(rightData.flat())
  2667. this.chartLimit.rightMin = Min
  2668. this.chartLimit.rightMax = Max
  2669. } else {
  2670. this.rightIndex = -1
  2671. this.chartLimit.rightMin = 0
  2672. this.chartLimit.rightMax = 0
  2673. }
  2674. if (rightTwoData.length) {
  2675. const {
  2676. Max,
  2677. Min
  2678. } = this.calcLimit(rightTwoData.flat())
  2679. this.chartLimit.rightTwoMin = Min
  2680. this.chartLimit.rightTwoMax = Max
  2681. } else {
  2682. this.rightTwoIndex = -1
  2683. this.chartLimit.rightTwoMin = 0
  2684. this.chartLimit.rightTwoMax = 0
  2685. }
  2686. console.table([{
  2687. 'y轴': '左轴',
  2688. '最大值': this.chartLimit.max,
  2689. '最小值': this.chartLimit.min
  2690. },
  2691. {
  2692. 'y轴': '右轴',
  2693. '最大值': this.chartLimit.rightMax,
  2694. '最小值': this.chartLimit.rightMin
  2695. },
  2696. {
  2697. 'y轴': '右二轴',
  2698. '最大值': this.chartLimit.rightTwoMax,
  2699. '最小值': this.chartLimit.rightTwoMin
  2700. }
  2701. ])
  2702. },
  2703. calcLimit(arr) {
  2704. return {
  2705. Max: Math.max(...arr),
  2706. Min: Math.min(...arr)
  2707. }
  2708. },
  2709. //图表详情-设置图表上下限
  2710. setLimitData(tableData=[]){
  2711. if([7,10,11].includes(this.chartInfo.ChartType)) return
  2712. const {
  2713. //左右轴极值字段
  2714. LeftMin=0,LeftMax=0,
  2715. RightMin=0,RightMax=0,
  2716. Right2Min=0,Right2Max=0,
  2717. MinMaxSave
  2718. } = this.chartInfo
  2719. if(MinMaxSave){
  2720. this.chartLimit.min = Number(LeftMin)
  2721. this.chartLimit.max = Number(LeftMax)
  2722. this.chartLimit.rightMin = Number(RightMin)
  2723. this.chartLimit.rightMax = Number(RightMax)
  2724. this.chartLimit.rightTwoMin = Number(Right2Min)
  2725. this.chartLimit.rightTwoMax = Number(Right2Max)
  2726. //若用户修改过,则检测轴的上下限是否为空,若为空,则需要计算对应轴的上下限
  2727. this.checkChartLimit(tableData)
  2728. }else{
  2729. this.calcYAxislimit(tableData)
  2730. }
  2731. },
  2732. checkChartLimit(tableData=[]){
  2733. //散点图单独处理
  2734. if(this.chartInfo.ChartType===5){
  2735. if(tableData[1]){
  2736. const {min,max} = this.chartLimit
  2737. if(Number(min)===0&&Number(max)===0){
  2738. this.chartLimit.min = tableData[1].MinData
  2739. this.chartLimit.max = tableData[1].MaxData
  2740. }
  2741. }
  2742. return
  2743. }
  2744. const {
  2745. min,max,rightMin,rightMax,rightTwoMin,rightTwoMax
  2746. } = this.chartLimit
  2747. //若轴的上下限均为0,则不管用户有没有修改过,都重新赋值
  2748. if(Number(min)===0&&Number(max)===0){
  2749. const leftData = tableData.filter(i=>i.IsAxis===1).map(i=>[Number(i.MinData),Number(i.MaxData)])
  2750. if(leftData.length){
  2751. const {Max,Min} = this.calcLimit(leftData.flat())
  2752. this.chartLimit.min=Min
  2753. this.chartLimit.max=Max
  2754. }
  2755. }
  2756. if(Number(rightMin)===0&&Number(rightMax)===0){
  2757. const rightData = tableData.filter(i => !i.IsAxis).map(i=>[Number(i.MinData),Number(i.MaxData)])
  2758. if(rightData.length){
  2759. const {Max,Min} = this.calcLimit(rightData.flat())
  2760. this.chartLimit.rightMin = Min
  2761. this.chartLimit.rightMax = Max
  2762. }
  2763. }
  2764. if(Number(rightTwoMin)===0&&Number(rightTwoMax)===0){
  2765. const rightTwoData = tableData.filter(i=>i.IsAxis===2).map(i=>[Number(i.MinData),Number(i.MaxData)])
  2766. if(rightTwoData.length){
  2767. const {Max,Min} = this.calcLimit(rightTwoData.flat())
  2768. this.chartLimit.rightTwoMin = Min
  2769. this.chartLimit.rightTwoMax = Max
  2770. }
  2771. }
  2772. },
  2773. /*-------------------- */
  2774. }
  2775. }