NGToolsAdmin/pages/uni-stat/device/activity/activity.vue

443 lines
12 KiB
Vue
Raw Permalink 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 class="uni-group">
<view class="uni-sub-title hide-on-phone">用户活跃度分析</view>
</view>
</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="应用选择" @change="changeAppid" 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" :yesterday="false" @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>
<view class="uni-stat--x p-m">
<view class="label-text mb-l">
趋势图
</view>
<uni-stat-tabs type="box" :tabs="chartTabs" class="mb-l" @change="changeChartTab" />
<view class="uni-charts-box">
<qiun-data-charts type="area" :chartData="chartData" echartsH5 echartsApp :errorMessage="errorMessage"/>
</view>
</view>
<view class="uni-stat--x p-m">
<uni-stat-table :data="tableData" :filedsMap="fieldsMap" :loading="loading" tooltip />
<view class="uni-pagination-box">
<uni-pagination show-icon show-page-size :page-size="options.pageSize"
:current="options.pageCurrent" :total="options.total" @change="changePageCurrent"
@pageSizeChange="changePageSize" />
</view>
</view>
</view>
<!-- #ifndef H5 -->
<fix-window />
<!-- #endif -->
</view>
</template>
<script>
import {
mapfields,
stringifyQuery,
stringifyField,
stringifyGroupField,
getTimeOfSomeDayAgo,
division,
format,
formatDate,
maxDeltaDay,
debounce,
} from '@/js_sdk/uni-stat/util.js'
import fieldsMap from './fieldsMap.js'
export default {
data() {
return {
tableName: 'uni-stat-result',
fieldsMap,
query: {
dimension: "day",
appid: '',
platform_id: '',
uni_platform: '',
version_id: '',
channel_id: '',
start_time: [],
},
options: {
pageSize: 20,
pageCurrent: 1, // 当前页
total: 0, // 数据总量
},
loading: false,
currentDateTab: 0,
currentChartTab: 'day',
tableData: [],
chartData: {},
channelData: [],
tabName: '日活',
errorMessage: "",
}
},
computed: {
chartTabs() {
const tabs = [{
_id: 'day',
name: '日活'
}, {
_id: 'week',
name: '周活'
}, {
_id: 'month',
name: '月活'
}]
if (maxDeltaDay(this.query.start_time, 7)) {
tabs.forEach((tab, index) => {
if (tab._id === 'month') {
tab.disabled = true
} else {
tab.disabled = false
}
})
}
return tabs
},
channelQuery() {
const platform_id = this.query.platform_id
return stringifyQuery({
platform_id
})
},
versionQuery() {
const {
appid,
uni_platform
} = this.query
const query = stringifyQuery({
appid,
uni_platform,
})
return query
},
},
created() {
this.debounceGet = debounce(() => {
this.getAllData(this.query);
}, 300);
this.getChannelData()
},
watch: {
query: {
deep: true,
handler(val) {
this.options.pageCurrent = 1 // 重置分页
this.debounceGet()
}
}
},
methods: {
useDatetimePicker() {
this.currentDateTab = -1
},
changeAppid(id) {
this.getChannelData(id, false)
},
changePlatform(id, index, name, item) {
this.getChannelData(null, id)
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.getTabelData(this.query)
},
changePageSize(pageSize) {
this.options.pageSize = pageSize
this.options.pageCurrent = 1 // 重置分页
this.getTabelData(this.query)
},
changeChartTab(type, index, name) {
this.currentChartTab = type
this.tabName = name
this.getChartData(this.query, type, name)
},
getAllData(query) {
if (!query.appid) {
this.errorMessage = "请先选择应用";
return; // 如果appid为空则不进行查询
}
this.errorMessage = "";
this.getChartData(query, this.currentChartTab, this.tabName)
this.getTabelData(query)
},
getChartData(query, type, name = '日活', field = 'active_device_count') {
this.chartData = {}
const options = {
categories: [],
series: [{
name,
data: []
}]
}
query = stringifyQuery(query, false, ['uni_platform'])
console.log('query: ', query)
const db = uniCloud.database()
if (type === 'day') {
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
this.chartData = []
for (const item of data) {
const x = formatDate(item.start_time, 'day')
const y = item[field]
options.series[0].data.push(y)
options.categories.push(x)
}
this.chartData = options
}).catch((err) => {
console.error(err)
})
} else {
// 周、月范围的处理
this.getRangeCountData(query, type).then(res => {
const oldType = type
if (type === 'week') type = 'isoWeek'
const {
count,
data
} = res.result
this.chartData = []
const wunWeekTime = 7 * 24 * 60 * 60 * 1000
for (const item of data) {
const date = +new Date(item.year, 0) + (Number(item[type]) * wunWeekTime - 1)
const x = formatDate(date, oldType)
const y = item[type + '_' + field]
if (y) {
options.series[0].data.push(y)
options.categories.push(x)
}
}
this.chartData = options
})
}
},
getTabelData(queryData, field = 'active_device_count') {
const {
pageCurrent
} = this.options
let query = stringifyQuery(queryData)
this.loading = true
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', 'desc')
.skip((pageCurrent - 1) * this.options.pageSize)
.limit(this.options.pageSize)
.get({
getCount: true
})
.then(res => {
const {
count,
data
} = res.result
let daysData = data,
daysCount = count,
weeks = [],
months = []
// 获取周活、月活
// stringifyQuery(queryData)
let query = JSON.parse(JSON.stringify(queryData))
query.dimension = 'week'
this.getRangeCountData(stringifyQuery(query), 'week').then(res => {
const {
count,
data
} = res.result
weeks = data
// queryData.dimension = 'month'
let query = JSON.parse(JSON.stringify(queryData))
query.dimension = 'month'
this.getRangeCountData(stringifyQuery(query), 'month').then(res => {
const {
count,
data
} = res.result
months = data
const allData = this.mapWithWeekAndMonth(daysData, weeks, months)
for (const item of allData) {
mapfields(fieldsMap, item, item)
}
this.tableData = []
this.options.total = daysCount
this.tableData = allData
}).finally(() => {
this.loading = false
})
})
}).catch((err) => {
console.error(err)
// err.message 错误信息
// err.code 错误码
})
},
getRangeCountData(query, type, field = 'active_device_count') {
if (type === 'week') type = 'isoWeek'
const {
pageCurrent
} = this.options
const db = uniCloud.database()
return db.collection(this.tableName)
.where(query)
.field(
`${field}, start_time, ${type}(add(new Date(0),start_time), "Asia/Shanghai") as ${type},year(add(new Date(0),start_time), "Asia/Shanghai") as year`
)
.groupBy(`year, ${type}`)
.groupField(`sum(${field}) as ${type}_${field}`)
.orderBy(`year asc, ${type} asc`)
.get({
getCount: true
})
},
// 匹配数据日期所在的周活、月活
mapWithWeekAndMonth(data, weeks, months, field = 'active_device_count') {
for (const item of data) {
const date = new Date(item.start_time)
const year = date.getUTCFullYear()
const month = date.getMonth() + 1
const week = this.getWeekNumber(date)
for (const w of weeks) {
if (w.isoWeek === week && w.year === year) {
item[`week_${field}`] = w[`isoWeek_${field}`]
}
}
for (const m of months) {
if (m.month === month && m.year === year) {
item[`month_${field}`] = m[`month_${field}`]
}
}
}
return data
},
//日期所在的周(一年中的第几周)
getWeekNumber(d) {
// Copy date so don't modify original
d = new Date(Date.UTC(d.getFullYear(), d.getMonth(), d.getDate()));
// Set to nearest Thursday: current date + 4 - current day number
// Make Sunday's day number 7
d.setUTCDate(d.getUTCDate() + 4 - (d.getUTCDay() || 7));
// Get first day of year
const yearStart = new Date(Date.UTC(d.getUTCFullYear(), 0, 1));
// Calculate full weeks to nearest Thursday
return Math.ceil((((d - yearStart) / 86400000) + 1) / 7);
},
//获取渠道信息
getChannelData(appid, platform_id) {
this.query.channel_id = ''
const db = uniCloud.database()
const condition = {}
//对应应用
appid = appid ? appid : this.query.appid
if (appid) {
condition.appid = appid
}
//对应平台
platform_id = platform_id ? platform_id : this.query.platform_id
if (platform_id) {
condition.platform_id = platform_id
}
let platformTemp = db.collection('uni-stat-app-platforms')
.field('_id, name')
.getTemp()
let channelTemp = db.collection('uni-stat-app-channels')
.where(condition)
.field('_id, channel_name, create_time, platform_id')
.getTemp()
db.collection(channelTemp, platformTemp)
.orderBy('platform_id', 'asc')
.get()
.then(res => {
let data = res.result.data
let channels = []
if (data.length > 0) {
let channelName
for (let i in data) {
channelName = data[i].channel_name ? data[i].channel_name : '默认'
if (data[i].platform_id.length > 0) {
channelName = data[i].platform_id[0].name + '-' + channelName
}
channels.push({
value: data[i]._id,
text: channelName
})
}
}
this.channelData = channels
})
.catch((err) => {
console.error(err)
// err.message 错误信息
// err.code 错误码
}).finally(() => {})
}
}
}
</script>
<style>
</style>