mirror of
https://github.com/plausible/analytics.git
synced 2024-12-22 00:51:36 +03:00
f576fa2a2c
* Add details=True to export API parameters This makes the ZIP export add `%{"details" => "True"}` to the query's `params` when fetching data internally for packaging in the ZIP. This adds bounce_rate and time_on_page to the data in pages.csv, and bounce_rate and visit_duration to sources.csv. * Make API return data with consistent names Some of the data types returned via the JSON or CSV API use inconsistent naming, and some have redundant name changes (i.e. count -> visitors -> count). This makes these all consistent and removes the redundancy. This addresses #1426, fixes some of the CSV headers, and unifies the JSON and CSV return data labels. * Update changelog * Test should use Timex.shift, not relative time * Return full country names in CSV export This also replaces the " character with ' in two country names, as those are the characters used in the names, yielding a more predictable and 'correct' output. * Fetch CSV exported data concurrently * Use spinner to indicate when export has started * Use 300 as default number of brekadown entries for export Higher numbers (e.g. 1000) seem to cause clickhouse errors when there many pages to request. It is unclear what is causing the error, as clickhouse returns an "unknown" error code and an empty error message.
203 lines
8.3 KiB
JavaScript
203 lines
8.3 KiB
JavaScript
import React from "react";
|
|
import { Link, withRouter } from 'react-router-dom'
|
|
|
|
import Modal from './modal'
|
|
import * as api from '../../api'
|
|
import numberFormatter, {durationFormatter} from '../../number-formatter'
|
|
import {parseQuery, toHuman} from '../../query'
|
|
import {formatFullDate} from '../../date'
|
|
|
|
class ReferrerDrilldownModal extends React.Component {
|
|
constructor(props) {
|
|
super(props)
|
|
this.state = {
|
|
loading: true,
|
|
query: parseQuery(props.location.search, props.site)
|
|
}
|
|
}
|
|
|
|
componentDidMount() {
|
|
const detailed = this.showExtra()
|
|
|
|
api.get(`/api/stats/${encodeURIComponent(this.props.site.domain)}/referrers/${this.props.match.params.referrer}`, this.state.query, {limit: 100, detailed})
|
|
.then((res) => this.setState({loading: false, referrers: res.referrers, totalVisitors: res.total_visitors}))
|
|
}
|
|
|
|
showExtra() {
|
|
return this.state.query.period !== 'realtime' && !this.state.query.filters.goal
|
|
}
|
|
|
|
showConversionRate() {
|
|
return !!this.state.query.filters.goal
|
|
}
|
|
|
|
label() {
|
|
if (this.state.query.period === 'realtime') {
|
|
return 'Current visitors'
|
|
}
|
|
|
|
if (this.showConversionRate()) {
|
|
return 'Conversions'
|
|
}
|
|
|
|
return 'Visitors'
|
|
}
|
|
|
|
formatBounceRate(ref) {
|
|
if (typeof(ref.bounce_rate) === 'number') {
|
|
return ref.bounce_rate + '%'
|
|
} else {
|
|
return '-'
|
|
}
|
|
}
|
|
|
|
formatDuration(referrer) {
|
|
if (typeof(referrer.visit_duration) === 'number') {
|
|
return durationFormatter(referrer.visit_duration)
|
|
} else {
|
|
return '-'
|
|
}
|
|
}
|
|
|
|
renderExternalLink(name) {
|
|
if (name !== 'Direct / None') {
|
|
return (
|
|
<a target="_blank" href={'//' + name} rel="noreferrer" className="hidden group-hover:block">
|
|
<svg className="inline h-4 w-4 ml-1 -mt-1 text-gray-600 dark:text-gray-400" fill="currentColor" viewBox="0 0 20 20"><path d="M11 3a1 1 0 100 2h2.586l-6.293 6.293a1 1 0 101.414 1.414L15 6.414V9a1 1 0 102 0V4a1 1 0 00-1-1h-5z"></path><path d="M5 5a2 2 0 00-2 2v8a2 2 0 002 2h8a2 2 0 002-2v-3a1 1 0 10-2 0v3H5V7h3a1 1 0 000-2H5z"></path></svg>
|
|
</a>
|
|
)
|
|
}
|
|
}
|
|
|
|
renderReferrerName(referrer) {
|
|
const query = new URLSearchParams(window.location.search)
|
|
query.set('referrer', referrer.name)
|
|
|
|
return (
|
|
<span className="flex group items-center">
|
|
<img src={`https://icons.duckduckgo.com/ip3/${referrer.url}.ico`} referrerPolicy="no-referrer" className="h-4 w-4 mr-2 inline" />
|
|
<Link className="block truncate hover:underline dark:text-gray-200" to={{search: query.toString(), pathname: '/' + this.props.site.domain}} title={referrer.name}>
|
|
{referrer.name}
|
|
</Link>
|
|
{ this.renderExternalLink(referrer.name) }
|
|
</span>
|
|
)
|
|
}
|
|
|
|
renderTweet(tweet, index) {
|
|
const authorUrl = `https://twitter.com/${tweet.author_handle}`
|
|
const tweetUrl = `${authorUrl}/status/${tweet.tweet_id}`
|
|
const border = index === 0 ? '' : ' pt-4 border-t border-gray-300 dark:border-gray-500'
|
|
|
|
return (
|
|
<div key={tweet.tweet_id}>
|
|
<div className={"flex items-center my-4" + border} >
|
|
<a className="flex items-center group" href={authorUrl} target="_blank" rel="noreferrer">
|
|
<img className="rounded-full w-8" src={tweet.author_image} />
|
|
<div className="ml-2 leading-tight">
|
|
<div className="font-bold group-hover:text-blue-500">{tweet.author_name}</div>
|
|
<div className="text-xs text-gray-500 dark:text-gray-400">@{tweet.author_handle}</div>
|
|
</div>
|
|
</a>
|
|
<a className="ml-auto twitter-icon" href={tweetUrl} target="_blank" rel="noreferrer"></a>
|
|
</div>
|
|
<div className="my-2 cursor-text tweet-text whitespace-pre-wrap" dangerouslySetInnerHTML={{__html: tweet.text}}>
|
|
</div>
|
|
<div className="text-xs text-gray-700 dark:text-gray-300 font-medium">
|
|
{formatFullDate(new Date(tweet.created))}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
renderReferrer(referrer) {
|
|
if (referrer.tweets) {
|
|
return (
|
|
<tr className="text-sm dark:text-gray-200" key={referrer.name}>
|
|
<td className="p-2">
|
|
{ this.renderReferrerName(referrer) }
|
|
<span className="text-gray-500 ml-2 text-xs">
|
|
appears in {referrer.tweets.length} tweets
|
|
</span>
|
|
<div className="my-4 pl-4 border-l-2 border-gray-300 dark:border-gray-500">
|
|
{ referrer.tweets.map(this.renderTweet) }
|
|
</div>
|
|
</td>
|
|
{this.showConversionRate() && <td className="p-2 w-32 font-medium" align="right">{numberFormatter(referrer.total_visitors)}</td> }
|
|
<td className="p-2 w-32 font-medium" align="right" valign="top">{numberFormatter(referrer.visitors)}</td>
|
|
{this.showExtra() && <td className="p-2 w-32 font-medium" align="right" valign="top">{this.formatBounceRate(referrer)}</td> }
|
|
{this.showExtra() && <td className="p-2 w-32 font-medium" align="right" valign="top">{this.formatDuration(referrer)}</td> }
|
|
{this.showConversionRate() && <td className="p-2 w-32 font-medium" align="right">{referrer.conversion_rate}%</td> }
|
|
</tr>
|
|
)
|
|
} else {
|
|
return (
|
|
<tr className="text-sm dark:text-gray-200" key={referrer.name}>
|
|
<td className="p-2">
|
|
{ this.renderReferrerName(referrer) }
|
|
</td>
|
|
{this.showConversionRate() && <td className="p-2 w-32 font-medium" align="right">{numberFormatter(referrer.total_visitors)}</td> }
|
|
<td className="p-2 w-32 font-medium" align="right">{numberFormatter(referrer.visitors)}</td>
|
|
{this.showExtra() && <td className="p-2 w-32 font-medium" align="right">{this.formatBounceRate(referrer)}</td> }
|
|
{this.showExtra() && <td className="p-2 w-32 font-medium" align="right">{this.formatDuration(referrer)}</td> }
|
|
{this.showConversionRate() && <td className="p-2 w-32 font-medium" align="right">{referrer.conversion_rate}%</td> }
|
|
</tr>
|
|
)
|
|
}
|
|
}
|
|
|
|
renderGoalText() {
|
|
if (this.state.query.filters.goal) {
|
|
return (
|
|
<h1 className="text-xl font-semibold text-gray-500 dark:text-gray-300 leading-none">completed {this.state.query.filters.goal}</h1>
|
|
)
|
|
}
|
|
}
|
|
|
|
renderBody() {
|
|
if (this.state.loading) {
|
|
return (
|
|
<div className="loading mt-32 mx-auto"><div></div></div>
|
|
)
|
|
} else if (this.state.referrers) {
|
|
return (
|
|
<React.Fragment>
|
|
<h1 className="text-xl font-bold dark:text-gray-100">Referrer drilldown</h1>
|
|
|
|
<div className="my-4 border-b border-gray-300 dark:border-gray-500"></div>
|
|
<main className="modal__content mt-0">
|
|
<h1 className="text-xl font-semibold mb-0 leading-none dark:text-gray-200">{this.state.totalVisitors} visitors from {decodeURIComponent(this.props.match.params.referrer)}<br /> {toHuman(this.state.query)}</h1>
|
|
{this.renderGoalText()}
|
|
|
|
<table className="w-max overflow-x-auto md:w-full table-striped table-fixed mt-4">
|
|
<thead>
|
|
<tr>
|
|
<th className="p-2 w-48 md:w-56 lg:w-1/3 text-xs tracking-wide font-bold text-gray-500 dark:text-gray-400" align="left">Referrer</th>
|
|
{this.showConversionRate() && <th className="p-2 w-32 text-xs tracking-wide font-bold text-gray-500 dark:text-gray-400" align="right">Total visitors</th>}
|
|
<th className="p-2 w-32 text-xs tracking-wide font-bold text-gray-500 dark:text-gray-400" align="right">{this.label()}</th>
|
|
{this.showExtra() && <th className="p-2 w-32 text-xs tracking-wide font-bold text-gray-500 dark:text-gray-400" align="right">Bounce rate</th>}
|
|
{this.showExtra() && <th className="p-2 w-32 text-xs tracking-wide font-bold text-gray-500 dark:text-gray-400" align="right">Visit duration</th>}
|
|
{this.showConversionRate() && <th className="p-2 w-32 text-xs tracking-wide font-bold text-gray-500 dark:text-gray-400" align="right">CR</th>}
|
|
</tr>
|
|
</thead>
|
|
<tbody>
|
|
{ this.state.referrers.map(this.renderReferrer.bind(this)) }
|
|
</tbody>
|
|
</table>
|
|
</main>
|
|
</React.Fragment>
|
|
)
|
|
}
|
|
}
|
|
|
|
render() {
|
|
return (
|
|
<Modal site={this.props.site}>
|
|
{ this.renderBody() }
|
|
</Modal>
|
|
)
|
|
}
|
|
}
|
|
|
|
export default withRouter(ReferrerDrilldownModal)
|