123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816 |
- <template>
- <div>
- <el-dialog
- :visible.sync="isCreateChart"
- :close-on-click-modal="false"
- :modal-append-to-body='false'
- @close="close"
- custom-class="create-chart-container"
- top="10vh"
- center
- width="80%"
- v-dialogDrag>
- <div slot="title" style="display: flex; align-items: center">
- <span style="font-size: 16px;">{{currentLang==='en'?(chartInfo.EdbName||chartInfo.EdbNameEn):chartInfo.EdbName}}</span>
- </div>
- <h2 style="margin-bottom: 15px" v-if="!isOnlyShowBaseChart">{{chart_type===1 ? '曲线图' : '季节性图'}}</h2>
- <div class="header">
- <template v-if="chart_type===1">
- <el-button
- type="primary"
- v-for="item in yearSelector"
- :key="item.value"
- size="medium"
- :plain="item.value !== year_select"
- class="year-btn"
- @click.native="changeYear(item)"
- >{{ item.name }}</el-button
- >
- <el-button type="text" class="btn-sty" @click="openDateDia">{{
- dateTip
- }}</el-button>
- <el-button type="text" class="btn-sty" @click="showOnChart('toggle')" v-if="!isOnlyShowBaseChart">{{ isShowOnyearData ? '隐藏同比图' : '展示同比图'}}</el-button>
- </template>
- <!-- 季节图时间选择 -->
-
- <div v-else-if="chart_type === 2" @click="openDateDia" class="date-setting">
- {{ season_year && season_year.length>0 ? season_year[0]+'~'+season_year[1]:"年份日期选择" }}
- </div>
- <span class="change-chart-btn" @click="chartTypeChange" v-if="!isOnlyShowBaseChart">切换{{chart_type==1?'季节性图':'曲线图'}}</span>
- </div>
- <div class="min-wrapper">
- <div class="chartWrapper" id="chartWrapper" v-if="showChart">
- <Chart :options="options" ref="chartRef"/>
- <div class="range-cont left" v-if="leftIndex != -1">
- <el-input
- style="width: 80px; display: block"
- size="mini"
- type="number"
- placeholder="上限"
- v-model="chartInfo.MaxValue"
- @change="() => { chart_type===1?setOptions():setSeasonOptions() }"
- />
- <el-input
- class="min-data-input"
- size="mini"
- type="number"
- placeholder="下限"
- v-model="chartInfo.MinValue"
- @change="() => { chart_type===1?setOptions():setSeasonOptions() }"
- />
- </div>
- <div class="range-cont right" v-if="isShowOnyearData">
- <el-input
- style="width: 80px; display: block"
- size="mini"
- type="number"
- placeholder="上限"
- v-model="limitData.rightMax"
- @change="changeLimit"
- />
- <el-input
- class="min-data-input"
- size="mini"
- type="number"
- placeholder="下限"
- v-model="limitData.rightMin"
- @change="changeLimit"
- />
- </div>
- <!-- 公历农历切换 只用于季节性图 -->
- <div style="text-align:center;">
- <el-radio-group
- v-model="calendar_type"
- class="calendar-cont"
- v-if="chart_type===2"
- @change="getDataByPath"
- >
- <el-radio-button label="公历" />
- <el-radio-button label="农历" />
- </el-radio-group>
- </div>
- </div>
- </div>
- </el-dialog>
-
- <!-- 自定义时间段选择弹窗 -->
- <DateChooseDia
- :isDateDia="isDateDia"
- :dateForm="dateForm"
- @cancel="isDateDia = false"
- @dateBack="dataChangeBack"
- />
- </div>
- </template>
- <script>
- import { dataBaseInterface } from '@/api/api.js';
- import * as supplyApi from '@/api/modules/supplyApi.js';
- import { yearSelector,defaultOpts,seasonOptions } from '@/utils/defaultOptions'
- import Chart from '../components/chart'
- import DateChooseDia from '../components/DateChooseDia';
- import Highcharts from 'highcharts'
- export default {
- components: { Chart,DateChooseDia },
- props: {
- isCreateChart: {
- type: Boolean,
- },
- edbid: {
- type: Number,
- },
- // 语言 ch/en
- chartLang:{
- type: String,
- default:()=> 'zh'
- }
- },
- computed: {
- // 同比,环比,环差,超季节性、 残差展示基础图
- isOnlyShowBaseChart() {
- return [6,12,13,35,37].includes(this.chartInfo.Source)
- },
- currentLang() {
- return this.$store.state.lang
- }
- },
- watch: {
- isCreateChart(newval) {
- newval && this.getDataByPath()
- },
- },
- data () {
- return {
- year_select: 10, //年份选择项 默认全部
- yearSelector: [
- {
- name: '全部',
- value: 10,
- },
- ...yearSelector,
- ],
- select_date: [], // 自定义时间段项
- options: {},
- showChart: false,
- chart_type: 1,// 1曲线 2季节
- isShowOnyearData: false, //是否展示同比
- chartInfo: {},
- dataList: [],
- leftIndex: 0,
- season_year:[],//季节图日期选择
- calendar_type:'公历',
- /* 日期弹窗 */
- isDateDia: false, // 时间段弹窗
- dateForm: {},
- dateTip: /* '请选择时间段' */this.$t('Chart.choose_time'),
- limitData: {
- letMin: 0,
- leftMax: 0,
- rightMin: 0,
- rightMax: 0,
- },
- oldOptions: {
- MinValue: 0,
- MaxValue: 0
- },
- // 最近几年
- count_year:'',
- };
- },
- methods: {
-
- getDataByPath() {
- const apiMap = {
- '/database': this.getEdbData,
- '/analyseVariety': this.getPlantEdbData
- }
- apiMap[this.$route.path]()
- },
- /* 获取指标所有数据 */
- async getEdbData() {
- let params = {
- EdbInfoId: this.edbid,
- DateType: this.year_select,
- StartYear:this.count_year || 0,
- StartDate: this.select_date[0] || '',
- EndDate: this.select_date[1] || ''
- }
- const res = this.chart_type === 1
- ? await dataBaseInterface.ebd_data(params)
- : await dataBaseInterface.ebdSeasonData({
- EdbInfoId: this.edbid,
- DateType: this.year_select,
- Calendar: this.calendar_type,
- StartYear:this.count_year || 0,
- StartDate: this.season_year[0] || '',
- EndDate: this.season_year[1] || '',
- })
-
- if(res.Ret !== 200) return
- const { DataList,EdbInfo } = res.Data;
- // DataList 表格数据列表
- this.chartInfo = (this.oldOptions.MinValue || this.oldOptions.MaxValue) ? {
- ...EdbInfo,
- ...this.oldOptions
- } : EdbInfo;
- this.tableData = DataList || [];
- if(this.chart_type==2 && !(this.season_year && this.season_year.length>0)){
- // 第一次进行初始化显示
- let latestYear = parseInt(this.chartInfo.LatestDate.substring(0,4))
- this.season_year = [`${latestYear-this.count_year+1}-01-01`,`${latestYear}-12-31`];
- }
- this.chart_type === 1 ? this.setOptions() : this.setSeasonOptions();
- this.showChart = true;
- },
- /* 获取装置指标数据 */
- async getPlantEdbData() {
- let params = {
- VarietyEdbId: this.edbid,
- DateType: this.year_select,
- StartYear:this.count_year || 0,
- StartDate: this.select_date[0] || '',
- EndDate: this.select_date[1] || ''
- }
- const res = this.chart_type === 1
- ? await supplyApi.getEdbDetailData(params)
- : await supplyApi.getEdbSeasonData({
- VarietyEdbId: this.edbid,
- DateType: this.year_select,
- Calendar: this.calendar_type,
- StartYear:this.count_year || 0,
- StartDate: this.season_year[0] || '',
- EndDate: this.season_year[1] || '',
- })
-
- if(res.Ret !== 200) return
- const { DataList,EdbInfo } = res.Data;
- // DataList 表格数据列表
- this.chartInfo = (this.oldOptions.MinValue || this.oldOptions.MinValue) ? {
- ...EdbInfo,
- ...this.oldOptions
- } : EdbInfo;
- this.tableData = DataList || [];
- if(this.chart_type==2 && !(this.season_year && this.season_year.length>0)){
- // 第一次进行初始化显示
- let latestYear = parseInt(this.chartInfo.LatestDate.substring(0,4))
- this.season_year = [`${latestYear-this.count_year+1}-01-01`,`${latestYear}-12-31`];
- }
- this.chart_type === 1 ? this.setOptions() : this.setSeasonOptions();
- this.showChart = true;
- },
- /* 年份改变 重新刷新图表接口 保存当前的图表配置和上下限 只改变图表 */
- changeYear(item) {
- this.year_select = item.value;
- this.select_date = [];
- this.dateTip = /* '请选择时间段' */this.$t('Chart.choose_time');
- const { MinValue,MaxValue } = this.chartInfo;
- this.oldOptions = {
- MinValue: Number(MinValue),
- MaxValue: Number(MaxValue)
- }
- this.getDataByPath()
- },
- /* 打开时间段弹窗 */
- openDateDia() {
- // 自定义时间段回显
- let selectDateStart = this.chart_type === 2?this.season_year[0]:this.select_date[0]
- let selectDateEnd = this.chart_type === 2?this.season_year[1]:this.select_date[1]
- this.dateForm = {
- date_type: this.year_select,
- start_date:
- this.year_select === 5 || this.year_select === 6
- ? selectDateStart
- : '',
- end_date: this.year_select === 5 ? selectDateEnd : '',
- count_year: this.year_select === 20 ? this.count_year : ''
- };
- this.isDateDia = true;
- },
- /* 保存完自定义日期 刷新数据 保存当前的图表配置和上下限 只改变图表*/
- dataChangeBack(data) {
- this.year_select = data.dateType;
- this.isDateDia = false;
- this.select_date = [data.start_date, data.end_date];
- this.count_year = data.count_year
- this.dateTip =
- data.dateType === 5
- ? `${data.start_date}~${data.end_date}`
- : data.dateType === 6
- ?/* `${data.start_date}~至今` */ this.$t('Chart.data_tip_since',{date:data.start_date})
- :/* `最近${this.count_year}年` */ this.$t('Chart.date_tip_count',{year:this.count_year})
- if(this.chart_type === 2){
- let latestYear = parseInt(this.chartInfo.LatestDate.substring(0,4))
- let dateStart = data.start_date
- let dateEnd = data.end_date
- if(data.dateType==20){
- dateStart = `${latestYear-this.count_year+1}-01-01`
- dateEnd = `${latestYear}-12-31`
- }else if(data.dateType==6){
- dateEnd = this.chartInfo.LatestDate
- }
- this.season_year = [dateStart, dateEnd];
- }
- const { MinValue,MaxValue } = this.chartInfo;
- this.oldOptions = {
- MinValue,
- MaxValue
- }
- this.getDataByPath();
- },
- chartTypeChange() {
- this.chart_type = this.chart_type===1 ? 2 : 1;
- this.init('no_reset')
- this.getDataByPath();
- },
- // 展示同比图
- async showOnChart(scene='') {
- if(scene==='toggle') {
- this.isShowOnyearData = this.isShowOnyearData ? false : true;
- if(!this.isShowOnyearData){
- const { MinValue,MaxValue } = this.chartInfo;
- this.oldOptions = {
- MinValue: Number(MinValue),
- MaxValue: Number(MaxValue)
- }
- this.getDataByPath();
- }
- }
- if(!this.isShowOnyearData) return;
- let params = {
- DateType: this.year_select,
- StartYear:this.count_year || 0,
- StartDate: this.select_date[0] || '',
- EndDate: this.select_date[1] || ''
- }
- const res = this.$route.path === '/analyseVariety'
- ? await supplyApi.getEdbOnyearData({VarietyEdbId:this.edbid,...params})
- : await dataBaseInterface.edbOnyearData({EdbInfoId: this.edbid,...params});
- if(res.Ret !== 200) return
- const { EdbInfo,DataList } = res.Data;
- if(!this.limitData.rightMin && !this.limitData.rightMax) {
- this.limitData.rightMin = Number(EdbInfo.MinValue);
- this.limitData.rightMax = Number(EdbInfo.MaxValue);
- }
- this.options.yAxis.push({
- title: {
- text: '',
- align: 'high',
- rotation: 0,
- y: -15,
- offset: 0
- },
- labels: {
- formatter: function (ctx) {
- return ctx.value;
- },
- align: 'center',
- },
- min: Number(this.limitData.rightMin),
- max: Number(this.limitData.rightMax),
- ...seasonOptions.yAxis,
- opposite: true,
- })
- this.options.series.push({
- data: DataList.map(item=>([item.DataTimestamp, item.Value])),
- type: 'spline',
- yAxis: 1,
- name: this.currentLang=='zh'?`${EdbInfo.EdbName}同比`:`${EdbInfo.EdbNameEn}同比`,
- lineWidth: 1
- })
- console.log(this.options.series)
- },
- changeLimit() {
- const { rightMax,rightMin } = this.limitData;
- this.options.yAxis[1].max = Number(rightMax);
- this.options.yAxis[1].min = Number(rightMin);
- },
- /* 图表配置 曲线*/
- setOptions() {
- const chartData = _.cloneDeep(this.tableData);
- //拼接标题 数据列
- let data = [],
- ydata = [];
- //y轴
- let yTitleText = this.currentLang=='zh'?this.chartInfo.Unit:this.chartInfo.UnitEn
- let yItem = {
- title: {
- text: yTitleText,
- align: 'high',
- rotation: 0,
- y: -15,
- offset: 0
- },
- labels: {
- formatter: function (ctx) {
- return ctx.value;
- },
- align: 'center',
- },
- min: Number(this.chartInfo.MinValue),
- max: Number(this.chartInfo.MaxValue),
- ...seasonOptions.yAxis,
- };
- // 图例名称和图例文字样式
- let dataName = this.currentLang=='zh'?this.chartInfo.EdbName:this.chartInfo.EdbNameEn
- let color = this.currentLang=='en'&&(!this.chartInfo.EdbNameEn)?'#999':'#333'
- let legend={
- ...defaultOpts.legend,
- itemStyle: {
- color
- },
- }
- //数据列
- let obj = {
- data: [],
- type: 'spline',
- yAxis: 0,
- name: dataName,
- lineWidth: 3
- };
- chartData.forEach((item, index) => {
- obj.data.push([item.DataTimestamp, item.Value]);
- });
-
- data.push(obj);
- ydata.push(yItem);
- // 范围为1年内 x轴显示为月/日 否则默认年/月
- let xAxis = {};
- const bool_time = this.xTimeDiffer();
- xAxis = bool_time
- ? {
- ...defaultOpts.xAxis,
- labels: {
- formatter: function (ctx) {
- return Highcharts.dateFormat('%m/%d', ctx.value);
- },
- },
- }
- : {
- ...defaultOpts.xAxis,
- labels: {
- formatter: function (ctx) {
- return Highcharts.dateFormat('%y/%m', ctx.value);
- },
- },
- };
- this.options = {
- series: data,
- yAxis: ydata,
- xAxis,
- legend
- };
- this.showOnChart();
- },
- /* 季节图 */
- setSeasonOptions() {
- console.log(this.tableData)
- const chartData = _.cloneDeep(this.tableData);
- let seasonYdata = [],
- seasonData = [];
- /* 公历数据处理 处理数据列 y轴 */
- if (this.calendar_type === '公历')
- for (let j of chartData) {
-
- let serie_item = {
- data: [],
- type: 'spline',
- yAxis: 0,
- name: j.Year,
- };
- const data_array = this.calendar_type === '农历'?_.cloneDeep(j.Items):_.cloneDeep(j.DataList);
- data_array &&
- data_array.forEach((item) => {
- serie_item.data.push([item.DataTimestamp, item.Value]);
- });
- const index = chartData.findIndex(
- (item) => item.Year === j.Year
- );
- const s_yItem = {
- labels: {
- formatter: function () {
- let val = this.value;
- return index !== 0 ? '' : val;
- },
- align: 'center',
- },
- title: {
- text: this.currentLang=='zh'?this.chartInfo.Unit:this.chartInfo.UnitEn,
- align: 'high',
- rotation: 0,
- y: -15,
- offset: -(12 * this.chartInfo.Unit.length),
- },
- max: Number(this.chartInfo.MaxValue),
- min: Number(this.chartInfo.MinValue),
- ...seasonOptions.yAxis,
- };
- seasonData.push(serie_item);
- seasonYdata.push(s_yItem);
- }
- /* 农历数据处理 */
- let filterArr =
- this.calendar_type === '农历'
- ? chartData.List.filter((item, index) => index > 0)
- : [];
- if (this.calendar_type === '农历')
- for (let j of filterArr) {
- let serie_item = {
- data: [],
- type: 'spline',
- yAxis: 0,
- name: j.Year
- };
- const data_array = _.cloneDeep(j.Items);
- data_array &&
- data_array.forEach((item) => {
- serie_item.data.push([item.DataTimestamp, item.Value]);
- });
- const index = filterArr.findIndex((item) => item.Year === j.Year);
- const s_yItem = {
- labels: {
- formatter: function () {
- let val = this.value;
- return index !== 0 ? '' : val;
- },
- align: 'center',
- },
- title: {
- text: this.currentLang=='zh'?this.chartInfo.Unit:this.chartInfo.UnitEn,
- align: 'high',
- rotation: 0,
- y: -15,
- offset: -(12 * this.chartInfo.Unit.length),
- },
- max: Number(this.chartInfo.MaxValue),
- min: Number(this.chartInfo.MinValue),
- ...seasonOptions.yAxis,
- };
- seasonData.push(serie_item);
- seasonYdata.push(s_yItem);
- }
- /* x轴显示月日 提示框显示月日*/
- defaultOpts.xAxis.labels = {
- formatter: function () {
- return Highcharts.dateFormat('%m/%d', this.value);
- },
- };
-
- const tooltip = {
- ...defaultOpts.tooltip,
- dateTimeLabelFormats: {
- // 时间格式化字符
- day: '%m/%d',
- week: '%m/%d',
- month: '%m/%d',
- year: '%m/%d',
- },
- xDateFormat: '%m/%d',
- }
- let rangeSelector =
- this.calendar_type === '农历'
- ? {
- 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:
- this.calendar_type === '公历'
- ? seasonOptions.colors.slice(-chartData.length)
- : seasonOptions.colors.slice(-filterArr.length),
- series: seasonData,
- yAxis: seasonYdata,
- rangeSelector,
- tooltip
- };
- console.log(this.options)
- },
- /* 查询范围为1年内 x轴显示为月/日 否则默认年/月 */
- xTimeDiffer() {
- const end_date =
- this.year_select === 5
- ? this.select_date[1]
- : this.year_select === 6
- ? new Date()
- : '';
- //年限差
- const year_differ = this.$moment(end_date).diff(
- this.$moment(this.select_date[0]),
- 'years',
- true
- );
- // console.log(year_differ)
- if ([5, 6].includes(this.year_select) && year_differ <= 1) {
- return true;
- } else {
- return false;
- }
- },
- init(type='') {
- if(type!=='no_reset') this.chart_type = 1;
- this.isShowOnyearData = false;
- this.year_select = this.chart_type==1 ? 10 : 20;
- // 季节性图默认展示最近5年数据
- this.count_year=5
- this.select_date = [];
- this.options = {};
- this.dateTip = /* '请选择时间段' */this.$t('Chart.choose_time');
- this.oldOptions = {};
- this.calendar_type = '公历';
- this.season_year = [];
- this.limitData = {
- letMin: 0,
- leftMax: 0,
- rightMin: 0,
- rightMax: 0,
- }
- },
- close() {
- this.init()
- this.$emit('close')
- }
- },
- created() {},
- mounted() {},
- }
- </script>
- <style lang='scss'>
- .create-chart-container {
- min-width: 1200px;
- /* .el-dialog__header {
- background-color: #fff;
- .el-dialog__close {
- color: #333;
- }
- } */
- @media screen and (max-width: 1710px) {
- .min-data-input {
- margin-top: 220px;
- }
- }
- @media screen and (min-width: 1711px) {
-
- .min-data-input {
- margin-top: 308px;
- }
- }
- .el-dialog__body {
- padding: 20px 50px 30px;
- }
- .header {
- position: relative;
- .close {
- font-size: 26px;
- position: absolute;
- right: 0;
- top: 50%;
- transform: translateY(-50%);
- cursor: pointer;
- }
- .year-btn {
- min-width: 98px;
- font-size: 14px;
- margin-right: 5px;
- margin-bottom: 5px;
- }
- .btn-sty {
- font-size: 16px;
- padding: 10px;
- border: 1px solid #409eff;
- }
- .date-setting{
- width: 210px;
- // height: 40px;
- padding: 10px;
- border: 1px solid #DCDFE6;
- border-radius: 4px;
- cursor: pointer;
- font-size: 14px;
- color: #333333;
- box-sizing: border-box;
- }
- .change-chart-btn{
- float: right;
- color: #409eff;
- font-size: 16px;
- cursor: pointer;
- &::before{
- content:'';
- display: inline-block;
- width: 18px;
- height: 18px;
- background-image: url('~@/assets/img/chart_m/change.png');
- background-size: cover;
- position: relative;
- top: 4px;
- right: 4px;
- }
- }
- }
- .min-wrapper {
- .el-input__inner {
- padding: 0 0 0 5px;
- }
- margin: 40px 0;
- min-height: 400px;
- .chartWrapper {
- position: relative;
- padding: 0 100px;
- .range-cont {
- position: absolute;
- top: 8%;
- .min-data-input {
- width: 80px;
- display: block;
- }
- &.left {
- left: 0px;
- }
- &.right {
- right: 0;
- }
- }
- }
- }
- }
- </style>
|