NGToolsAdmin/pages/uni-stat/user/overview/overview.vue

420 lines
11 KiB
Vue
Raw Normal View History

2024-09-13 08:39:31 +00:00
<template>
<!-- 对应页面注册用户统计-概况 -->
<view class="fix-top-window">
<view class="uni-header">
<uni-stat-breadcrumb class="uni-stat-breadcrumb-on-phone" />
</view>
<view class="uni-container">
<view class="uni-stat--x flex p-1015">
<view class="uni-stat--app-select">
<uni-data-select collection="opendb-app-list" field="appid as value, name as text" orderby="text asc" :defItem="1" label="应用选择" v-model="query.appid" :clear="false" />
<uni-data-select collection="opendb-app-versions" :where="versionQuery" class="ml-m" field="_id as value, version as text, uni_platform as label, create_date as date" format="{label} - {text}" orderby="date desc" label="版本选择" v-model="query.version_id" />
</view>
</view>
<view class="uni-stat--x flex">
<uni-stat-tabs label="日期选择" :current="currentDateTab" mode="date" :today="true"
@change="changeTimeRange" />
<uni-datetime-picker type="datetimerange" :end="new Date().getTime()" v-model="query.start_time"
returnType="timestamp" :clearIcon="false" class="uni-stat-datetime-picker"
:class="{'uni-stat__actived': currentDateTab < 0 && !!query.start_time.length}"
@change="useDatetimePicker" />
</view>
<view class="uni-stat--x">
<uni-stat-tabs label="平台选择" type="boldLine" mode="platform" v-model="query.platform_id"
@change="changePlatform" />
<uni-data-select ref="version-select" v-if="query.platform_id && query.platform_id.indexOf('==') === -1" collection="uni-stat-app-channels" :where="channelQuery" class="p-channel" field="_id as value, channel_name as text" orderby="text asc" label="渠道/场景值选择" v-model="query.channel_id" />
</view>
<uni-stat-panel :items="panelData" :contrast="true" />
<view class="uni-stat--x p-m">
<view class="uni-stat-card-header">
趋势图
</view>
<uni-stat-tabs type="box" v-model="chartTab" :tabs="chartTabs" class="mb-l" @change="changeChartTab" />
<view class="uni-charts-box">
<qiun-data-charts type="area" :chartData="chartData" :eopts="eopts" echartsH5 echartsApp
tooltipFormat="tooltipCustom" :errorMessage="errorMessage"/>
</view>
</view>
</view>
<!-- #ifndef H5 -->
<fix-window />
<!-- #endif -->
</view>
</template>
<script>
import {
mapfields,
stringifyQuery,
stringifyField,
stringifyGroupField,
getTimeOfSomeDayAgo,
division,
format,
formatDate,
parseDateTime,
getFieldTotal,
debounce
} from '@/js_sdk/uni-stat/util.js'
import {
fieldsMap,
resFieldsMap,
entFieldsMap
} from './fieldsMap.js'
const panelOption = fieldsMap.filter(f => f.hasOwnProperty('value'))
export default {
data() {
return {
tableName: 'uni-stat-result',
fieldsMap,
resFieldsMap,
entFieldsMap,
query: {
dimension: 'hour',
appid: '',
platform_id: '',
uni_platform: '',
version_id: '',
start_time: [],
},
options: {
pageCurrent: 1, // 当前页
total: 0, // 数据总量
pageSizeIndex: 0, // 与 pageSizeRange 一起计算得出 pageSize
pageSizeRange: [10, 20, 50, 100],
},
loading: false,
currentDateTab: 2,
chartTab: 'new_user_count',
tableData: [],
resTableData: [],
entTableData: [],
panelData: panelOption,
chartData: {},
eopts: {
seriesTemplate: [{
itemStyle: {
borderWidth: 2,
borderColor: '#1890FF',
color: '#1890FF'
},
areaStyle: {
color: {
colorStops: [{
offset: 0,
color: '#1890FF', // 0% 处的颜色
}, {
offset: 1,
color: '#FFFFFF' // 100% 处的颜色
}]
}
}
}, {
// smooth: false,
lineStyle: {
color: '#ea7ccc',
width: 2,
type: 'dashed'
},
itemStyle: {
borderWidth: 1,
borderColor: '#ea7ccc',
color: '#ea7ccc'
},
areaStyle: null
}]
},
tabIndex: 0,
errorMessage: "",
}
},
onLoad(option) {
const {
appid
} = option
if (appid) {
this.query.appid = appid
}
},
computed: {
pageSize() {
const {
pageSizeRange,
pageSizeIndex
} = this.options
return pageSizeRange[pageSizeIndex]
},
chartTabs() {
const tabs = []
fieldsMap.forEach(item => {
const _id = item.field
const name = item.title
if (_id && name) {
tabs.push({
_id,
name
})
}
})
return tabs
},
versionQuery() {
const {
appid,
uni_platform
} = this.query
const query = stringifyQuery({
appid,
uni_platform
})
return query
},
channelQuery() {
const {
appid,
platform_id,
} = this.query
const query = stringifyQuery({
appid,
platform_id
})
return query
},
},
created() {
this.debounceGet = debounce(() => {
this.getAllData(this.query);
}, 300);
},
watch: {
query: {
deep: true,
handler(val) {
this.options.pageCurrent = 1 // 重置分页
this.debounceGet()
}
}
},
methods: {
useDatetimePicker() {
this.currentDateTab = null
},
changePlatform(id, index, name, item) {
this.query.version_id = 0
this.query.uni_platform = item.code
},
changeTimeRange(id, index) {
this.currentDateTab = index
const day = 24 * 60 * 60 * 1000
let start, end
start = getTimeOfSomeDayAgo(id)
if (!id) {
end = getTimeOfSomeDayAgo(0) + day - 1
} else {
end = getTimeOfSomeDayAgo(0) - 1
}
this.query.start_time = [start, end]
},
changePageCurrent(e) {
this.options.pageCurrent = e.current
this.getChartData(this.query)
},
changePageSize(e) {
const {
value
} = e.detail
this.options.pageCurrent = 1 // 重置分页
this.options.pageSizeIndex = value
this.getChartData(this.query)
},
changeChartTab(id, index, name) {
this.tabIndex = index
this.getChartData(this.query, id, name)
},
getAllData(query) {
if (!query.appid) {
this.errorMessage = "请先选择应用";
return; // 如果appid为空则不进行查询
}
this.errorMessage = "";
this.getPanelData()
this.getChartData(query)
},
getDays() {
if (!this.query.start_time.length) return true
const day = 24 * 60 * 60 * 1000
const [start, end] = this.query.start_time
const lessTwoDay = end - start >= day
return lessTwoDay
},
getPanelData() {
const {
appid,
platform_id,
version_id
} = this.query
const query = stringifyQuery({
appid,
platform_id,
version_id,
start_time: [getTimeOfSomeDayAgo(1), new Date().getTime()]
})
const db = uniCloud.database()
const subTable = db.collection(this.tableName)
.where(query)
.field(
`${stringifyField(fieldsMap)},dimension,stat_date`
)
.groupBy(`stat_date, dimension`)
.groupField(stringifyGroupField(fieldsMap))
.orderBy('stat_date', 'desc')
.get()
.then(res => {
const data = res.result.data
const today = data.find(item => item.stat_date === parseDateTime(getTimeOfSomeDayAgo(0), '',
'')) || {}
today.total_users = 0
const yesterday = data.find(item => item.dimension === 'day' && item.stat_date ===
parseDateTime(getTimeOfSomeDayAgo(1), '', ''))
this.panelData = []
this.panelData = mapfields(fieldsMap, today)
this.panelData.map(item => {
mapfields(fieldsMap, yesterday, item, '', 'contrast')
})
getFieldTotal.call(this, query, 'total_users')
})
},
getChartData(query, field = this.chartTabs[this.tabIndex]._id, name = this.chartTabs[this.tabIndex].name) {
// this.chartData = {}
const {
pageCurrent
} = this.options
const days = this.currentDateTab
const date = getTimeOfSomeDayAgo(days)
const day = 24 * 60 * 60 * 1000
let start_time
// 范围小于一天的日期(今日、昨日)做增大一天处理,目的是对比当前日期和前一天(今日和昨日、昨日和上前日)
if (!this.getDays()) {
const start = date - day
const end = date + day - 1
query = JSON.parse(JSON.stringify(query))
start_time = query.start_time = [start, end]
}
query = stringifyQuery(query, true, ['uni_platform'])
const db = uniCloud.database()
db.collection(this.tableName)
.where(query)
.field(`${stringifyField(fieldsMap, field)}, start_time`)
.groupBy(`start_time`)
.groupField(stringifyGroupField(fieldsMap, field))
.orderBy('start_time', 'asc')
.get({
getCount: true
})
.then(res => {
const {
count,
data
} = res.result
const options = {
categories: [],
series: [{
name,
data: []
}]
}
let mapper = fieldsMap.filter(f => f.field === field)
mapper = JSON.parse(JSON.stringify(mapper))
delete mapper[0].value
mapper[0].formatter = ''
if (!this.getDays()) {
const [start, end] = start_time
const line = options.series[1] = {
name: formatDate(start),
data: [],
}
const cont = options.series[0] = {
name: formatDate(end),
data: [],
}
for (let i = 0; i < 24; ++i) {
const hour = i < 10 ? '0' + i : i
const x = `${hour}:00 ~ ${hour}:59`
options.categories.push(x)
line.data[i] = 0
cont.data[i] = 0
data.forEach(item => {
mapfields(mapper, item, item)
let val = Number(item[field])
const d = new Date(item.start_time)
if (item.start_time < date) {
if (d.getHours() === i) {
line.data[i] = val
}
} else {
if (d.getHours() === i) {
cont.data[i] = val
}
}
})
}
} else {
for (const item of data) {
mapfields(mapper, item, item)
const x = formatDate(item.start_time, 'day')
let y = Number(item[field])
options.series[0].data.push(y)
options.categories.push(x)
}
}
this.chartData = options
}).catch((err) => {
console.error(err)
// err.message 错误信息
// err.code 错误码
}).finally(() => {})
},
getAppAccessTimes(query) {
const db = uniCloud.database()
return db.collection(this.tableName)
.where(query)
.groupBy('appid')
.groupField(`sum(page_visit_count) as total_app_access`)
.get()
},
navTo(url) {
if (!url) return
uni.navigateTo({
url
})
}
}
}
</script>
<style>
.uni-stat-card-header {
display: flex;
justify-content: space-between;
color: #555;
font-size: 14px;
font-weight: 600;
padding: 10px 0;
margin-bottom: 15px;
}
.uni-stat-card-header-link {
cursor: pointer;
}
</style>