2022-04-13 10:38:47 +03:00
|
|
|
import React from "react";
|
2022-09-01 11:22:04 +03:00
|
|
|
import { Tooltip } from '../../util/tooltip'
|
2023-01-16 11:30:22 +03:00
|
|
|
import { SecondsSinceLastLoad } from '../../util/seconds-since-last-load'
|
2022-11-22 15:50:58 +03:00
|
|
|
import classNames from "classnames";
|
2022-09-01 11:22:04 +03:00
|
|
|
import numberFormatter, { durationFormatter } from '../../util/number-formatter'
|
2022-11-15 17:38:39 +03:00
|
|
|
import { METRIC_MAPPING } from './graph-util'
|
2022-04-13 10:38:47 +03:00
|
|
|
|
|
|
|
export default class TopStats extends React.Component {
|
2022-09-01 11:22:04 +03:00
|
|
|
renderComparison(name, comparison) {
|
2022-04-13 10:38:47 +03:00
|
|
|
const formattedComparison = numberFormatter(Math.abs(comparison))
|
|
|
|
|
|
|
|
if (comparison > 0) {
|
|
|
|
const color = name === 'Bounce rate' ? 'text-red-400' : 'text-green-500'
|
|
|
|
return <span className="text-xs dark:text-gray-100"><span className={color + ' font-bold'}>↑</span> {formattedComparison}%</span>
|
|
|
|
} else if (comparison < 0) {
|
|
|
|
const color = name === 'Bounce rate' ? 'text-green-500' : 'text-red-400'
|
|
|
|
return <span className="text-xs dark:text-gray-100"><span className={color + ' font-bold'}>↓</span> {formattedComparison}%</span>
|
|
|
|
} else if (comparison === 0) {
|
2022-10-28 17:21:07 +03:00
|
|
|
return <span className="text-xs text-gray-700 dark:text-gray-300">〰 0%</span>
|
2022-04-13 10:38:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
topStatNumberShort(stat) {
|
|
|
|
if (['visit duration', 'time on page'].includes(stat.name.toLowerCase())) {
|
|
|
|
return durationFormatter(stat.value)
|
|
|
|
} else if (['bounce rate', 'conversion rate'].includes(stat.name.toLowerCase())) {
|
|
|
|
return stat.value + '%'
|
|
|
|
} else {
|
|
|
|
return numberFormatter(stat.value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-01 11:22:04 +03:00
|
|
|
topStatNumberLong(stat) {
|
|
|
|
if (['visit duration', 'time on page'].includes(stat.name.toLowerCase())) {
|
|
|
|
return durationFormatter(stat.value)
|
|
|
|
} else if (['bounce rate', 'conversion rate'].includes(stat.name.toLowerCase())) {
|
|
|
|
return stat.value + '%'
|
2022-04-13 10:38:47 +03:00
|
|
|
} else {
|
2022-09-01 11:22:04 +03:00
|
|
|
return stat.value.toLocaleString()
|
2022-04-13 10:38:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-01 11:22:04 +03:00
|
|
|
topStatTooltip(stat) {
|
|
|
|
let statName = stat.name.toLowerCase()
|
|
|
|
statName = stat.value === 1 ? statName.slice(0, -1) : statName
|
|
|
|
|
2023-03-10 11:28:01 +03:00
|
|
|
const { topStatData, lastLoadTimestamp } = this.props
|
|
|
|
const showingImported = topStatData?.imported_source && topStatData?.with_imported
|
|
|
|
|
2022-09-01 11:22:04 +03:00
|
|
|
return (
|
|
|
|
<div>
|
2022-09-13 12:46:01 +03:00
|
|
|
<div className="whitespace-nowrap">{this.topStatNumberLong(stat)} {statName}</div>
|
2023-03-10 11:28:01 +03:00
|
|
|
{stat.name === 'Current visitors' && <p className="font-normal text-xs">Last updated <SecondsSinceLastLoad lastLoadTimestamp={lastLoadTimestamp}/>s ago</p>}
|
|
|
|
{stat.name === 'Views per visit' && showingImported && <p className="font-normal text-xs whitespace-nowrap">Based only on native data</p>}
|
2022-09-01 11:22:04 +03:00
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
canMetricBeGraphed(stat) {
|
|
|
|
const isTotalUniqueVisitors = this.props.query.filters.goal && stat.name === 'Unique visitors'
|
|
|
|
const isKnownMetric = Object.keys(METRIC_MAPPING).includes(stat.name)
|
|
|
|
|
|
|
|
return isKnownMetric && !isTotalUniqueVisitors
|
|
|
|
}
|
|
|
|
|
|
|
|
maybeUpdateMetric(stat) {
|
|
|
|
if (this.canMetricBeGraphed(stat)) {
|
|
|
|
this.props.updateMetric(METRIC_MAPPING[stat.name])
|
2022-04-13 10:38:47 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-16 11:30:22 +03:00
|
|
|
blinkingDot() {
|
2022-04-13 10:38:47 +03:00
|
|
|
return (
|
2023-01-16 11:30:22 +03:00
|
|
|
<div key="dot" className="block pulsating-circle" style={{ left: '125px', top: '52px' }}></div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
renderStatName(stat) {
|
|
|
|
const { metric } = this.props
|
|
|
|
const isSelected = metric === METRIC_MAPPING[stat.name]
|
|
|
|
|
|
|
|
const [statDisplayName, statExtraName] = stat.name.split(/(\(.+\))/g)
|
|
|
|
|
|
|
|
const statDisplayNameClass = classNames('text-xs font-bold tracking-wide text-gray-500 uppercase dark:text-gray-400 whitespace-nowrap flex w-content border-b', {
|
|
|
|
'text-indigo-700 dark:text-indigo-500 border-indigo-700 dark:border-indigo-500': isSelected,
|
|
|
|
'group-hover:text-indigo-700 dark:group-hover:text-indigo-500 border-transparent': !isSelected
|
|
|
|
})
|
|
|
|
|
|
|
|
return(
|
|
|
|
<div className={statDisplayNameClass}>
|
|
|
|
{statDisplayName}
|
|
|
|
{statExtraName && <span className="hidden sm:inline-block ml-1">{statExtraName}</span>}
|
|
|
|
</div>
|
2022-04-13 10:38:47 +03:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2023-01-16 11:30:22 +03:00
|
|
|
const { topStatData, query } = this.props
|
2022-04-13 10:38:47 +03:00
|
|
|
|
|
|
|
const stats = topStatData && topStatData.top_stats.map((stat, index) => {
|
|
|
|
|
2022-09-01 11:22:04 +03:00
|
|
|
const className = classNames('px-4 md:px-6 w-1/2 my-4 lg:w-auto group select-none', {
|
|
|
|
'cursor-pointer': this.canMetricBeGraphed(stat),
|
|
|
|
'lg:border-l border-gray-300': index > 0,
|
|
|
|
'border-r lg:border-r-0': index % 2 === 0
|
|
|
|
})
|
|
|
|
|
2022-04-13 10:38:47 +03:00
|
|
|
return (
|
2023-01-16 11:30:22 +03:00
|
|
|
<Tooltip key={stat.name} info={this.topStatTooltip(stat)} className={className} onClick={() => { this.maybeUpdateMetric(stat) }} boundary={this.props.tooltipBoundary}>
|
|
|
|
{this.renderStatName(stat)}
|
|
|
|
<div className="flex items-center justify-between my-1 whitespace-nowrap">
|
|
|
|
<b className="mr-4 text-xl md:text-2xl dark:text-gray-100" id={METRIC_MAPPING[stat.name]}>{this.topStatNumberShort(stat)}</b>
|
|
|
|
{this.renderComparison(stat.name, stat.change)}
|
|
|
|
</div>
|
|
|
|
</Tooltip>
|
2022-04-13 10:38:47 +03:00
|
|
|
)
|
|
|
|
})
|
|
|
|
|
2022-11-22 15:50:58 +03:00
|
|
|
if (stats && query && query.period === 'realtime') {
|
2023-01-16 11:30:22 +03:00
|
|
|
stats.push(this.blinkingDot())
|
2022-04-13 10:38:47 +03:00
|
|
|
}
|
|
|
|
|
2022-11-22 15:50:58 +03:00
|
|
|
return stats || null;
|
2022-04-13 10:38:47 +03:00
|
|
|
}
|
|
|
|
}
|