You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

780 lines
26 KiB

<template>
<div class="shore-power-usage">
<!-- Average Mode -->
<template v-if="displayMode === 'average'">
<div class="left average" style="width: 40%;">
<div v-for="card in cards.slice(0, 3)" :key="card.id" class="card digital-twin-card--deep-blue"
@click="handleSelectCard(card.id)">
<div style="display: flex; align-items: center; justify-content: space-between;">
<div class="card-title">
<div class="vertical-line"></div>
<img src="@/assets/svgs/data.svg" class="title-icon" />
<span class="title-text">{{ card.title }}</span>
</div>
<div v-if="card.type === 'overview'" class="time-range-selector">
<button v-for="option in timeRangeOptions" :key="option.value"
:class="['time-range-btn', { active: timeRange === option.value }]"
@click.stop="handleTimeRangeChange(option.value)">
{{ option.label }}
</button>
</div>
<div v-if="card.type === 'chart'" class="show-value">
<span class="show-value-label">{{timeRangeOptions.find(option => option.value === timeRange)?.label ||
''}}总量:</span>
<div class="show-value-value">{{ chartData[timeRange][card.value] }}</div>
</div>
</div>
<div class="card-content">
<div v-if="card.type === 'overview'" class="overview-grid">
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].totalPower }}</div>
<div class="overview-label">累计用电(千瓦时)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].fuel }}</div>
<div class="overview-label">减少燃油(吨)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].co2 }}</div>
<div class="overview-label">减少二氧化碳排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].pm25 }}</div>
<div class="overview-label">减少PM2.5排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].nox }}</div>
<div class="overview-label">减少氮氧化物(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].so2 }}</div>
<div class="overview-label">减少二氧化硫(千克)</div>
</div>
</div>
<WaveLineChart v-else :chart-data="getChartData(card.id)" :title="card.title" :color="card.color" />
</div>
</div>
</div>
<div class="right average" style="width: 40%;">
<div v-for="card in cards.slice(3)" :key="card.id" class="card digital-twin-card--deep-blue"
@click="handleSelectCard(card.id)">
<div style="display: flex; align-items: center; justify-content: space-between;">
<div class="card-title">
<div class="vertical-line"></div>
<img src="@/assets/svgs/data.svg" class="title-icon" />
<span class="title-text">{{ card.title }}</span>
</div>
<div v-if="card.type === 'overview'" class="time-range-selector">
<button v-for="option in timeRangeOptions" :key="option.value"
:class="['time-range-btn', { active: timeRange === option.value }]"
@click.stop="handleTimeRangeChange(option.value)">
{{ option.label }}
</button>
</div>
<div v-if="card.type === 'chart'" class="show-value">
<span class="show-value-label">{{timeRangeOptions.find(option => option.value === timeRange)?.label ||
''}}总量:</span>
<div class="show-value-value">{{ chartData[timeRange][card.value] }}</div>
</div>
</div>
<div class="card-content">
<div v-if="card.type === 'overview'" class="overview-grid">
<div class="overview-item">
<div class="overview-value">{{ totalPower }}</div>
<div class="overview-label">累计用电(千瓦时)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ fuelReduction }}</div>
<div class="overview-label">减少燃油(吨)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ co2Reduction }}</div>
<div class="overview-label">减少二氧化碳排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ pm25Reduction }}</div>
<div class="overview-label">减少PM2.5排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ noxReduction }}</div>
<div class="overview-label">减少氮氧化物(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ so2Reduction }}</div>
<div class="overview-label">减少二氧化硫(千克)</div>
</div>
</div>
<WaveLineChart v-else :chart-data="getChartData(card.id)" :title="card.title" :color="card.color" />
</div>
</div>
</div>
</template>
<!-- Magnify Mode -->
<div v-if="displayMode === 'magnify'" class="magnify">
<div class="big">
<div v-for="card in cards.filter(c => c.id === selectedCard)" :key="card.id"
class="card digital-twin-card--deep-blue" @click="handleSelectCard(card.id)">
<div style="display: flex; align-items: center; justify-content: space-between;">
<div class="card-title">
<div class="vertical-line"></div>
<img src="@/assets/svgs/data.svg" class="title-icon" />
<span class="title-text">{{ card.title }}</span>
</div>
<div v-if="card.type === 'overview'" class="time-range-selector">
<button v-for="option in timeRangeOptions" :key="option.value"
:class="['time-range-btn', { active: timeRange === option.value }]"
@click.stop="handleTimeRangeChange(option.value)">
{{ option.label }}
</button>
</div>
<div v-if="card.type === 'chart'" class="show-value">
<span class="show-value-label">{{timeRangeOptions.find(option => option.value === timeRange)?.label ||
''}}总量:</span>
<div class="show-value-value">{{ chartData[timeRange][card.value] }}</div>
</div>
</div>
<div class="card-content">
<div v-if="card.type === 'overview'" class="overview-grid">
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].totalPower }}</div>
<div class="overview-label">累计用电(千瓦时)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].fuel }}</div>
<div class="overview-label">减少燃油(吨)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].co2 }}</div>
<div class="overview-label">减少二氧化碳排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].pm25 }}</div>
<div class="overview-label">减少PM2.5排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].nox }}</div>
<div class="overview-label">减少氮氧化物(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].so2 }}</div>
<div class="overview-label">减少二氧化硫(千克)</div>
</div>
</div>
<WaveLineChart v-else :chart-data="getChartData(card.id)" :title="card.title" :color="card.color"
:magnify-mode="true" />
</div>
</div>
</div>
<div class="one-row">
<div v-for="card in cards.filter(c => c.id !== selectedCard)" :key="card.id"
class="card digital-twin-card--deep-blue" @click="handleSelectCard(card.id)">
<div style="display: flex; align-items: center; justify-content: space-between;">
<div class="card-title">
<div class="vertical-line"></div>
<img src="@/assets/svgs/data.svg" class="title-icon" />
<span class="title-text">{{ card.title }}</span>
</div>
<div v-if="card.type === 'overview'" class="time-range-selector">
<button v-for="option in timeRangeOptions" :key="option.value"
:class="['time-range-btn', { active: timeRange === option.value }]"
@click.stop="handleTimeRangeChange(option.value)">
{{ option.label }}
</button>
</div>
<div v-if="card.type === 'chart'" class="show-value">
<span class="show-value-label">{{timeRangeOptions.find(option => option.value === timeRange)?.label ||
''}}总量:</span>
<div class="show-value-value">{{ chartData[timeRange][card.value] }}</div>
</div>
</div>
<div class="card-content">
<div v-if="card.type === 'overview'" class="overview-grid">
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].totalPower }}</div>
<div class="overview-label">累计用电(千瓦时)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].fuel }}</div>
<div class="overview-label">减少燃油(吨)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].co2 }}</div>
<div class="overview-label">减少二氧化碳排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].pm25 }}</div>
<div class="overview-label">减少PM2.5排放(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].nox }}</div>
<div class="overview-label">减少氮氧化物(千克)</div>
</div>
<div class="overview-item">
<div class="overview-value">{{ chartData[timeRange].so2 }}</div>
<div class="overview-label">减少二氧化硫(千克)</div>
</div>
</div>
<WaveLineChart v-else :chart-data="getChartData(card.id)" :title="card.title" :color="card.color" />
</div>
</div>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, onBeforeUnmount } from 'vue'
import WaveLineChart from './charts/WaveLineChart.vue'
import { MapApi } from "@/api/shorepower/map";
import dayjs from 'dayjs';
// 定义组件属性
interface ChartDataItem {
name: string;
value: number;
}
interface CardInfo {
id: string;
title: string;
type: 'overview' | 'chart';
unit?: string;
color?: string;
value: string;
}
interface ShorePowerUsageData {
totalPower: number; // 累计用电千瓦时
fuelReduction: number; // 减少燃油
co2Reduction: number; // 减少二氧化碳排放千克
pm25Reduction: number; // 减少PM2.5排放千克
noxReduction: number; // 减少氮氧化物千克
so2Reduction: number; // 减少二氧化硫千克
}
export interface MeasureDataRealtimeRespVO {
/**
* 创建时间
*/
createTime: Date;
/**
* 设备编号code
*/
deviceCode: string;
/**
* 设备编号
*/
deviceId: number;
/**
* 设备名称
*/
deviceName: string;
/**
* 设备状态
*/
deviceStatus: number;
/**
* 编号
*/
id: number;
/**
* 增量费用
*/
incrementCost?: number;
/**
* 增量值
*/
incrementValue: number;
/**
* 采集时间
*/
measureTime: Date;
/**
* 采集值
*/
measureValue: number;
}
export interface deviceData {
[key: string]: MeasureDataRealtimeRespVO
}
// const props = defineProps<Props>()
const totalPower = ref<string>('0')
const fuelReduction = ref<string>('0')
const co2Reduction = ref<string>('0')
const pm25Reduction = ref<string>('0')
const noxReduction = ref<string>('0')
const so2Reduction = ref<string>('0')
const displayMode = ref<'average' | 'magnify'>('magnify')
const selectedCard = ref<string>('overview')
// 创建统一的模拟数据源
const commonChartData = ref<ChartDataItem[]>([])
// 实时数据图表数组
const totalPowerChartData = ref<ChartDataItem[]>([])
const co2ReductionChartData = ref<ChartDataItem[]>([])
const so2ReductionChartData = ref<ChartDataItem[]>([])
const noxReductionChartData = ref<ChartDataItem[]>([])
const pm25ReductionChartData = ref<ChartDataItem[]>([])
const fuelReductionChartData = ref<ChartDataItem[]>([])
const chartData = ref({
'realtime': {
'totalPower': 0,
'fuel': 0,
'co2': 0,
'pm25': 0,
'nox': 0,
'so2': 0,
},
'day': {
'totalPower': 0,
'fuel': 0,
'co2': 0,
'pm25': 0,
'nox': 0,
'so2': 0,
},
'week': {
'totalPower': 0,
'fuel': 0,
'co2': 0,
'pm25': 0,
'nox': 0,
'so2': 0,
},
'month': {
'totalPower': 0,
'fuel': 0,
'co2': 0,
'pm25': 0,
'nox': 0,
'so2': 0,
},
'quarter': {
'totalPower': 0,
'fuel': 0,
'co2': 0,
'pm25': 0,
'nox': 0,
'so2': 0,
},
'year': {
'totalPower': 0,
'fuel': 0,
'co2': 0,
'pm25': 0,
'nox': 0,
'so2': 0,
},
})
// 获取对应卡片的图表数据
const getChartData = (cardId: string): ChartDataItem[] => {
switch (cardId) {
case 'overview':
return totalPowerChartData.value
case 'fuel':
return fuelReductionChartData.value
case 'co2':
return co2ReductionChartData.value
case 'pm25':
return pm25ReductionChartData.value
case 'nox':
return noxReductionChartData.value
case 'so2':
return so2ReductionChartData.value
default:
return commonChartData.value
}
}
// 时间范围选项
const timeRange = ref<'realtime' | 'day' | 'week' | 'month' | 'quarter' | 'year'>('realtime')
// 时间范围选项定义
const timeRangeOptions = [
{ value: 'realtime', label: '实时' },
{ value: 'day', label: '日' },
{ value: 'week', label: '周' },
{ value: 'month', label: '月' },
// { value: 'quarter', label: '季' },
{ value: 'year', label: '年' }
]
// 处理时间范围选择
const handleTimeRangeChange = (range: 'realtime' | 'day' | 'week' | 'month' | 'quarter' | 'year') => {
timeRange.value = range
// 这里可以添加根据时间范围切换数据源的逻辑
}
// 定义卡片信息
const cards = ref<CardInfo[]>([
{
id: 'overview',
title: '总览',
type: 'overview'
},
{
id: 'fuel',
title: '减少燃油(吨)',
type: 'chart',
unit: '吨',
color: '#4CAF50', // 绿色 - 代表节能、清洁能源
value: 'fuel'
},
{
id: 'co2',
title: '减少CO₂排放(千克)',
type: 'chart',
unit: '千克',
color: '#2E7D32', // 深绿色/蓝绿 - 温室气体(国际常用低碳绿色系)
value: 'co2'
},
{
id: 'pm25',
title: '减少PM2.5排放(千克)',
type: 'chart',
unit: '千克',
color: '#9E9E9E', // 灰色 - 颗粒物(国际通用)
value: 'pm25'
},
{
id: 'nox',
title: '减少NOₓ排放(千克)',
type: 'chart',
unit: '千克',
color: '#FF9800', // 橙色 - 氮氧化物(国际通用)
value: 'nox'
},
{
id: 'so2',
title: '减少SO₂排放(千克)',
type: 'chart',
unit: '千克',
color: '#FFEB3B', // 黄色 - 二氧化硫(国际通用)
value: 'so2'
}
])
const handleSelectCard = (cardId: string) => {
// 如果当前已经是放大模式且点击的是当前选中的大卡片,则返回平均模式
if (displayMode.value === 'magnify' && selectedCard.value === cardId) {
displayMode.value = 'average'
} else {
// 否则进入放大模式并选中指定卡片
displayMode.value = 'magnify'
selectedCard.value = cardId
}
}
// 生成初始数据
const generateInitialData = () => {
const now = new Date()
const data: ChartDataItem[] = []
// 生成过去5分钟的数据,每10秒一条
for (let i = 29; i >= 0; i--) {
const time = new Date(now.getTime() - i * 10000)
data.push({
name: time.toLocaleTimeString(),
value: Math.floor(Math.random() * 100) + 10
})
}
return data
}
// 定时器引用
let timer: ReturnType<typeof setInterval>;
// 追加新数据
const appendNewData = () => {
const now = new Date()
commonChartData.value.push({
name: now.toLocaleTimeString(),
value: Math.floor(Math.random() * 100) + 10
})
}
const handleGetRealTimeAllData = async () => {
try {
const res: deviceData = await MapApi.getRealtimeAllData({})
const ids = Object.values(res).map(item => item.id);
console.log(ids);
const params = {
ids: ids.join(','),
// year: new Date().getFullYear()
}
const yearDataRes: deviceData = await MapApi.getYearDataByIdList(params);
const weekDataRes: deviceData = await MapApi.getWeekDataByIdList(params);
const monthDataRes: deviceData = await MapApi.getMonthDataByIdList(params);
const dayDataRes: deviceData = await MapApi.getDayDataByIdList(params);
// const quarterDataRes: deviceData = await MapApi.getQuarterDataByIdList(params);
const realTimeSum = Object.values(res).reduce((acc, item) => acc + item.measureValue, 0);
const daySum = realTimeSum - Object.values(dayDataRes).reduce((acc, item) => acc + item.measureValue, 0);
const weekSum = realTimeSum - Object.values(weekDataRes).reduce((acc, item) => acc + item.measureValue, 0);
const monthSum = realTimeSum - Object.values(monthDataRes).reduce((acc, item) => acc + item.measureValue, 0);
// const quarterSum = Object.values(quarterDataRes).reduce((acc, item) => acc + item.measureValue, 0);
const yearSum = realTimeSum - Object.values(yearDataRes).reduce((acc, item) => acc + item.measureValue, 0);
// totalPower.value = realTimeSum.toFixed(2)
// co2Reduction.value = (realTimeSum * 670 / 1000000).toFixed(2) // 克转化为吨
// so2Reduction.value = (realTimeSum * 10.5 / 1000000).toFixed(2) // 克转化为吨
// noxReduction.value = (realTimeSum * 18.1 / 1000000).toFixed(2) // 克转化为吨
// pm25Reduction.value = (realTimeSum * 1.46 / 1000000).toFixed(2) // 克转化为吨
// fuelReduction.value = (realTimeSum * 0.22 / 1000).toFixed(2) // 转化为吨
const incrementRealTimeSum = Object.values(res).reduce((acc, item) => acc + item.incrementValue, 0);
chartData.value.realtime = {
totalPower: Number(realTimeSum.toFixed(2)),
fuel: Number((realTimeSum * 0.22 / 1).toFixed(2)), // 转化为千克
co2: Number((realTimeSum * 670 / 1000).toFixed(2)), // 克转化为千克
pm25: Number((realTimeSum * 1.46 / 1000).toFixed(2)), // 克转化为千克
nox: Number((realTimeSum * 18.1 / 1000).toFixed(2)), // 克转化为千克
so2: Number((realTimeSum * 10.5 / 1000).toFixed(2)), // 克转化为千克
}
chartData.value.day = {
totalPower: Number(daySum.toFixed(2)),
fuel: Number((daySum * 0.22 / 1).toFixed(2)), // 转化为千克
co2: Number((daySum * 670 / 1000).toFixed(2)), // 克转化为千克
pm25: Number((daySum * 1.46 / 1000).toFixed(2)), // 克转化为千克
nox: Number((daySum * 18.1 / 1000).toFixed(2)), // 克转化为千克
so2: Number((daySum * 10.5 / 1000).toFixed(2)), // 克转化为千克
}
chartData.value.week = {
totalPower: Number(weekSum.toFixed(2)),
fuel: Number((weekSum * 0.22 / 1).toFixed(2)), // 转化为吨
co2: Number((weekSum * 670 / 1000).toFixed(2)), // 克转化为千克
pm25: Number((weekSum * 1.46 / 1000).toFixed(2)), // 克转化为千克
nox: Number((weekSum * 18.1 / 1000).toFixed(2)), // 克转化为千克
so2: Number((weekSum * 10.5 / 1000).toFixed(2)), // 克转化为千克
}
chartData.value.month = {
totalPower: Number(monthSum.toFixed(2)),
fuel: Number((monthSum * 0.22 / 1).toFixed(2)), // 转化为吨
co2: Number((monthSum * 670 / 1000).toFixed(2)), // 克转化为千克
pm25: Number((monthSum * 1.46 / 1000).toFixed(2)), // 克转化为千克
nox: Number((monthSum * 18.1 / 1000).toFixed(2)), // 克转化为千克
so2: Number((monthSum * 10.5 / 1000).toFixed(2)), // 克转化为千克
}
chartData.value.year = {
totalPower: Number(yearSum.toFixed(2)),
fuel: Number((yearSum * 0.22 / 1).toFixed(2)), // 转化为千克
co2: Number((yearSum * 670 / 1000).toFixed(2)), // 克转化为千克
pm25: Number((yearSum * 1.46 / 1000).toFixed(2)), // 克转化为千克
nox: Number((yearSum * 18.1 / 1000).toFixed(2)), // 克转化为千克
so2: Number((yearSum * 10.5 / 1000).toFixed(2)), // 克转化为千克
}
// 记录实时数据到图表数组
const now = new Date()
const timestamp = now.toLocaleTimeString()
// 计算各指标值(转换为合适的单位)并保留两位小数
const totalPowerValue = Number(incrementRealTimeSum.toFixed(2))
const co2Value = Number((incrementRealTimeSum * 670 / 1000).toFixed(2)) // 克转化为千克
const so2Value = Number((incrementRealTimeSum * 10.5 / 1000).toFixed(2)) // 克转化为千克
const noxValue = Number((incrementRealTimeSum * 18.1 / 1000).toFixed(2)) // 克转化为千克
const pm25Value = Number((incrementRealTimeSum * 1.46 / 1000).toFixed(2)) // 克转化为千克
const fuelValue = Number((incrementRealTimeSum * 0.22 / 1).toFixed(2)) // 转化为千克
// 添加到各图表数据数组
totalPowerChartData.value.push({ name: timestamp, value: totalPowerValue })
co2ReductionChartData.value.push({ name: timestamp, value: co2Value })
so2ReductionChartData.value.push({ name: timestamp, value: so2Value })
noxReductionChartData.value.push({ name: timestamp, value: noxValue })
pm25ReductionChartData.value.push({ name: timestamp, value: pm25Value })
fuelReductionChartData.value.push({ name: timestamp, value: fuelValue })
// 保持数组最多包含20个数据点(对应约3分钟,每10秒一条)
const maxDataPoints = 10
totalPowerChartData.value = totalPowerChartData.value.slice(-maxDataPoints)
co2ReductionChartData.value = co2ReductionChartData.value.slice(-maxDataPoints)
so2ReductionChartData.value = so2ReductionChartData.value.slice(-maxDataPoints)
noxReductionChartData.value = noxReductionChartData.value.slice(-maxDataPoints)
pm25ReductionChartData.value = pm25ReductionChartData.value.slice(-maxDataPoints)
fuelReductionChartData.value = fuelReductionChartData.value.slice(-maxDataPoints)
// 更新通用图表数据(保持向后兼容)
commonChartData.value.push({ name: timestamp, value: totalPowerValue })
commonChartData.value = commonChartData.value.slice(-maxDataPoints)
} catch (error) {
console.error(error)
}
}
const handleGetAllDeviceDataByTimeRange = async () => {
try {
// 当前时间的时间戳(毫秒)
const nowTimestamp = dayjs().valueOf().toString();
// 两天前的时间戳(毫秒)
const twoDaysAgoTimestamp = dayjs().subtract(2, 'day').valueOf().toString();
const params = {
start: twoDaysAgoTimestamp,
end: nowTimestamp,
timeType: '3'
}
const res = await MapApi.getByStartAndEndTimeAndTimeType(params)
console.log(res)
} catch (error) {
console.error(error)
}
}
// 组件挂载时初始化数据和定时器
onMounted(() => {
handleGetAllDeviceDataByTimeRange()
// commonChartData.value = generateInitialData()
// timer = window.setInterval(appendNewData, 1000)
let i = 1
handleGetRealTimeAllData()
timer = setInterval(() => {
handleGetRealTimeAllData(i)
i++
}, 5000)
})
// 组件销毁前清除定时器
onBeforeUnmount(() => {
if (timer) {
window.clearInterval(timer)
}
})
</script>
<style lang="scss" scoped>
.shore-power-usage {
display: flex;
height: 100%;
gap: 10px;
}
.average {
.overview-value {
font-size: 36px;
}
.overview-label {
font-size: 18px;
}
}
.magnify {
width: 100%;
padding: 12px;
height: calc(100vh - 72px);
position: absolute;
top: 72px;
left: 0px;
display: flex;
flex-direction: column;
/* background-color: red; */
gap: 8px;
overflow-y: auto;
.big {
height: 70%;
.card {
height: 100%;
}
.overview-value {
font-size: 48px;
}
.overview-label {
font-size: 28px;
}
}
.one-row {
height: 30%;
display: flex;
gap: 8px;
.card .card-title .title-text {
font-size: 14px;
}
.card {
flex: 1;
}
}
}
.card {
padding: 6px;
cursor: pointer;
}
.time-range-selector {
display: flex;
gap: 8px;
}
.show-value {
font-size: 18px;
font-weight: 600;
font-weight: bold;
display: flex;
align-items: center;
gap: 6px;
margin-bottom: 8px;
color: #FFF;
}
.show-value-label {
font-size: 12px;
font-weight: 600;
font-weight: bold;
}
.show-value-value {
font-size: 18px;
font-weight: 600;
font-weight: bold;
}
.time-range-btn {
padding: 4px 12px;
border: 1px solid rgba(255, 255, 255, 0.3);
border-radius: 4px;
background: transparent;
color: rgba(255, 255, 255, 0.7);
font-size: 12px;
cursor: pointer;
transition: all 0.3s ease;
}
.time-range-btn:hover {
background: rgba(255, 255, 255, 0.1);
border-color: rgba(255, 255, 255, 0.5);
}
.time-range-btn.active {
background: rgba(76, 175, 80, 0.8);
border-color: #4CAF50;
color: #fff;
}
.right {
/* gap: 0 */
}
</style>