import {
defaultOpts,
seasonOptions,
yearSelector,
} from '@/utils/defaultOptions';
import Highcharts from 'highcharts';
import { dataBaseInterface } from '@/api/api';
import * as preDictEdbInterface from '@/api/modules/predictEdbApi.js';
import futuresInterface from '@/api/modules/futuresBaseApi';
import chartRelevanceApi from '@/api/modules/chartRelevanceApi';
import { fittingEquationInterface,statisticFeatureInterface,crossVarietyInterface } from '@/api/modules/chartRelevanceApi';
/* 散点x轴 */
const scatterXAxis = {
tickPosition: 'inside',
lineColor: '#bfbfbf',
tickColor: '#bfbfbf',
tickLength:5,
ordinal: false,
type: 'linear'
}
/* 曲线基础y轴静态配置 */
const basicYAxis = {
tickLength: 5,
lineWidth: 1,
lineColor: '#bfbfbf',
tickColor: '#bfbfbf',
// offset: 0,
visible: true,
gridLineWidth: 0,
tickPosition: 'inside',
endOnTick: false,
startOnTick: false,
showLastLabel: true,
tickPixelInterval: 50,
}
/**备注一下 越多越乱
* @params
* Source 1 ; chartType 1曲线 2季节 3面积 4堆积柱 5散点 6组合 7柱形 10截面散点 11雷达图
* 2 商品价格
* 3 相关性
* 4 滚动相关性
* 5 商品利润
* 6 拟合方程
* 7 统计特征/标准差
* 8 统计特征/百分位
* 9 统计特征/频率
* 10 跨品种分析
*/
export const chartSetMixin = {
data() {
return {
/**
* 默认区间15年至今 值等于 5、6、20 是自定义时间段
* 20代表最近几年 季节性图默认 最近5年
* 6代表至今 只需要选择开始日期
* 5代表范围 需要选择开始日期和结束日期
*/
year_select: yearSelector[0].value,
sameOptionType:[1,3,4,5,6],// 头部一样 配置一样的图表类型 曲线 散点 柱状 面积 组合
// yearSelector, //年份按钮组
select_date: '', //自定义时间段
count_year: 0, //最近年数
isDateDia: false, // 时间段弹窗
earliestDate:'', // 最早的日期 - 起始日期
latestDate:'', //最晚日期 - 最新日期
dateForm: {},
dateTip: /* '请选择时间段' */ this.$t('Chart.choose_time'),
predefineColors: defaultOpts.colors.slice(0, 2), //定义颜色蓝,红 默认颜色
fre_options: ['年', '季', '月', '周', '天'], //领先指标频度配置
//领先频度对应英文
leadUnitEnMap: {
'年': 'Y',
'季': 'Q',
'月': 'M',
'周': 'W',
'天': 'D',
},
tableData: [],//表格指标数据
options: {}, //配置options
leftIndex: -1, //左侧上下限对应下标
rightIndex: -1, //右侧上下限对应下标
rightTwoIndex: -1,//右2上下限对应下标
left_extreme: [],
right_extreme: [],
chartItemStyleArr:[
{ label: '曲线图', key: 1 ,value: 'spline'},
{ label: '折线图', key: 2 ,value: 'line'},
// { label: '堆积面积图', key: 3 ,value: 'areaspline'},
{ label: '堆积柱状图', key: 4 ,value: 'column'},
],//组合图可选样式
isSetExtremeValue: false,//控制添加散点图时是否重置上下限
isShowSaveOther: false,//图表另存为弹窗
// -------------切换图表中英文
// currentLang:'zh', // ch(中文) en(英文)
setEnName:false,
// 传入的formItem所需内容
formItemArray:[],
/* 奇怪柱形图 */
barDateList: [],//柱形图的绘图数据
barXData: [],//柱形图的x轴
barXIdList: [],//x轴id数组 用于切换英文遍历用
barEdbData: [],//柱形图的表格数据 只用于取值
chartLimit: {
min:'', //左轴上下限
max:'',
rightMin:'',//右轴上下限
rightMax:'',
rightTwoMin:'',//右二轴上下限
rightTwoMax:'',
},
/* 商品价格曲线 本来逻辑分开写 但需求又得加入myeta和eta图库共存 还是写一块算了 */
commodityChartData: [],
commodityXData: [],
commodityEdbList: [],
/* 时间截面散点图 */
sectionScatterData: {},
/* 相关性图表 */
relevanceChartData:null,
relevanceUnitEnMap:{
'年': 'Year',
'季': 'Season',
'月': 'Month',
'周': 'Week',
'天': 'Day',
},
/* 统计频率图 */
statisticFrequencyData: {},
// 季节性图和额外配置 这里为了预览
SeasonExtraConfig:{
ChartLegend:[], // 图例名称数组
XStartDate:"01-01", // 横坐标显示范围 - 开始
XEndDate:"12-31", // 横坐标显示范围-结束
JumpYear:0, //是否跨年
},
/* 跨品种分析图 */
crossVarietyChartData: {},
/* 雷达图 */
radarChartData: {},
/* 修改对应版本信息弹窗 替换原有设置英文*/
isLangInfoDia: false
}
},
computed:{
tableColums(){
return [
{
label: this.$t('Edb.Detail.e_name'),
key: 'EdbName',
enKey:'EdbNameEn',
inputTip:'点击输入英文指标名称',
minwidthsty: '150px',
},
{
label: this.$t('Edb.Detail.e_id'),
key: 'EdbCode',
widthsty: '120px',
},
{
label: this.$t('Edb.Detail.e_fre'),
key: 'Frequency',
enKey:'FrequencyEn',
minwidthsty: '60px',
},
{
label: this.$t('Edb.Detail.e_unit'),
key: 'Unit',
enKey:'UnitEn',
inputTip:'英文单位',
minwidthsty: '50px',
},
{
label: this.$t('Edb.Detail.e_start_time'),
key: 'StartDate',
minwidthsty: '100px',
},
{
label: this.$t('Edb.Detail.e_latest_date'),
key: 'LatestDate',
minwidthsty: '90px',
},
{
label: this.$t('Edb.Detail.e_latest_value'),
key: 'LatestValue',
minwidthsty: '90px',
},
{
label: this.$t('Edb.Detail.e_recent_time'),
key: 'ModifyTime',
minwidthsty: '100px',
},
{
label: this.$t('Edb.Detail.source'),
key: 'SourceName',
},
]
},
yearSelector() {
return yearSelector
},
currentLang() {
return this.$store.state.lang
}
},
watch: {
/* 奇怪柱状图数据 */
barDateList: {
handler(newval) {
console.log('bar')
newval.length && this.barXData.length && this.setBarChart();
},
deep: true
},
sectionScatterData: {
handler(newval) {
newval.DataList && this.setSectionScatterChart();
},
deep: true
},
currentLang() {
this.changeLanguage()
}
},
methods: {
/* --------------------------------------------切换中英文 */
async changeLanguage(){
this.search_txt = '';
if(!this.$refs.chartRef) return
//不同图表来源 1eta图 2商品价格图 3相关性图表
const sourceMap = {
1: this.changeOptions,
2: this.changeCommodityLang,
3: this.changeRelevanceLang,
4: this.changeRelevanceLang,
5: this.changeCommodityLang
}
sourceMap[this.chartInfo.Source]&&sourceMap[this.chartInfo.Source]()
},
// // 打开设置英文信息弹窗
// async openEnNameDia(){
// this.formItemArray={}
// this.formItemArray.chartInfo=[]
// this.formItemArray.chartsList=[]
// this.formItemArray.chartInfo.push({
// label:/* '图表名称' */this.$t('Chart.Detail.chart_name'),
// value:this.chartInfo.ChartName,
// key:'ChartName',
// id:this.chartInfo.ChartInfoId,
// source: this.chartInfo.Source,
// notEdit:true
// },
// {
// label:/* '英文图表名称' */this.$t('Chart.Detail.chart_en_name'),
// value:this.chartInfo.ChartNameEn,
// key:'ChartNameEn',
// id:this.chartInfo.ChartInfoId,
// placeholder:/* '请输入英文图表名称' */this.$t('Chart.InputHolderAll.input_en_name')
// })
// if([1,2,5].includes(this.chartInfo.Source)){ //需要设置指标的
// this.tableData.map(item =>{
// if(item.Unit){
// this.formItemArray.chartsList.push([
// {
// label:/* '指标名称' */this.$t('Edb.Detail.e_name'),
// value:item.EdbName,
// key:'EdbName',
// id:item.EdbInfoId,
// notEdit:true
// },
// {
// label:/* '单位' */this.$t('Edb.Detail.e_unit'),
// value:item.Unit,
// key:'Unit',
// id:item.EdbInfoId,
// notEdit:true
// },
// {
// label:/* '英文指标名称' */this.$t('Edb.Detail.e_en_name'),
// value:item.EdbNameEn,
// key:'EdbNameEn',
// id:item.EdbInfoId,
// placeholder:/* '请输入英文指标名称' */ this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_en_name')})
// },
// {
// label:/* '英文单位' */this.$t('Edb.Detail.e_en_unit'),
// value:item.UnitEn,
// key:'UnitEn',
// id:item.EdbInfoId,
// placeholder:/* '请输入英文单位' */this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_en_unit')})
// }
// ])
// }else{
// this.formItemArray.chartsList.push([
// {
// label:/* '指标名称' */this.$t('Edb.Detail.e_name'),
// value:item.EdbName,
// key:'EdbName',
// id:item.EdbInfoId,
// notEdit:true
// },
// {
// label:/* '英文指标名称' */this.$t('Edb.Detail.e_en_name'),
// value:item.EdbNameEn,
// key:'EdbNameEn',
// id:item.EdbInfoId,
// placeholder:this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_en_name')})
// }
// ])
// }
// })
// }
// //价格曲线
// if(this.chartInfo.Source===2) {
// this.formItemArray.chartInfo.push({
// label:/* '期货名称' */this.$t('Chart.Detail.good_name'),
// value:this.commodityEdbList[1].EdbName,
// key:'FutureGoodName',
// id:this.chartInfo.ChartInfoId,
// notEdit:true
// },
// {
// label:/* '英文期货名称' */this.$t('Chart.Detail.good_en_name'),
// value:this.commodityEdbList[1].EdbNameEn,
// key:'FutureGoodNameEn',
// id:this.chartInfo.ChartInfoId,
// placeholder:/* '请输入英文期货名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.good_en_name')})
// })
// }
// //利润曲线
// else if(this.chartInfo.Source===5) {
// this.formItemArray.chartInfo.push({
// label:/* '盘面利润名称' */this.$t('Chart.Detail.profit_name'),
// value:this.chartInfo.ProfitName,
// key:'ProfitName',
// id:this.chartInfo.ChartInfoId,
// notEdit:true
// },
// {
// label:/* '英文盘面利润名称' */this.$t('Chart.Detail.profit_en_name'),
// value:this.chartInfo.ProfitNameEn,
// key:'ProfitNameEn',
// id:this.chartInfo.ChartInfoId,
// placeholder:/* '请输入英文盘面利润名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.profit_en_name')})
// })
// }
// //跨品种分析
// else if(this.chartInfo.Source===10) {
// let { Data } = await crossVarietyInterface.chartLangOption({ChartInfoId: this.chartInfo.ChartInfoId})
// const { TagList,VarietyList } = Data;
// this.formItemArray.chartInfo.push({
// label:/* 'X轴名称' */this.$t('Chart.Detail.x_name'),
// value:this.crossVarietyChartData.XName,
// key:'XName',
// id:TagList[0].ChartTagId,
// notEdit:true
// },
// {
// label:/* '英文X轴名称' */this.$t('Chart.Detail.x_en_name'),
// value:this.crossVarietyChartData.XNameEn,
// key:'XNameEn',
// id:TagList[0].ChartTagId,
// placeholder:/* '请输入英文X轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.x_en_name')})
// },{
// label:/* 'Y轴名称' */this.$t('Chart.Detail.y_name'),
// value:this.crossVarietyChartData.YName,
// key:'YName',
// id:TagList[1].ChartTagId,
// notEdit:true
// },
// {
// label:/* '英文Y轴名称' */this.$t('Chart.Detail.y_en_name'),
// value:this.crossVarietyChartData.YNameEn,
// key:'YNameEn',
// id:TagList[1].ChartTagId,
// placeholder:/* '请输入英文Y轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.y_en_name')})
// })
// VarietyList.forEach(item => {
// this.formItemArray.chartsList.push([
// {
// label:/* '品种名称' */this.$t('Chart.Detail.variety_name'),
// value:item.ChartVarietyName,
// key:'ChartVarietyName',
// id:item.ChartVarietyId,
// notEdit:true
// },
// {
// label:/* '英文品种名称' */this.$t('Chart.Detail.variety_en_name'),
// value:item.ChartVarietyNameEn,
// key:'ChartVarietyNameEn',
// id:item.ChartVarietyId,
// placeholder:/* '请输入英文品种名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.variety_en_name')})
// }
// ])
// })
// }
// this.setEnName = true
// },
// // 更新英文信息
// async updateEnName(enNameData){
// // console.log(enNameData)
// let res=null
// if([2,5].includes(this.chartInfo.Source)){//商品价格
// res=await futuresInterface.editChartEn({
// ChartInfoId: enNameData.ChartInfoId,
// ChartNameEn: enNameData.ChartNameEn,
// UnitEn: enNameData.ChartEdbInfoList[0].UnitEn || '',
// EdbNameEn: enNameData.ChartEdbInfoList[0].EdbNameEn || '',
// FutureGoodNameEn: enNameData.FutureGoodNameEn || '',
// ProfitNameEn: enNameData.ProfitNameEn || ''
// })
// }else if(this.chartInfo.Source===3){//相关性
// res=await chartRelevanceApi.editChartEn({
// ChartInfoId: enNameData.ChartInfoId,
// ChartNameEn: enNameData.ChartNameEn
// })
// }else if(this.chartInfo.Source===6){//拟合方程
// res=await fittingEquationInterface.editChartEn({
// ChartInfoId: enNameData.ChartInfoId,
// ChartNameEn: enNameData.ChartNameEn
// })
// }else if([7,8,9].includes(this.chartInfo.Source)){//统计特征
// res=await statisticFeatureInterface.editChartEn({
// ChartInfoId: enNameData.ChartInfoId,
// ChartNameEn: enNameData.ChartNameEn
// })
// }else if(this.chartInfo.Source === 10) {//跨品种分析
// res=await crossVarietyInterface.editChartEn(enNameData)
// }else{
// res = await dataBaseInterface.chartInfoEditEn(enNameData)
// }
// if(res.Ret !==200) return
// this.$message.success(this.$t('MsgPrompt.edit_msg'))
// this.getChartInfo()
// this.setEnName = false
// },
/* 打开编辑信息弹窗 */
async openLangInfoDia() {
this.formItemArray={}
this.formItemArray.chartInfo=[]
this.formItemArray.chartsList=[]
this.formItemArray.chartInfo.push({
label:/* '图表名称' */this.$t('Chart.Detail.chart_name'),
value: this.currentLang==='en'?this.chartInfo.ChartNameEn:this.chartInfo.ChartName,
key:'ChartName',
id:this.chartInfo.ChartInfoId,
source: this.chartInfo.Source,
placeholder:/* '请输入图表名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.chart_name')})
})
if([1,2,5].includes(this.chartInfo.Source)){ //需要设置指标的
this.tableData.map(item =>{
this.formItemArray.chartsList.push([
{
label:/* '指标名称' */this.$t('Edb.Detail.e_name'),
value:this.currentLang==='en'?item.EdbNameEn:item.EdbName,
key:'EdbName',
id:item.EdbInfoId,
placeholder:/* '请输入指标名称' */ this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_name')})
},
{
label:/* '单位' */this.$t('Edb.Detail.e_unit'),
value:this.currentLang==='en'?item.UnitEn:item.Unit,
key:'Unit',
id:item.EdbInfoId,
placeholder:/* '请输入单位' */ this.$t('Edb.InputHolderAll.input_common',{label: this.$t('Edb.Detail.e_unit')})
}
])
})
}
//价格曲线
if(this.chartInfo.Source===2) {
this.formItemArray.chartInfo.push({
label:/* '期货名称' */this.$t('Chart.Detail.good_name'),
value:this.currentLang==='en'?this.commodityEdbList[1].EdbNameEn:this.commodityEdbList[1].EdbName,
key:'FutureGoodName',
id:this.chartInfo.ChartInfoId,
placeholder:/* '请输入期货名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.good_name')})
})
}
//利润曲线
else if(this.chartInfo.Source===5) {
this.formItemArray.chartInfo.push({
label:/* '盘面利润名称' */this.$t('Chart.Detail.profit_name'),
value: this.currentLang==='en'?this.chartInfo.ProfitNameEn:this.chartInfo.ProfitName,
key:'ProfitName',
id:this.chartInfo.ChartInfoId,
placeholder:/* '请输入盘面利润名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.profit_en_name')})
})
}
//跨品种分析
else if(this.chartInfo.Source===10) {
let { Data } = await crossVarietyInterface.chartLangOption({ChartInfoId: this.chartInfo.ChartInfoId})
const { TagList,VarietyList } = Data;
this.formItemArray.chartInfo.push({
label:/* 'X轴名称' */this.$t('Chart.Detail.x_name'),
value:this.currentLang==='en'?this.crossVarietyChartData.XNameEn:this.crossVarietyChartData.XName,
key:'XName',
id:TagList[0].ChartTagId,
placeholder:/* '请输入X轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.x_name')})
},
{
label:/* 'Y轴名称' */this.$t('Chart.Detail.y_name'),
value:this.currentLang==='en'?this.crossVarietyChartData.YNameEn:this.crossVarietyChartData.YName,
key:'YName',
id:TagList[1].ChartTagId,
placeholder:/* '请输入Y轴名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.y_name')})
})
VarietyList.forEach(item => {
this.formItemArray.chartsList.push(
{
label:/* '品种名称' */this.$t('Chart.Detail.variety_name'),
value:this.currentLang==='en'?item.ChartVarietyNameEn:item.ChartVarietyName,
key:'ChartVarietyName',
id:item.ChartVarietyId,
placeholder:/* '请输入品种名称' */this.$t('Chart.InputHolderAll.input_common',{label:this.$t('Chart.Detail.variety_name')})
}
)
})
}
this.isLangInfoDia = true
},
/* 更新版本图表信息 */
async updateLang(paramsData) {
let res=null
if([2,5].includes(this.chartInfo.Source)){//商品价格
res=await futuresInterface.setChartLangInfo({
ChartInfoId: paramsData.ChartInfoId,
ChartName: paramsData.ChartName,
Unit: paramsData.ChartEdbInfoList[0].Unit || '',
EdbName: paramsData.ChartEdbInfoList[0].EdbName || '',
FutureGoodName: paramsData.FutureGoodName || '',
ProfitName: paramsData.ProfitName || ''
})
}else if(this.chartInfo.Source===3){//相关性
res=await chartRelevanceApi.setChartLangInfo({
ChartInfoId: paramsData.ChartInfoId,
ChartName: paramsData.ChartName
})
}else if(this.chartInfo.Source===6){//拟合方程
res=await fittingEquationInterface.setChartLangInfo({
ChartInfoId: paramsData.ChartInfoId,
ChartNameEn: paramsData.ChartName
})
}else if([7,8,9].includes(this.chartInfo.Source)){//统计特征
res=await statisticFeatureInterface.setChartLangInfo({
ChartInfoId: paramsData.ChartInfoId,
ChartName: paramsData.ChartName
})
}else if(this.chartInfo.Source === 10) {//跨品种分析
res=await crossVarietyInterface.setChartLangInfo(paramsData)
}else{
res = await dataBaseInterface.setChartLangInfo(paramsData)
}
if(res.Ret !==200) return
this.$message.success(this.$t('MsgPrompt.edit_msg'))
this.getChartInfo()
this.isLangInfoDia = false
},
/* 切换柱形图中英文 */
changeBarOptions() {
//x轴
this.options.xAxis.categories = this.barXIdList.map(_ => this.currentLang === 'zh'
? this.barEdbData.find(edb => edb.EdbInfoId===_).EdbAliasName
: this.barEdbData.find(edb => edb.EdbInfoId===_).EdbNameEn)
// 单位
this.options.yAxis.forEach(item => {
item.title.text = this.currentLang === 'zh' ? this.chartInfo.Unit : this.chartInfo.Unit;
});
},
// 切换中英文时,更改图表配置
changeOptions(){
// console.log(this.tableData,'tableData');
// console.log(this.currentLang,this.chartInfo.ChartType);
console.log(this.options,'options');
// 散点图
if(this.chartInfo.ChartType == 5){
this.options.yAxis.title.text = this.currentLang == 'zh' ? this.options.yAxis.title.textCh : this.options.yAxis.title.textEn;
this.options.xAxis.title.text = this.currentLang == 'zh' ? this.options.xAxis.title.textCh : this.options.xAxis.title.textEn
this.options.series.forEach(item => {
item.name = this.currentLang == 'zh' ? item.nameCh : item.nameEn;
});
this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn;
}else{
// 单位
this.options.yAxis.forEach(item => {
item.title.text = this.currentLang == 'zh' ? item.title.textCh : item.title.textEn;
});
// 图例 名称
if(this.chartInfo.ChartType != 2){
// 季节图 不更改图例名称
this.options.series.forEach(item => {
item.name = this.currentLang == 'zh'
? item.nameCh
: (item.nameEn||item.nameCh)
});
}
//截面散点 x轴标题
if(this.chartInfo.ChartType === 10){
this.options.xAxis.title.text = this.currentLang == 'zh' ? this.options.xAxis.title.textCh : this.options.xAxis.title.textEn;
this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn;
this.options.series.forEach(item => {
if(!item.linkedTo) {
item.data.forEach(point => {
point.dataLabels.format = this.currentLang == 'zh' ? point.dataLabels.formatCh : (point.dataLabels.formatEn||point.dataLabels.formatCh);
point.dataLabels.color = (this.currentLang==='en' && !point.dataLabels.formatEn) ? '#999' : '#333'
})
}
});
}
//柱形图额外设置x轴中英文
this.chartInfo.ChartType ===7 && this.changeBarOptions();
}
},
/* 切换商品价格图中英文 */
changeCommodityLang() {
console.log(this.options)
this.options.yAxis.forEach(item => {
item.title.text = this.currentLang == 'zh' ? item.title.textCh : item.title.textEn
});
//图例
this.options.series.forEach(item => {
item.name = this.currentLang == 'zh' ? item.nameCh : item.nameEn
});
//tooltip
this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn
//x轴
this.options.xAxis.categories = this.commodityXData.map(_ => this.currentLang == 'zh' ? _.Name:_.NameEn);
},
/* 切换相关性图中英文 */
changeRelevanceLang(){
this.options.yAxis.forEach(item => {
item.title.text = this.currentLang == 'zh' ? item.title.textCh : item.title.textEn
});
//图例
this.options.series.forEach(item => {
item.name = this.currentLang == 'zh' ? item.nameCh : item.nameEn
});
//tooltip
this.options.tooltip.formatter = this.currentLang == 'zh' ? this.options.tooltip.formatterCh : this.options.tooltip.formatterEn
this.options.xAxis.title.text=this.currentLang == 'zh' ? this.options.xAxis.title.textCh : this.options.xAxis.title.textEn
},
/* ================================================================ */
/* 图表的配置项*/
setChartOptionHandle(newval) {
/* 显示类型 数据结构单独处理 */
const chartSetMap = {
1: this.setDefaultChart,
2: this.setSeasonChart,
3: this.setStackOrCombinChart,
4: this.setStackOrCombinChart,
5: this.setScatterChart,
6: this.setStackOrCombinChart,
};
chartSetMap[this.chartInfo.ChartType]&&chartSetMap[this.chartInfo.ChartType](newval)
//myeta内或者数据指标库内
if(this.$route.path === '/mychart'||this.$route.path==='/database') this.showData = true;
},
/* 曲线图设置 */
setDefaultChart(newval) {
/* 上下限显示和值的设置 */
if([4,6,7,8].includes(this.chartInfo.Source)) {
this.leftIndex = -1;
this.rightIndex = -1;
this.rightTwoIndex = -1;
this.chartLimit = {
min:'', //左轴上下限
max:'',
rightMin:'',//右轴上下限
rightMax:'',
rightTwoMin:'',//右二轴上下限
rightTwoMax:'',
};
}else {
this.leftIndex = newval.findIndex((item) => item.IsAxis===1);
this.rightIndex = newval.findIndex((item) => !item.IsAxis);
this.rightTwoIndex = newval.findIndex((item) => item.IsAxis ===2);
}
/* 主题样式*/
const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null;
//拼接标题 数据列
let data = [];
let ydata = [];
//有右二轴时排个序 按照左 右 右2的顺序
let chartData = newval.some(_ =>_.IsAxis===2) ? this.changeEdbOrder(newval) : _.cloneDeep(newval);
chartData.forEach((item, index) => {
//轴位置值相同的下标
let sameSideIndex = chartData.findIndex(
(i) => i.IsAxis === item.IsAxis
);
//获取对应轴的上下限
//预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除
const useTableLimit = ['/predictEdb','/addpredictEdb','/editpredictEdb','/chartThemeSet'].includes(this.$route.path)
//非ETA图库图表也不设置自定义上下限,相关性和统计特征也会用到曲线图
//若chartInfo.Source为1,需在之前调用setLimitData
const isETASource = this.chartInfo.Source===1
let minLimit = 0,maxLimit = 0
if(useTableLimit||!isETASource){
minLimit = newval[sameSideIndex].MinData
maxLimit = newval[sameSideIndex].MaxData
}
if(!useTableLimit&&isETASource){
const limitMap = {
0:['rightMin','rightMax'],
1:['min','max'],
2:['rightTwoMin','rightTwoMax']
}
if(limitMap[item.IsAxis]){
minLimit = this.chartLimit[`${limitMap[item.IsAxis][0]}`]||0
maxLimit = this.chartLimit[`${limitMap[item.IsAxis][1]}`]||0
}
}
//y轴
const textZh = item.ConvertUnit||item.Unit
const textEn = item.ConvertEnUnit||item.UnitEn||item.ConvertUnit||item.Unit
let yItem = {
...basicYAxis,
title: {
text: textZh,
textCh:textZh, // 中文
// 中文不存在,无论英文有无都显示空
textEn:textZh?textEn:'', // 英文
style:{
...chartTheme&&chartTheme.yAxisOptions.style
},
align: 'high',
rotation: 0,
y: -12,
x: (item.IsAxis===0 && this.rightTwoIndex>-1) ? -chartData[this.rightTwoIndex].Unit.length*12 : 0,
textAlign: item.IsAxis===1 ? 'left' : 'right',
reserveSpace: false
},
labels: {
formatter: function (ctx) {
return ctx.value;
},
align: 'center',
x: [0,2].includes(item.IsAxis) ? 5 : -5,
style: {
...chartTheme&&chartTheme.yAxisOptions.style,
}
},
opposite: [0,2].includes(item.IsAxis),
reversed: item.IsOrder,
min: Number(minLimit),
max: Number(maxLimit),
tickWidth: 1,
visible: sameSideIndex === index,
plotBands: this.setAxisPlotAreas(item.IsAxis),
plotLines: this.setAxisPlotLines(item.IsAxis)
};
//拼接标题 判断相同指标名称拼接来源
let dynamic_title = item.EdbName;
let dynamic_arr = chartData.filter(
(item) => dynamic_title === item.EdbName
);
// 拼接配置 IsAxis左轴1 右轴0 右2轴2 IsOrder正序false 逆序true EdbInfoType是否是领先指标
let dynamic_tag = this.concatDynamicTag(item);
let dynamic_tag_en = this.concatDynamicTag(item,'en');
//预测指标配置
let predict_params = item.EdbInfoCategoryType === 1 ? this.getPredictParams(item) : {};
//数据列
let obj = {
data: [],
type: (chartTheme&&chartTheme.lineOptions.lineType) || 'spline',
dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid',
yAxis: sameSideIndex,
name:
dynamic_arr.length > 1
? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}`
: `${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
nameCh:dynamic_arr.length > 1
? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}`
: `${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
nameEn:item.EdbNameEn?`${item.EdbNameEn}${dynamic_tag_en}`:`${item.EdbAliasName||item.EdbName}${dynamic_tag}`,
color: item.ChartColor,
lineWidth: Number(item.ChartWidth)||(chartTheme&&chartTheme.lineOptions.lineWidth),
...predict_params
};
item.DataList = item.DataList || [];
for (let i of item.DataList) {
obj.data.push([i.DataTimestamp, i.Value]);
}
data.push(obj);
ydata.push(yItem);
});
/* x轴处理 */
let isLessThanOneYear = this.xLabelDealHandle();
let xAxis = {
...defaultOpts.xAxis,
labels: {
formatter: function (ctx) {
return isLessThanOneYear
? Highcharts.dateFormat('%m/%d', ctx.value)
: Highcharts.dateFormat('%y/%m', ctx.value);
},
style: {
...chartTheme&&chartTheme.xAxisOptions.style
}
},
plotBands: this.setAxisPlotAreas(3,'datetime'),
plotLines: this.setAxisPlotLines(3,'datetime')
}
this.options = {
series: data,
yAxis: ydata,
xAxis
};
//滚动相关性独立tooltip
if(this.chartInfo.Source === 4) {
const { LeadValue,LeadUnit } = this.relevanceChartData.CorrelationChartInfo;
let relevanceUnitEnMap = this.relevanceUnitEnMap;
this.options.tooltip = {
formatter: function() {
let str = `${Highcharts.dateFormat('%Y/%m/%d',this.x)}
相关性系数:${this.y.toFixed(4)}
领先${LeadValue+LeadUnit}
` return str }, formatterCh: function() { let str = `${Highcharts.dateFormat('%Y/%m/%d',this.x)}相关性系数:${this.y.toFixed(4)}
领先${LeadValue+LeadUnit}
` return str }, formatterEn: function() { let str = `${Highcharts.dateFormat('%Y/%m/%d',this.x)}Correlation coefficient:${this.y.toFixed(4)}
lead${LeadValue+relevanceUnitEnMap[LeadUnit]}
` return str } } } if(this.currentLang=='en') this.changeOptions() }, /* 堆叠图/组合图设置 本来和曲线图逻辑基本一致兼容下即可 为了以后便于维护和阅读还是拆开写吧 */ setStackOrCombinChart(newval) { const chartTypeMap = { 3: 'areaspline', 4: 'column', 6: '' }; let chartStyle = chartTypeMap[this.chartInfo.ChartType]; /* 主题样式*/ const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null; //拼接标题 数据列 let data = []; let ydata = []; //有右二轴时排个序 按照左 右 右2的顺序 let chartData = newval.some(_ =>_.IsAxis===2) ? this.changeEdbOrder(newval) : _.cloneDeep(newval); chartData.forEach((item, index) => { //轴位置值相同的下标 let sameSideIndex = chartData.findIndex( (i) => i.IsAxis === item.IsAxis ); //堆叠图的yAxis必须一致 数据列所对应的y轴 let serie_yIndex = index; if([3,4].includes(this.chartInfo.ChartType)) { // 类型为堆叠图时公用第一个指标y轴 serie_yIndex = 0; } else if(this.chartInfo.ChartType ===6 && ['areaspline','column'].includes(item.ChartStyle)) { // 组合图找第一个堆叠柱状或面积的作为公用 serie_yIndex = chartData.findIndex(i => i.ChartStyle === item.ChartStyle); } //数据对应的y轴是公用轴则配置也共享 item.IsAxis = serie_yIndex === index ? item.IsAxis : chartData[serie_yIndex].IsAxis; item.IsOrder = serie_yIndex === index ? item.IsOrder : chartData[serie_yIndex].IsOrder; /* 上下限显示的设置 */ this.leftIndex = [3,4].includes(this.chartInfo.ChartType) ? (chartData[serie_yIndex].IsAxis ? serie_yIndex : -1) : newval.findIndex((item) => item.IsAxis===1); this.rightIndex = [3,4].includes(this.chartInfo.ChartType) ? (chartData[serie_yIndex].IsAxis ? -1 : serie_yIndex) : newval.findIndex((item) => !item.IsAxis); this.rightTwoIndex = [3,4].includes(this.chartInfo.ChartType) ? -1 : newval.findIndex((item) => item.IsAxis===2); //获取对应轴的上下限 //预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除 const useTableLimit = ['/predictEdb','/chartThemeSet'].includes(this.$route.path) let minLimit = 0,maxLimit = 0 if(useTableLimit){ minLimit = chartData[sameSideIndex].MinData maxLimit = chartData[sameSideIndex].MaxData }else{ const limitMap = { 0:['rightMin','rightMax'], 1:['min','max'], 2:['rightTwoMin','rightTwoMax'] } if(limitMap[item.IsAxis]){ minLimit = this.chartLimit[`${limitMap[item.IsAxis][0]}`]||0 maxLimit = this.chartLimit[`${limitMap[item.IsAxis][1]}`]||0 } } //y轴 const textZh = item.ConvertUnit||item.Unit const textEn = item.ConvertEnUnit||item.UnitEn||item.ConvertUnit||item.Unit let yItem = { ...basicYAxis, title: { text: textZh, textCh:textZh, // 中文 // 中文不存在,无论英文有无都显示空 textEn:textZh?textEn:'', // 英文 style:{ ...chartTheme&&chartTheme.yAxisOptions.style }, align: 'high', rotation: 0, y: -12, x: (item.IsAxis===0 && this.rightTwoIndex>-1) ? -chartData[this.rightTwoIndex].Unit.length*12 : 0, textAlign: item.IsAxis===1 ? 'left' : 'right', reserveSpace: false }, labels: { formatter: function (ctx) { return ctx.value; }, align: 'center', x: [0,2].includes(item.IsAxis) ? 5 : -5, style: { ...chartTheme&&chartTheme.yAxisOptions.style } }, opposite: [0,2].includes(item.IsAxis), reversed: item.IsOrder, min: Number(minLimit), max: Number(maxLimit), tickWidth: 1, visible: serie_yIndex === index && sameSideIndex ===index, plotBands: this.setAxisPlotAreas(item.IsAxis), plotLines: this.setAxisPlotLines(item.IsAxis) }; //拼接标题 判断相同指标名称拼接来源 let dynamic_title = item.EdbName; let dynamic_arr = chartData.filter( (item) => dynamic_title === item.EdbName ); // 拼接配置 IsAxis左轴1 右轴0 IsOrder正序false 逆序true EdbInfoType是否是领先指标 let dynamic_tag = this.concatDynamicTag(item); let dynamic_tag_en = this.concatDynamicTag(item,'en'); //预测指标配置 let predict_params = item.EdbInfoCategoryType === 1 ? this.getPredictParams(item,chartStyle) : {}; //数据列 let obj = { data: [], type: chartStyle || item.ChartStyle, yAxis: serie_yIndex, name: dynamic_arr.length > 1 ? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}` : `${item.EdbAliasName||item.EdbName}${dynamic_tag}`, nameCh:dynamic_arr.length > 1 ? `${item.EdbAliasName||item.EdbName}(${item.SourceName})${dynamic_tag}` : `${item.EdbAliasName||item.EdbName}${dynamic_tag}`, nameEn:item.EdbNameEn?`${item.EdbNameEn}${dynamic_tag_en}`:`${item.EdbAliasName||item.EdbName}${dynamic_tag}`, color: item.ChartColor, lineWidth: Number(item.ChartWidth)||(chartTheme&&chartTheme.lineOptions.lineWidth), fillColor: (this.chartInfo.ChartType === 3 || (this.chartInfo.ChartType === 6 && item.ChartStyle === 'areaspline')) ? item.ChartColor : undefined, borderWidth: 1, borderColor: item.ChartColor, zIndex: (this.chartInfo.ChartType === 6 && ['line','spline'].includes(item.ChartStyle)) ? 1 : 0, //防止组合图曲线被遮住 ...predict_params }; item.DataList = item.DataList || []; for (let i of item.DataList) { obj.data.push([i.DataTimestamp, i.Value]); } data.push(obj); ydata.push(yItem); }); let isLessThanOneYear = this.xLabelDealHandle(); let xAxis = { ...defaultOpts.xAxis, labels: { formatter: function (ctx) { return isLessThanOneYear ? Highcharts.dateFormat('%m/%d', ctx.value) : Highcharts.dateFormat('%y/%m', ctx.value); }, style: { ...chartTheme&&chartTheme.xAxisOptions.style } }, plotBands: this.setAxisPlotAreas(3,'datetime'), plotLines: this.setAxisPlotLines(3,'datetime') } this.options = { series: data, yAxis: ydata, xAxis, }; if(this.currentLang=='en') this.changeOptions() }, /* 季节图设置 农历 公历 */ setSeasonChart(newval) { /* 季节性图的图表配置 */ this.leftIndex = 0; this.rightIndex = -1; this.rightTwoIndex = -1; const chartData = newval[0]; // 农历数据需要去除第一项 农历和公历处理逻辑一样 /** * isPredictorChart - 预测指标的chartInfo.vue组件内定义的变量 * 预测指标成图 还是之前农历图的逻辑,有空再拆吧,做个兼容 */ const chartDataHandle=this.calendar_type === '农历'? this.isPredictorChart?chartData.DataList.List.filter((item, index) => index > 0): chartData.DataList.filter((item, index) => index > 0): chartData.DataList /* 主题样式*/ const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null; let seasonYdata = [], seasonData = []; //获取对应轴的上下限 //预测指标-走势图;图表配置-主题设置;不使用自定义上下限,剔除 const useTableLimit = ['/predictEdb','/chartThemeSet','/addpredictEdb','/editpredictEdb'].includes(this.$route.path) let minLimit = 0,maxLimit = 0 if(useTableLimit){ minLimit = chartData.MinData maxLimit = chartData.MaxData }else{ minLimit = this.chartLimit.min||0 maxLimit = this.chartLimit.max||0 } //数据列 for (let j of chartDataHandle) { //预测指标配置 let predict_params = chartData.EdbInfoCategoryType === 1 ? this.getSeasonPredictParams(j.CuttingDataTimestamp) : {}; let serie_item = { data: [], type: (chartTheme&&chartTheme.lineOptions.lineType) || chartData.ChartStyle, dashStyle: (chartTheme&&chartTheme.lineOptions.dashStyle)||'Solid', yAxis: 0, name: this.isPredictorChart?j.Year:j.ChartLegend, lineWidth: (chartTheme&&chartTheme.lineOptions.lineWidth) || 1, ...predict_params }; const data_array = this.calendar_type === '农历' && this.isPredictorChart?_.cloneDeep(j.Items):_.cloneDeep(j.DataList); data_array && data_array.forEach((item) => { serie_item.data.push([item.DataTimestamp, item.Value]); }); seasonData.push(serie_item); } //y轴 const textZh = chartData.ConvertUnit||chartData.Unit const textEn = chartData.ConvertEnUnit||chartData.UnitEn||chartData.ConvertUnit||chartData.Unit seasonYdata = [{ ...seasonOptions.yAxis, labels: { formatter: function () { let val = this.value; return val; }, align: 'center', style: { ...chartTheme&&chartTheme.yAxisOptions.style } }, title: { text: `${textZh}`, textCh:textZh, // 中文 // 中文不存在,无论英文有无都显示空 textEn:textZh?textEn:'', // 英文 style:{ ...chartTheme&&chartTheme.yAxisOptions.style }, // text: null, align: 'high', rotation: 0, y: -12, x: 0, textAlign: 'left', reserveSpace: false }, max: Number(maxLimit), min: Number(minLimit), plotBands: this.setAxisPlotAreas(1), plotLines: this.setAxisPlotLines(1) }]; /* x轴显示月日 */ const xAxis = { ...defaultOpts.xAxis, labels: { formatter: function () { return Highcharts.dateFormat('%m/%d', this.value); }, style: { ...chartTheme&&chartTheme.xAxisOptions.style } }, plotBands: this.setAxisPlotAreas(3,'datetime'), plotLines: this.setAxisPlotLines(3,'datetime') } const tooltip = { ...defaultOpts.tooltip, dateTimeLabelFormats: { // 时间格式化字符 day: '%m/%d', week: '%m/%d', month: '%m/%d', year: '%m/%d', }, xDateFormat: '%m/%d', } let colors = chartTheme&&chartTheme.colorsOptions.reverse()||seasonOptions.colors; // 季节性 农历成图老逻辑,需要截取 指标成图还在用,后面需要统一或者拆分 let rangeSelector = this.calendar_type === '农历' && this.isPredictorChart ? { enabled: true, selected: 0, inputStyle: { display: 'none', }, labelStyle: { display: 'none', }, buttonTheme: { style: { display: 'none', }, }, buttons: [ { type: 'month', count: 12, text: '12月', }, { type: 'month', count: 15, text: '15月', }, { type: 'all', text: '全部', type: 'all', }, ], } : { enabled: false, }; this.options = { colors: colors.slice(-chartDataHandle.length), series: seasonData, yAxis: seasonYdata, xAxis, rangeSelector, tooltip }; if(this.currentLang=='en') this.changeOptions() }, /* 散点图设置 只允许2指标画图第一个指标值为x轴 第二个指标为y轴 */ setScatterChart(newval) { console.log(newval); this.leftIndex = 1; this.rightIndex = -1; this.rightTwoIndex = -1; if(newval.length !== 2) { this.leftIndex = -1 this.options = {} return }; console.log('===========:',this.isSetExtremeValue) /* 主题样式*/ const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null; // 取2个指标中日期相同的数据 let real_data = []; let tmpData_date = {};//用来取点对应的日期 let data1 = _.cloneDeep(newval)[0].DataList || []; let data2 = _.cloneDeep(newval)[1].DataList || []; data1.forEach(_item => { data2.forEach(_item2 => { if(_item.DataTimestamp === _item2.DataTimestamp) { _item.DataTime = _item.DataTime.replace(/-/g,'/'); //日期 let itemIndex =_item.Value + "_" +_item2.Value if(tmpData_date[itemIndex]) { tmpData_date[itemIndex].push(_item.DataTime) } else { tmpData_date[itemIndex] = [_item.DataTime] } //值 real_data.push({ x: _item.Value, y: _item2.Value }) } }) }) real_data.sort((x,y) => x-y); //悬浮窗 拼接日期 原始指标名称 let tooltip = { formatter: function() { return `${ 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() }相关性系数:${this.y.toFixed(4)}
领先${ Source===3 ?this.x+'期' : LeadValue+LeadUnit}
` return str }, formatterCh: function() { let str = `相关性系数:${this.y.toFixed(4)}
领先${ Source===3 ?this.x+'期' : LeadValue+LeadUnit}
` return str }, formatterEn: function() { let str = `Correlation coefficient:${this.y.toFixed(4)}
lead${ Source===3 ? this.x+'stage' : LeadValue+relevanceUnitEnMap[LeadUnit]}
` return str } } this.options = { isRelevanceChart: Source===3, title: { text:'' }, series: seriesData, yAxis: [yAxis] , xAxis:xAxis, tooltip } console.log('处理结束相关性图表',this.options); }, /* 截面散点图获取详情赋值 */ initSectionScatterData({ DataResp }) { this.chartLimit = { min: Number(DataResp.YMinValue), max: Number(DataResp.YMaxValue), x_min: Number(DataResp.XMinValue), x_max: Number(DataResp.XMaxValue) } this.sectionScatterData = DataResp; }, /* 截面散点图设置 sectionScatterData */ setSectionScatterChart() { this.leftIndex = -1; this.rightIndex = -1; this.rightTwoIndex = -1; const { DataList,XName,XNameEn,YName,YNameEn } = this.sectionScatterData; const { min,max,x_min,x_max } = this.chartLimit; /* 主题样式*/ const chartTheme = this.chartInfo.ChartThemeStyle ? JSON.parse(this.chartInfo.ChartThemeStyle) : null; //y轴 let yAxis = { ...basicYAxis, title: { text: YName, textCh:YName,// 中文 textEn:YNameEn||YName, style:{ ...chartTheme&&chartTheme.yAxisOptions.style }, align: 'middle', }, labels: { style:{ ...chartTheme&&chartTheme.yAxisOptions.style }, }, opposite: false, reversed: false, min: Number(min), max: Number(max), tickWidth: 1, plotBands: this.setAxisPlotAreas(1), plotLines: this.setAxisPlotLines(1) } //x轴 let xAxis = { ...scatterXAxis, title: { text: XName, textCh:XName,// 中文 textEn:XNameEn || XName, style: { ...chartTheme&&chartTheme.xAxisOptions.style }, align: 'middle', }, labels: { style:{ ...chartTheme&&chartTheme.xAxisOptions.style }, }, min: Number(x_min), max: Number(x_max), plotBands: this.setAxisPlotAreas(3), plotLines: this.setAxisPlotLines(3) } //数据列 let series = []; DataList.forEach(item => { //数据列 let series_item = { data: [], type: 'scatter', name: item.Name, nameCh: item.Name, nameEn: item.NameEn||item.Name, color: item.Color, chartType: 'linear', zIndex:1, marker: { radius: (chartTheme&&chartTheme.lineOptions.radius)||5, }, } item.EdbInfoList.forEach(_ => { series_item.data.push({ x: _.XValue, y: _.YValue, dataLabels: { enabled: _.IsShow, allowOverlap: true, align: 'left', format: _.Name, formatCh: _.Name, formatEn: _.NameEn||_.Name } }) }) series.push(series_item); //趋势线 if(item.ShowTrendLine) { let trend_data = item.TrendLimitData.map((_,_index) => ( _index === item.TrendLimitData.length-1 ? { x: _.X, y: _.Y, dataLabels: { enabled: item.ShowRSquare || item.ShowFitEquation, align: 'left', color: '#666', x: 20, y: 30, zIndex: 9, allowOverlap: true, formatter: function(){ let tag = ''; item.ShowRSquare && item.ShowFitEquation ? tag =`${item.TrendLine}