|
@@ -1,9 +1,12 @@
|
|
|
<script setup>
|
|
|
import chartBox from './component/chartBox.vue'
|
|
|
+import noAuth from './component/noAuth.vue'
|
|
|
+import sharePoster from '../components/SharePoster.vue'
|
|
|
import { Popup, Toast,Picker } from 'vant';
|
|
|
-import {ref,onMounted, reactive, watch} from 'vue'
|
|
|
+import {ref,onMounted, reactive, watch,computed} from 'vue'
|
|
|
import {useRoute, useRouter,onBeforeRouteUpdate} from 'vue-router'
|
|
|
-import moment, { min } from 'moment'
|
|
|
+import moment from 'moment'
|
|
|
+import _ from 'lodash';
|
|
|
import Highcharts from 'highcharts/highstock';
|
|
|
import {apiChartInfo,apiChartList,apiChartSave,apiChartBeforeAndNext,apiChartRefresh} from '@/api/hzyb/chart.js'
|
|
|
const router=useRouter()
|
|
@@ -12,6 +15,50 @@ document.title='图表详情'
|
|
|
|
|
|
localStorage.setItem('hzyb-token',route.query.token)
|
|
|
|
|
|
+// 散点x轴
|
|
|
+const scatterXAxis = {
|
|
|
+ tickPosition: 'inside',
|
|
|
+ lineColor: '#bfbfbf',
|
|
|
+ tickColor: '#bfbfbf',
|
|
|
+ tickLength:5,
|
|
|
+ ordinal: false,
|
|
|
+ type: 'linear',
|
|
|
+}
|
|
|
+
|
|
|
+// 基础y轴配置
|
|
|
+const basicYAxis = {
|
|
|
+ tickWidth: 1,
|
|
|
+ tickLength: 5,
|
|
|
+ lineWidth: 1,
|
|
|
+ lineColor: '#bfbfbf',
|
|
|
+ tickColor: '#bfbfbf',
|
|
|
+ offset: 0,
|
|
|
+ visible: true,
|
|
|
+ gridLineWidth: 0,
|
|
|
+ tickPosition: 'inside',
|
|
|
+ endOnTick: false,
|
|
|
+ startOnTick: false,
|
|
|
+ showLastLabel: true,
|
|
|
+ tickPixelInterval: 50,
|
|
|
+}
|
|
|
+
|
|
|
+//基础x轴配置
|
|
|
+const basicXAxis={
|
|
|
+ tickPosition: 'inside',
|
|
|
+ lineColor: '#bfbfbf',
|
|
|
+ tickColor: '#bfbfbf',
|
|
|
+ tickLength:5,
|
|
|
+ type: 'datetime',
|
|
|
+ ordinal: false,
|
|
|
+ dateTimeLabelFormats: {
|
|
|
+ day: '%y/%m',
|
|
|
+ week: '%y/%m',
|
|
|
+ month: '%y/%m',
|
|
|
+ year: '%y/%m',
|
|
|
+ },
|
|
|
+ xDateFormat:'%Y-%m-%d'
|
|
|
+}
|
|
|
+
|
|
|
|
|
|
// 获取用户信息
|
|
|
import {apiUserInfo} from '@/api/hzyb/user'
|
|
@@ -64,7 +111,7 @@ const makeTimeData=(type)=>{
|
|
|
}
|
|
|
const handleShowDate=()=>{
|
|
|
// if(columns.value.length===0){
|
|
|
- if(resData.value.ChartInfo.ChartType===1){//曲线图
|
|
|
+ if( sameOptionType.value.includes(resData.value.ChartInfo.ChartType)){//曲线图
|
|
|
columns.value=makeTimeData(1)
|
|
|
}else if(resData.value.ChartInfo.ChartType===2){//季节性图表
|
|
|
columns.value=makeTimeData(2)
|
|
@@ -75,7 +122,7 @@ const handleShowDate=()=>{
|
|
|
// 确定选择时间
|
|
|
const handleConfirmDate=(e)=>{
|
|
|
let start='',end=''
|
|
|
- if(resData.value.ChartInfo.ChartType===1){
|
|
|
+ if(sameOptionType.value.includes(resData.value.ChartInfo.ChartType)){
|
|
|
start=e[0]+'-'+e[1]
|
|
|
end=e[2]+'-'+e[3]
|
|
|
}else if(resData.value.ChartInfo.ChartType===2){
|
|
@@ -103,10 +150,10 @@ let dateTypeList=ref([
|
|
|
// name: '18年至今',
|
|
|
// value: 7,
|
|
|
// },
|
|
|
- {
|
|
|
- name: '19年至今',
|
|
|
- value: 8,
|
|
|
- },
|
|
|
+ // {
|
|
|
+ // name: '19年至今',
|
|
|
+ // value: 8,
|
|
|
+ // },
|
|
|
{
|
|
|
name: '20年至今',
|
|
|
value: 9,
|
|
@@ -115,6 +162,10 @@ let dateTypeList=ref([
|
|
|
name: '21年至今',
|
|
|
value: 4,
|
|
|
},
|
|
|
+ {
|
|
|
+ name: '22年至今',
|
|
|
+ value: 11,
|
|
|
+ },
|
|
|
{
|
|
|
name: '全部',
|
|
|
value: 0,
|
|
@@ -128,11 +179,13 @@ const dateTypeClick=(item)=>{
|
|
|
getChartInfo()
|
|
|
}
|
|
|
|
|
|
-let calendarType=ref('公历')//季节图 公历/农历
|
|
|
+let calendarType=ref('')//季节图 公历/农历
|
|
|
// 公历/农历切换
|
|
|
const calendarTypeChange=(val)=>{
|
|
|
calendarType.value=val
|
|
|
- getChartInfo('init')
|
|
|
+ // startDate.value=''
|
|
|
+ // endDate.value=''
|
|
|
+ getChartInfo()
|
|
|
}
|
|
|
|
|
|
|
|
@@ -146,6 +199,16 @@ let chartData=ref({
|
|
|
})// 图表配置数据
|
|
|
let resData=ref(null)//接口详情数据
|
|
|
let loading=ref(false)
|
|
|
+const sameOptionType = ref([1,3,4,5,6]);//筛选框一样的图表类型 曲线/面积/柱状/散点/组合 常规图
|
|
|
+const chartItemStyleArr = ref([
|
|
|
+ { label: '曲线图', key: 1 ,value: 'spline'},
|
|
|
+ { label: '面积图', key: 3 ,value: 'areaspline'},
|
|
|
+ { label: '柱状图', key: 4 ,value: 'column'},
|
|
|
+ { label: '散点图', key: 5 ,value: 'scatter'}
|
|
|
+])//组合图配置时可选类型
|
|
|
+
|
|
|
+let noauth=ref(false)
|
|
|
+let noAuthData=ref(null)
|
|
|
// 如果type:init 则是初始化获取数据
|
|
|
const getChartInfo=async (type)=>{
|
|
|
// resData.value=null
|
|
@@ -153,12 +216,13 @@ const getChartInfo=async (type)=>{
|
|
|
const res=await apiChartInfo({
|
|
|
ChartInfoId:ChartInfoId,
|
|
|
DateType:dateType.value,
|
|
|
- StartDate:startDate.value&&resData.value.ChartInfo.ChartType===1?startDate.value:'',
|
|
|
- EndDate:endDate.value&&resData.value.ChartInfo.ChartType===1?endDate.value:'',
|
|
|
+ StartDate:startDate.value&&sameOptionType.value.includes(resData.value.ChartInfo.ChartType)?startDate.value:'',
|
|
|
+ EndDate:endDate.value&&sameOptionType.value.includes(resData.value.ChartInfo.ChartType)?endDate.value:'',
|
|
|
SeasonStartDate:startDate.value&&resData.value.ChartInfo.ChartType===2?startDate.value:'',
|
|
|
SeasonEndDate:endDate.value&&resData.value.ChartInfo.ChartType===2?endDate.value:'',
|
|
|
Calendar:calendarType.value,
|
|
|
- Authorization:route.query.token
|
|
|
+ Authorization:route.query.token,
|
|
|
+ MyChartClassifyId:Number(route.query.MyChartClassifyId)
|
|
|
})
|
|
|
loading.value=false
|
|
|
if(res.code===200){
|
|
@@ -166,22 +230,50 @@ const getChartInfo=async (type)=>{
|
|
|
// document.title=res.data.ChartInfo.ChartName
|
|
|
|
|
|
// 设置highchart配置 ChartType: 1曲线图 2季节图:季节图中公历和农历数据结构不同
|
|
|
- if(res.data.ChartInfo.ChartType===1){
|
|
|
+ if( res.data.ChartInfo.ChartType !==2 ){
|
|
|
if(type=='init'){
|
|
|
- dateType.value=res.data.ChartInfo.DateType||3
|
|
|
+ dateType.value=res.data.ChartInfo.DateType
|
|
|
startDate.value=res.data.ChartInfo.StartDate||''
|
|
|
endDate.value=res.data.ChartInfo.EndDate||''
|
|
|
+ calendarType.value=res.data.ChartInfo.Calendar||'公历'
|
|
|
}
|
|
|
- setSplineOpt(res.data.EdbInfoList)
|
|
|
+ const chartSetMap = {
|
|
|
+ 1: setSplineOpt,
|
|
|
+ 3: setStackOrCombinChart,
|
|
|
+ 4: setStackOrCombinChart,
|
|
|
+ 5: setScatterOptions,
|
|
|
+ 6: setStackOrCombinChart
|
|
|
+ };
|
|
|
+
|
|
|
+ chartSetMap[res.data.ChartInfo.ChartType](res.data.EdbInfoList)
|
|
|
}else{
|
|
|
if(type=='init'){
|
|
|
- dateType.value=res.data.ChartInfo.DateType||3
|
|
|
+ dateType.value=res.data.ChartInfo.DateType
|
|
|
startDate.value=res.data.ChartInfo.SeasonStartDate||''
|
|
|
endDate.value=res.data.ChartInfo.SeasonEndDate||''
|
|
|
+ calendarType.value=res.data.ChartInfo.Calendar||'公历'
|
|
|
}
|
|
|
|
|
|
setSeasonOpt(res.data.EdbInfoList[0])
|
|
|
}
|
|
|
+
|
|
|
+ // 向小程序发送分享数据
|
|
|
+ let postData = {
|
|
|
+ params:{
|
|
|
+ chartInfoId:ChartInfoId,
|
|
|
+ searchVal:decodeURIComponent(route.query.searchVal)||'',
|
|
|
+ MyChartId:route.query.MyChartId||'',
|
|
|
+ MyChartClassifyId:route.query.MyChartClassifyId||'',
|
|
|
+ },
|
|
|
+ title: res.data.ChartInfo.ChartName,
|
|
|
+ shareImg:res.data.ChartInfo.ChartImage
|
|
|
+ };
|
|
|
+ wx.miniProgram.postMessage({ data: postData });
|
|
|
+
|
|
|
+
|
|
|
+ }else if(res.code==403){
|
|
|
+ noauth.value=true
|
|
|
+ noAuthData.value=res.data
|
|
|
}
|
|
|
}
|
|
|
getChartInfo('init')
|
|
@@ -193,7 +285,7 @@ onBeforeRouteUpdate((nval)=>{
|
|
|
dateType.value=null
|
|
|
startDate.value=''
|
|
|
endDate.value=''
|
|
|
- // router.go(0)
|
|
|
+
|
|
|
getChartInfo('init')
|
|
|
})
|
|
|
|
|
@@ -219,25 +311,11 @@ const handleCloseLimit=()=>{//点击遮罩层关闭弹窗或者点击取消关
|
|
|
}
|
|
|
|
|
|
|
|
|
-// 设置曲线图配置
|
|
|
+// 设置常规图配置 曲线
|
|
|
const setSplineOpt=(data)=>{
|
|
|
let series=[]
|
|
|
- let xAxis={
|
|
|
- tickPosition: 'inside',
|
|
|
- lineColor: '#bfbfbf',
|
|
|
- tickColor: '#bfbfbf',
|
|
|
- tickLength:5,
|
|
|
- type: 'datetime',
|
|
|
- ordinal: false,
|
|
|
- dateTimeLabelFormats: {
|
|
|
- day: '%y/%m',
|
|
|
- week: '%y/%m',
|
|
|
- month: '%y/%m',
|
|
|
- year: '%y/%m',
|
|
|
- },
|
|
|
- xDateFormat:'%Y-%m-%d'
|
|
|
- }
|
|
|
let yAxis=[]
|
|
|
+ let xAxis = {}
|
|
|
|
|
|
let temYLeftArr=[]
|
|
|
let temYRightArr=[]
|
|
@@ -247,36 +325,25 @@ const setSplineOpt=(data)=>{
|
|
|
let minAndMaxTimeTemArr=[]//存放所有指标的最大最小时间
|
|
|
|
|
|
data.forEach((item,index)=>{
|
|
|
+
|
|
|
+ //轴位置值相同的下标
|
|
|
+ let sameSideIndex = data.findIndex(i => i.IsAxis === item.IsAxis);
|
|
|
+
|
|
|
let dynamic_title = item.EdbName;
|
|
|
let dynamic_arr = data.filter(
|
|
|
(item) => dynamic_title === item.EdbName
|
|
|
);
|
|
|
- // 拼接配置 IsAxis左轴1 右轴0 IsOrder正序false 逆序true EdbInfoType是否是领先指标
|
|
|
- let dynamic_tag =item.IsAxis && item.IsOrder && item.EdbInfoType
|
|
|
- ? '(逆序)'
|
|
|
- : !item.IsAxis && item.IsOrder && item.EdbInfoType
|
|
|
- ? '(右轴,逆序)'
|
|
|
- : !item.IsAxis && !item.IsOrder && item.EdbInfoType
|
|
|
- ? '(右轴)'
|
|
|
- : !item.IsAxis && !item.IsOrder && !item.EdbInfoType
|
|
|
- ? `(右轴,领先${item.LeadValue}${item.LeadUnit})`
|
|
|
- : !item.IsAxis && item.IsOrder && !item.EdbInfoType
|
|
|
- ? `(右轴,逆序,领先${item.LeadValue}${item.LeadUnit})`
|
|
|
- : item.IsAxis && item.IsOrder && !item.EdbInfoType
|
|
|
- ? `(逆序,领先${item.LeadValue}${item.LeadUnit})`
|
|
|
- : item.IsAxis && !item.IsOrder && !item.EdbInfoType
|
|
|
- ? `(领先${item.LeadValue}${item.LeadUnit})`
|
|
|
- : '';
|
|
|
+ //处理数据列name
|
|
|
+ let temName= setDyncmicSerieName(item,dynamic_arr)
|
|
|
+
|
|
|
let seriesItemObj={
|
|
|
data:[],
|
|
|
dataGrouping:{
|
|
|
enabled:false
|
|
|
},
|
|
|
- type:item.ChartStyle,
|
|
|
+ type: 'spline',
|
|
|
yAxis:index,
|
|
|
- name:dynamic_arr.length > 1
|
|
|
- ? `${item.EdbName}(${item.SourceName})${dynamic_tag}`
|
|
|
- : `${item.EdbName}${dynamic_tag}`,//拼接标题 判断相同指标名称拼接来源
|
|
|
+ name:temName,
|
|
|
color: item.ChartColor,
|
|
|
lineWidth: Number(item.ChartWidth),
|
|
|
visible:true,
|
|
@@ -298,31 +365,27 @@ const setSplineOpt=(data)=>{
|
|
|
}
|
|
|
|
|
|
let yItem={
|
|
|
+ ...basicYAxis,
|
|
|
IsAxis:item.IsAxis,
|
|
|
labels: {
|
|
|
formatter: function (ctx) {
|
|
|
- return ctx.value;
|
|
|
+ return sameSideIndex !== index ? '' : ctx.value;
|
|
|
},
|
|
|
align: 'center',
|
|
|
y:5,
|
|
|
},
|
|
|
+ tickWidth: sameSideIndex !== index ? 0 : 1,
|
|
|
+ // title: {
|
|
|
+ // text: `单位:${item.Unit}`,
|
|
|
+ // align: 'high',
|
|
|
+ // rotation: 0,
|
|
|
+ // y: 0,
|
|
|
+ // offset: -60,
|
|
|
+ // },
|
|
|
opposite: item.IsAxis === 0,
|
|
|
reversed: item.IsOrder,
|
|
|
min: item.MinData,
|
|
|
max: item.MaxData,
|
|
|
- tickWidth: 1,
|
|
|
- tickLength: 5,
|
|
|
- lineWidth: 1,
|
|
|
- lineColor: '#bfbfbf',
|
|
|
- tickColor: '#bfbfbf',
|
|
|
- offset: 0,
|
|
|
- visible: true,
|
|
|
- gridLineWidth: 0,
|
|
|
- tickPosition: 'inside',
|
|
|
- endOnTick: false,
|
|
|
- startOnTick: false,
|
|
|
- showLastLabel: true,
|
|
|
- tickPixelInterval: 50,
|
|
|
chartEdbInfo:item//指标数据用于在保存时读取指标数据
|
|
|
}
|
|
|
yAxis.push(yItem)
|
|
@@ -347,7 +410,7 @@ const setSplineOpt=(data)=>{
|
|
|
const bool_time = xTimeDiffer(minTime,maxTime)
|
|
|
if(bool_time){
|
|
|
xAxis={
|
|
|
- ...xAxis,
|
|
|
+ ...basicXAxis,
|
|
|
labels: {
|
|
|
formatter: (ctx)=> {
|
|
|
return Highcharts.dateFormat('%m/%d', ctx.value)
|
|
@@ -360,7 +423,7 @@ const setSplineOpt=(data)=>{
|
|
|
// let minYear=new Date(minTime).getFullYear()
|
|
|
|
|
|
xAxis={
|
|
|
- ...xAxis,
|
|
|
+ ...basicXAxis,
|
|
|
tickInterval:((maxTime-minTime)/6)/(24*3600*1000)>30?(maxTime-minTime)/6:24*3600*1000*30,
|
|
|
}
|
|
|
|
|
@@ -393,7 +456,208 @@ const setSplineOpt=(data)=>{
|
|
|
}
|
|
|
})
|
|
|
chartData.value.yAxis=yAxis
|
|
|
+ chartData.value.rangeSelector={ enabled: false}
|
|
|
+
|
|
|
+}
|
|
|
+
|
|
|
+/* 堆叠图/组合图设置
|
|
|
+ 本来和曲线图逻辑基本一致兼容下即可 为了以后便于维护和阅读还是拆开写吧
|
|
|
+*/
|
|
|
+const setStackOrCombinChart = data => {
|
|
|
+ //图表类型
|
|
|
+ const chartTypeMap = {
|
|
|
+ 3: 'areaspline',
|
|
|
+ 4: 'column',
|
|
|
+ 6: ''
|
|
|
+ };
|
|
|
+ let chartStyle = chartTypeMap[resData.value.ChartInfo.ChartType];
|
|
|
+
|
|
|
+ let series=[]
|
|
|
+ let yAxis=[]
|
|
|
+ let xAxis = {}
|
|
|
+
|
|
|
+ let temYLeftArr=[]
|
|
|
+ let temYRightArr=[]
|
|
|
+ let temYLeftIndex,temYRightIndex;
|
|
|
+
|
|
|
+ let minAndMaxTimeTemArr=[]//存放所有指标的最大最小时间
|
|
|
+
|
|
|
+
|
|
|
+ data.forEach((item,index)=>{
|
|
|
+
|
|
|
+ //轴位置值相同的下标
|
|
|
+ let sameSideIndex = data.findIndex(i => i.IsAxis === item.IsAxis);
|
|
|
+
|
|
|
+ //堆叠图的yAxis必须一致 数据列所对应的y轴
|
|
|
+ let serie_yIndex = index;
|
|
|
+ if([3,4].includes(resData.value.ChartInfo.ChartType)) {
|
|
|
+ // 类型为堆叠图时公用第一个指标y轴
|
|
|
+ serie_yIndex = 0;
|
|
|
+ } else if(resData.value.ChartInfo.ChartType ===6 && ['areaspline','column'].includes(item.ChartStyle)) {
|
|
|
+ // 组合图找第一个堆叠柱状或面积的作为公用
|
|
|
+ serie_yIndex = data.findIndex(i => i.ChartStyle === item.ChartStyle);
|
|
|
+ }
|
|
|
+ //数据对应的y轴是公用轴则配置也共享
|
|
|
+ item.IsAxis = serie_yIndex === index ? item.IsAxis : data[serie_yIndex].IsAxis;
|
|
|
+ item.IsOrder = serie_yIndex === index ? item.IsOrder : data[serie_yIndex].IsOrder;
|
|
|
+
|
|
|
+ temYLeftIndex = [3,4].includes(resData.value.ChartInfo.ChartType)
|
|
|
+ ? (data[serie_yIndex].IsAxis ? serie_yIndex : -1)
|
|
|
+ : data.findIndex((item) => item.IsAxis);
|
|
|
+ temYRightIndex = [3,4].includes(resData.value.ChartInfo.ChartType)
|
|
|
+ ? (data[serie_yIndex].IsAxis ? -1 : serie_yIndex)
|
|
|
+ : data.findIndex((item) => !item.IsAxis);
|
|
|
+
|
|
|
+ let dynamic_title = item.EdbName;
|
|
|
+ let dynamic_arr = data.filter(
|
|
|
+ (item) => dynamic_title === item.EdbName
|
|
|
+ );
|
|
|
+ //处理数据列name
|
|
|
+ let temName= setDyncmicSerieName(item,dynamic_arr)
|
|
|
+
|
|
|
+ let seriesItemObj={
|
|
|
+ data:[],
|
|
|
+ dataGrouping:{
|
|
|
+ enabled:false
|
|
|
+ },
|
|
|
+ type: chartStyle || item.ChartStyle,
|
|
|
+ yAxis:serie_yIndex,
|
|
|
+ name:temName,
|
|
|
+ color: item.ChartColor,
|
|
|
+ lineWidth: (resData.value.ChartInfo.ChartType === 6 && item.ChartStyle === 'spline') ? Number(item.ChartWidth) : 0,
|
|
|
+ fillColor: (resData.value.ChartInfo.ChartType === 3 || (resData.value.ChartInfo.ChartType === 6 && item.ChartStyle === 'areaspline')) ? item.ChartColor : undefined,
|
|
|
+ visible:true,
|
|
|
+ LatestDate:item.LatestDate,
|
|
|
+ LatestValue:item.LatestValue
|
|
|
+ }
|
|
|
+ item.DataList = item.DataList || [];
|
|
|
+ for (let i of item.DataList) {
|
|
|
+ seriesItemObj.data.push([i.DataTimestamp, i.Value]);
|
|
|
+ }
|
|
|
+ series.push(seriesItemObj)
|
|
|
+
|
|
|
+
|
|
|
+ // 设置y轴
|
|
|
+ if(item.IsAxis){
|
|
|
+ temYLeftArr.push(item)
|
|
|
+ }else{
|
|
|
+ temYRightArr.push(item)
|
|
|
+ }
|
|
|
+
|
|
|
+ let yItem={
|
|
|
+ ...basicYAxis,
|
|
|
+ IsAxis:item.IsAxis,
|
|
|
+ labels: {
|
|
|
+ formatter: function (ctx) {
|
|
|
+ return sameSideIndex !== index ? '' : ctx.value;
|
|
|
+ },
|
|
|
+ align: 'center',
|
|
|
+ y:5,
|
|
|
+ },
|
|
|
+ // title: {
|
|
|
+ // text: `单位:${item.Unit}`,
|
|
|
+ // align: 'high',
|
|
|
+ // rotation: 0,
|
|
|
+ // y: 0,
|
|
|
+ // offset: -60,
|
|
|
+ // },
|
|
|
+ opposite: item.IsAxis === 0,
|
|
|
+ reversed: item.IsOrder,
|
|
|
+ min: item.MinData,
|
|
|
+ max: item.MaxData,
|
|
|
+ tickWidth: sameSideIndex !== index ? 0 : 1,
|
|
|
+ visible: serie_yIndex === index ? true : false,
|
|
|
+ chartEdbInfo:item//指标数据用于在保存时读取指标数据
|
|
|
+ }
|
|
|
+ yAxis.push(yItem)
|
|
|
+
|
|
|
+ if(item.DataList.length>0){
|
|
|
+ minAndMaxTimeTemArr.push(item.DataList[0].DataTimestamp)
|
|
|
+ minAndMaxTimeTemArr.push(item.DataList[item.DataList.length-1].DataTimestamp)
|
|
|
+ }
|
|
|
+
|
|
|
+ })
|
|
|
+
|
|
|
+ // 设置值
|
|
|
+ chartData.value.series=series
|
|
|
+
|
|
|
+
|
|
|
+ // 设置x轴
|
|
|
+ // 找出所有指标的最大和最小时间 分成6段
|
|
|
+ let minTime=Math.min.apply(null,minAndMaxTimeTemArr)
|
|
|
+ let maxTime=Math.max.apply(null,minAndMaxTimeTemArr)
|
|
|
+
|
|
|
+
|
|
|
+ const bool_time = xTimeDiffer(minTime,maxTime)
|
|
|
+ if(bool_time){
|
|
|
+ xAxis={
|
|
|
+ ...basicXAxis,
|
|
|
+ labels: {
|
|
|
+ formatter: (ctx)=> {
|
|
|
+ return Highcharts.dateFormat('%m/%d', ctx.value)
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ xAxis={
|
|
|
+ ...basicXAxis,
|
|
|
+ tickInterval:((maxTime-minTime)/6)/(24*3600*1000)>30?(maxTime-minTime)/6:24*3600*1000*30,
|
|
|
+ }
|
|
|
+
|
|
|
+ chartData.value.xAxis=[xAxis]
|
|
|
+
|
|
|
+
|
|
|
+ yAxis.forEach(item=>{
|
|
|
+ if(item.IsAxis){//左轴
|
|
|
+ hasLeftAxis.value=true
|
|
|
+ item.min=data[temYLeftIndex].MinData
|
|
|
+ item.max=data[temYLeftIndex].MaxData
|
|
|
+ axisLimitData.leftMin=data[temYLeftIndex].MinData
|
|
|
+ axisLimitData.leftMax=data[temYLeftIndex].MaxData
|
|
|
+ }else{
|
|
|
+ hasRightAxis.value=true
|
|
|
+ item.min=data[temYRightIndex].MinData
|
|
|
+ item.max=data[temYRightIndex].MaxData
|
|
|
+ axisLimitData.rightMin=data[temYRightIndex].MinData
|
|
|
+ axisLimitData.rightMax=data[temYRightIndex].MaxData
|
|
|
+ }
|
|
|
+ })
|
|
|
+ chartData.value.yAxis=yAxis
|
|
|
+ chartData.value.rangeSelector={ enabled: false}
|
|
|
+}
|
|
|
+
|
|
|
+/* 拼接数据列动态name */
|
|
|
+const setDyncmicSerieName = (item,dynamic_arr) => {
|
|
|
+ // 拼接配置 IsAxis左轴1 右轴0 IsOrder正序false 逆序true EdbInfoType是否是领先指标
|
|
|
+ let dynamic_tag =item.IsAxis && item.IsOrder && item.EdbInfoType
|
|
|
+ ? '(逆序)'
|
|
|
+ : !item.IsAxis && item.IsOrder && item.EdbInfoType
|
|
|
+ ? '(右轴,逆序)'
|
|
|
+ : !item.IsAxis && !item.IsOrder && item.EdbInfoType
|
|
|
+ ? '(右轴)'
|
|
|
+ : !item.IsAxis && !item.IsOrder && !item.EdbInfoType
|
|
|
+ ? `(右轴,领先${item.LeadValue}${item.LeadUnit})`
|
|
|
+ : !item.IsAxis && item.IsOrder && !item.EdbInfoType
|
|
|
+ ? `(右轴,逆序,领先${item.LeadValue}${item.LeadUnit})`
|
|
|
+ : item.IsAxis && item.IsOrder && !item.EdbInfoType
|
|
|
+ ? `(逆序,领先${item.LeadValue}${item.LeadUnit})`
|
|
|
+ : item.IsAxis && !item.IsOrder && !item.EdbInfoType
|
|
|
+ ? `(领先${item.LeadValue}${item.LeadUnit})`
|
|
|
+ : '';
|
|
|
+ let temName = dynamic_arr.length > 1
|
|
|
+ ? `${item.EdbName}(${item.SourceName})${dynamic_tag}`
|
|
|
+ : `${item.EdbName}${dynamic_tag}`
|
|
|
+ if(temName.length>20){
|
|
|
+ let temArr=[]
|
|
|
+ for(let i=0;i<temName.length/20;i++){
|
|
|
+ temArr.push(temName.slice(i*20,i*20+20))
|
|
|
+ }
|
|
|
+ // console.log(temArr);
|
|
|
+ temName=temArr.join('<br>')
|
|
|
+ }
|
|
|
|
|
|
+ return temName
|
|
|
}
|
|
|
|
|
|
//设置季节图配置
|
|
@@ -434,6 +698,13 @@ const setSeasonOpt=(data)=>{
|
|
|
align: 'center',
|
|
|
y:5
|
|
|
},
|
|
|
+ // title: {
|
|
|
+ // text: `单位:${data.Unit}`,
|
|
|
+ // align: 'high',
|
|
|
+ // rotation: 0,
|
|
|
+ // y: 5,
|
|
|
+ // offset: -60,
|
|
|
+ // },
|
|
|
max: Number(data.MaxData),
|
|
|
min: Number(data.MinData),
|
|
|
lineWidth: 1,
|
|
@@ -462,7 +733,7 @@ const setSeasonOpt=(data)=>{
|
|
|
// 农历
|
|
|
if(calendarType.value==='农历'){
|
|
|
let filterArr=data.DataList.List&&data.DataList.List.slice(1,data.DataList.List.length)||[]
|
|
|
- // console.log('aaa',filterArr);
|
|
|
+ console.log('aaa',filterArr);
|
|
|
filterArr.forEach((item,index)=>{
|
|
|
let seriesItem={
|
|
|
data:[],
|
|
@@ -472,7 +743,7 @@ const setSeasonOpt=(data)=>{
|
|
|
type:data.ChartStyle,
|
|
|
yAxis:index,
|
|
|
name:item.Year,
|
|
|
- color:colorsArr.slice(-filterArr.length)[index],
|
|
|
+ color:colorsArr.slice(-filterArr.length)[index],
|
|
|
visible:true
|
|
|
}
|
|
|
let temarr=item.Items||[]
|
|
@@ -490,6 +761,13 @@ const setSeasonOpt=(data)=>{
|
|
|
align: 'center',
|
|
|
y:5
|
|
|
},
|
|
|
+ // title: {
|
|
|
+ // text: `单位:${data.Unit}`,
|
|
|
+ // align: 'high',
|
|
|
+ // rotation: 0,
|
|
|
+ // y: 5,
|
|
|
+ // offset: -60,
|
|
|
+ // },
|
|
|
max: Number(data.MaxData),
|
|
|
min: Number(data.MinData),
|
|
|
lineWidth: 1,
|
|
@@ -599,6 +877,114 @@ const setSeasonOpt=(data)=>{
|
|
|
|
|
|
}
|
|
|
|
|
|
+/* 散点图 第一个指标值为x轴 第二个指标为y轴*/
|
|
|
+const setScatterOptions = (dataList) => {
|
|
|
+ const { ChartInfo } = resData.value;
|
|
|
+
|
|
|
+ // 取2个指标中日期相同的数据
|
|
|
+ const real_data = [];
|
|
|
+ let tmpData_date = {};//用来取点对应的日期
|
|
|
+ let data1 = _.cloneDeep(dataList)[0].DataList || [];
|
|
|
+ let data2 = _.cloneDeep(dataList)[1].DataList || [];
|
|
|
+ data1.forEach((_item) => {
|
|
|
+ data2.forEach((_item2) => {
|
|
|
+ if(_item.DataTimestamp === _item2.DataTimestamp) {
|
|
|
+ //日期
|
|
|
+ let itemIndex =_item.Value + "_" +_item2.Value
|
|
|
+ if(tmpData_date[itemIndex]) {
|
|
|
+ tmpData_date[itemIndex].push( moment(_item.DataTimestamp).format('YYYY/MM/DD'))
|
|
|
+ } else {
|
|
|
+ tmpData_date[itemIndex] = [moment(_item.DataTimestamp).format('YYYY/MM/DD')]
|
|
|
+ }
|
|
|
+
|
|
|
+ //值
|
|
|
+ real_data.push({
|
|
|
+ x: _item.Value,
|
|
|
+ y: _item2.Value
|
|
|
+ })
|
|
|
+ }
|
|
|
+ })
|
|
|
+ })
|
|
|
+ real_data.sort((x,y) => x-y);
|
|
|
+
|
|
|
+ //悬浮窗 拼接日期 原始指标名称
|
|
|
+ let tooltip = {
|
|
|
+ formatter: function() {
|
|
|
+ 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>
|
|
|
+ ${dataList[0].EdbName}: <span style="font-weight: 600"> ${this.x}</span><br>
|
|
|
+ ${dataList[1].EdbName}: <span style="font-weight: 600"> ${this.y}</span>
|
|
|
+ `
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ const { IsOrder,ChartColor } = dataList[0];
|
|
|
+ //y轴
|
|
|
+ let yAxis = {
|
|
|
+ // title: {
|
|
|
+ // text: `单位:${dataList[1].Unit}`,
|
|
|
+ // align: 'high',
|
|
|
+ // rotation: 0,
|
|
|
+ // y: 0,
|
|
|
+ // offset: -60,
|
|
|
+ // },
|
|
|
+ labels: {
|
|
|
+ formatter: function (ctx) {
|
|
|
+ return ctx.value;
|
|
|
+ },
|
|
|
+ align: 'center',
|
|
|
+ },
|
|
|
+ opposite: false,
|
|
|
+ reversed: IsOrder,
|
|
|
+ min: Number(dataList[0].MinData),
|
|
|
+ max: Number(dataList[0].MaxData),
|
|
|
+ tickWidth: 1,
|
|
|
+ tickLength: 5,
|
|
|
+ lineWidth: 1,
|
|
|
+ lineColor: '#bfbfbf',
|
|
|
+ tickColor: '#bfbfbf',
|
|
|
+ offset: 0,
|
|
|
+ visible: true,
|
|
|
+ gridLineWidth: 0,
|
|
|
+ tickPosition: 'inside',
|
|
|
+ endOnTick: false,
|
|
|
+ startOnTick: false,
|
|
|
+ showLastLabel: true,
|
|
|
+ tickPixelInterval: 50
|
|
|
+ }
|
|
|
+
|
|
|
+ //数据列
|
|
|
+ let series = {
|
|
|
+ data: [],
|
|
|
+ type: 'scatter',
|
|
|
+ name: `${ChartInfo.ChartName}${IsOrder ? '(逆序)' : ''}`,
|
|
|
+ color: ChartColor,
|
|
|
+ visible:true,
|
|
|
+ lineWidth: 0
|
|
|
+ }
|
|
|
+ real_data.forEach(_ => {
|
|
|
+ series.data.push([_.x,_.y])
|
|
|
+ })
|
|
|
+
|
|
|
+ chartData.value = {
|
|
|
+ title: {
|
|
|
+ text:''
|
|
|
+ },
|
|
|
+ series: [ series ],
|
|
|
+ yAxis,
|
|
|
+ xAxis: {
|
|
|
+ ...scatterXAxis,
|
|
|
+ // title: {
|
|
|
+ // text: `单位:${dataList[0].Unit}`,
|
|
|
+ // align: 'high',
|
|
|
+ // rotation: 0,
|
|
|
+ // x: 0,
|
|
|
+ // offset: 20,
|
|
|
+ // },
|
|
|
+ },
|
|
|
+ tooltip
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
// 查询范围为1年内 x轴显示为月/日 否则默认年/月
|
|
|
const xTimeDiffer=(minTime,maxTime)=>{
|
|
|
//年限差
|
|
@@ -664,7 +1050,7 @@ const handleGoSearch=()=>{
|
|
|
}
|
|
|
|
|
|
// 获取当前图表
|
|
|
-let searchVal=route.query.searchVal
|
|
|
+let searchVal=decodeURIComponent(route.query.searchVal)
|
|
|
let searchListData=ref([])//搜索的数据
|
|
|
const getSearchListData=async ()=>{
|
|
|
const res=await apiChartList({Keywords:searchVal,Page:1,Limit:10000,Authorization:route.query.token})
|
|
@@ -684,6 +1070,7 @@ const pageChange=async (type)=>{
|
|
|
if(!searchListData.value[index-1]){
|
|
|
Toast('当前已是第一张图')
|
|
|
}else{
|
|
|
+ calendarType.value=''
|
|
|
router.replace({
|
|
|
query:{
|
|
|
...route.query,
|
|
@@ -697,6 +1084,7 @@ const pageChange=async (type)=>{
|
|
|
if(!searchListData.value[index+1]){
|
|
|
Toast('当前已是最后一张图')
|
|
|
}else{
|
|
|
+ calendarType.value=''
|
|
|
router.replace({
|
|
|
query:{
|
|
|
...route.query,
|
|
@@ -730,6 +1118,7 @@ const pageChange=async (type)=>{
|
|
|
})
|
|
|
}, 500);
|
|
|
}
|
|
|
+ calendarType.value=''
|
|
|
router.replace({
|
|
|
query:{
|
|
|
...route.query,
|
|
@@ -753,7 +1142,7 @@ const pageChange=async (type)=>{
|
|
|
})
|
|
|
}, 500);
|
|
|
}
|
|
|
-
|
|
|
+ calendarType.value=''
|
|
|
router.replace({
|
|
|
query:{
|
|
|
...route.query,
|
|
@@ -779,7 +1168,7 @@ onMounted(()=>{
|
|
|
// 保存
|
|
|
const handleSaveChart=async ()=>{
|
|
|
let params={}
|
|
|
- if(resData.value.ChartInfo.ChartType===1){//曲线图
|
|
|
+ if(sameOptionType.value.includes(resData.value.ChartInfo.ChartType)){//曲线图
|
|
|
let arr=chartData.value.yAxis.map(item=>{
|
|
|
return {
|
|
|
ChartColor: item.chartEdbInfo.ChartColor,
|
|
@@ -840,7 +1229,7 @@ const handleRefreshChart=async ()=>{
|
|
|
setTimeout(() => {
|
|
|
Toast.success(res.msg)
|
|
|
}, 200);
|
|
|
- getChartInfo('init')
|
|
|
+ getChartInfo()
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -873,10 +1262,31 @@ const pageTouchmove=(e)=>{
|
|
|
|
|
|
event.preventDefault();//阻止页面移动
|
|
|
}
|
|
|
+
|
|
|
+// 生成海报所需跳转到小程序页面参数
|
|
|
+const code_scene=computed(()=>{
|
|
|
+ let obj= {
|
|
|
+ chartInfoId:ChartInfoId,
|
|
|
+ searchVal:decodeURIComponent(route.query.searchVal)||'',
|
|
|
+ MyChartId:route.query.MyChartId||'',
|
|
|
+ MyChartClassifyId:route.query.MyChartClassifyId||'',
|
|
|
+ from:'share'
|
|
|
+ }
|
|
|
+ return JSON.stringify(obj)
|
|
|
+})
|
|
|
+// 生成海报图片所需要的数据
|
|
|
+const posterParams=computed(()=>{
|
|
|
+ return {
|
|
|
+ chart_name:resData.value.ChartInfo.ChartName,
|
|
|
+ chart_image:resData.value.ChartInfo.ChartImage
|
|
|
+ }
|
|
|
+})
|
|
|
+
|
|
|
+
|
|
|
</script>
|
|
|
|
|
|
<template>
|
|
|
- <div class="chart-detail">
|
|
|
+ <div class="chart-detail" v-if="!loading&&!noauth">
|
|
|
<div class="chart-title">{{resData.ChartInfo.ChartName}}</div>
|
|
|
<div class="top-box">
|
|
|
<div class="flex calendar-box" style="float:left" @click="handleShowDate">
|
|
@@ -886,6 +1296,14 @@ const pageTouchmove=(e)=>{
|
|
|
<span class="date">{{endDate||'结束日期'}}</span>
|
|
|
</div>
|
|
|
<img class="icon" src="../../../assets/hzyb/chart/search.png" alt="" @click="handleGoSearch">
|
|
|
+ <share-poster
|
|
|
+ :shareData="{
|
|
|
+ type:'chart_detail',
|
|
|
+ code_scene:code_scene,
|
|
|
+ code_page:'pages-chart/chartDetail',
|
|
|
+ data:posterParams
|
|
|
+ }"
|
|
|
+ ></share-poster>
|
|
|
<img class="icon" src="../../../assets/hzyb/chart/save.png" alt="" @click="handleSaveChart" v-if="canSave">
|
|
|
<img class="icon" src="../../../assets/hzyb/chart/refresh.png" alt="" @click="handleRefreshChart">
|
|
|
</div>
|
|
@@ -893,16 +1311,17 @@ const pageTouchmove=(e)=>{
|
|
|
<chartBox :options='chartData' v-if="!loading"></chartBox>
|
|
|
|
|
|
<div class="flex source-box">
|
|
|
- <span>来源:弘则研究</span>
|
|
|
- <div class="season-change-box" v-if="resData&&resData.ChartInfo.ChartType===2">
|
|
|
+ <div :style="{flex:resData&&resData.ChartInfo.ChartType===2?1:2}"><span v-if="resData&&resData.ChartInfo.ChartType!==2">来源:{{resData&&resData.ChartInfo.ChartSource}}</span></div>
|
|
|
+ <div class="season-change-box" style="flex:1" v-if="resData&&resData.ChartInfo.ChartType===2">
|
|
|
<span :class="calendarType==='农历'&&'active'" @click="calendarTypeChange('农历')">农历</span>
|
|
|
<span :class="calendarType==='公历'&&'active'" @click="calendarTypeChange('公历')">公历</span>
|
|
|
</div>
|
|
|
- <span style="color:#E3B377" @click="showLimit=true">上下限设置</span>
|
|
|
+ <span style="color:#E3B377;flex:1;text-align:right" @click="showLimit=true">上下限设置</span>
|
|
|
</div>
|
|
|
+ <div class="source-box" style="margin-top:5px" v-if="resData&&resData.ChartInfo.ChartType===2">来源:{{resData&&resData.ChartInfo.ChartSource}}</div>
|
|
|
|
|
|
<!-- 日期类型 -->
|
|
|
- <div class="date-type-box" v-if="resData&&resData.ChartInfo.ChartType===1">
|
|
|
+ <div class="date-type-box" v-if="resData&&sameOptionType.includes(resData.ChartInfo.ChartType)">
|
|
|
<div
|
|
|
:class="['item',item.value==dateType?'active':'']"
|
|
|
v-for="item in dateTypeList"
|
|
@@ -914,10 +1333,10 @@ const pageTouchmove=(e)=>{
|
|
|
<!-- 最新值 -->
|
|
|
<div class="latest-value-wrap" v-if="resData">
|
|
|
<p style="margin-bottom:10px">最新数值</p>
|
|
|
- <ul class="list" v-if="resData.ChartInfo.ChartType===1">
|
|
|
+ <ul class="list" v-if="sameOptionType.includes(resData.ChartInfo.ChartType)">
|
|
|
<li v-for="item in chartData.series" :key="item.name">
|
|
|
<p style="color:#333">{{moment(item.LatestDate).format('YYYY-MM-DD')}}</p>
|
|
|
- <p :style="{color:item.color,flex:1}">{{item.name}}</p>
|
|
|
+ <p :style="{color:item.color,flex:1}">{{item.name.length>20?item.name.replace(/<br>/g,''):item.name}}</p>
|
|
|
<p style="color:#1F243A">{{item.LatestValue}}</p>
|
|
|
</li>
|
|
|
</ul>
|
|
@@ -932,6 +1351,7 @@ const pageTouchmove=(e)=>{
|
|
|
|
|
|
<!-- 上一张下一张图切换 -->
|
|
|
<div
|
|
|
+ v-if="$route.query.from!='share'"
|
|
|
class="change-page-wrap"
|
|
|
:style="{left:pageBoxPosition.left+'px',top:pageBoxPosition.top+'px'}"
|
|
|
@touchmove.stop="pageTouchmove"
|
|
@@ -989,12 +1409,15 @@ const pageTouchmove=(e)=>{
|
|
|
</div>
|
|
|
</Popup>
|
|
|
</div>
|
|
|
-
|
|
|
-
|
|
|
+ <!-- 无权限 -->
|
|
|
+ <noAuth v-if="noauth" :data="noAuthData"></noAuth>
|
|
|
|
|
|
</template>
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
+::v-deep(.highcharts-axis-title) {
|
|
|
+ font-size: 20px;
|
|
|
+}
|
|
|
.chart-detail{
|
|
|
.flex{
|
|
|
display: flex;
|
|
@@ -1032,7 +1455,7 @@ const pageTouchmove=(e)=>{
|
|
|
float: right;
|
|
|
width: 40px;
|
|
|
height: 40px;
|
|
|
- margin-left: 50px;
|
|
|
+ margin-left: 30px;
|
|
|
}
|
|
|
}
|
|
|
.select-date-box-head{
|