2020-03-18 22:14:11 +01:00
// @flow
import * as ICONS from 'constants/icons' ;
import * as PAGES from 'constants/pages' ;
import React from 'react' ;
import { Lbryio } from 'lbryinc' ;
import ClaimPreview from 'component/claimPreview' ;
import Card from 'component/common/card' ;
import Spinner from 'component/spinner' ;
import Icon from 'component/common/icon' ;
import Button from 'component/button' ;
import Yrbl from 'component/yrbl' ;
import { useHistory } from 'react-router-dom' ;
2020-03-19 17:00:15 +01:00
import analytics from 'analytics' ;
2020-03-18 22:14:11 +01:00
type Props = {
2020-04-22 21:07:38 +02:00
claim : ? ChannelClaim ,
2020-03-18 22:14:11 +01:00
fetchingChannels : boolean ,
prepareEdit : string => void ,
} ;
2020-04-22 21:07:38 +02:00
const UNAUTHENTICATED _ERROR = 'unauthenticated' ;
const GENERIC _ERROR = 'error' ;
2020-03-19 15:38:20 +01:00
export default function CreatorAnalytics ( props : Props ) {
2020-04-22 21:07:38 +02:00
const { prepareEdit , claim } = props ;
2020-03-18 22:14:11 +01:00
const history = useHistory ( ) ;
const [ stats , setStats ] = React . useState ( ) ;
2020-04-22 21:07:38 +02:00
const [ error , setError ] = React . useState ( ) ;
2020-03-18 22:14:11 +01:00
const [ fetchingStats , setFetchingStats ] = React . useState ( false ) ;
2020-04-22 21:07:38 +02:00
const claimId = claim && claim . claim _id ;
const channelHasClaims = claim && claim . meta && claim . meta . claims _in _channel && claim . meta . claims _in _channel > 0 ;
2020-03-18 22:14:11 +01:00
React . useEffect ( ( ) => {
2020-04-22 21:07:38 +02:00
setStats ( null ) ;
} , [ claimId ] ) ;
2020-03-18 22:14:11 +01:00
2020-04-22 21:07:38 +02:00
const channelForEffect = JSON . stringify ( claim ) ;
2020-03-18 22:14:11 +01:00
React . useEffect ( ( ) => {
2020-04-23 22:38:09 +02:00
if ( claimId && channelForEffect && channelHasClaims ) {
2020-03-18 22:14:11 +01:00
setFetchingStats ( true ) ;
2020-04-22 21:07:38 +02:00
Lbryio . call ( 'reports' , 'content' , { claim _id : claimId } )
2020-03-18 22:14:11 +01:00
. then ( res => {
setFetchingStats ( false ) ;
setStats ( res ) ;
} )
2020-04-22 21:07:38 +02:00
. catch ( error => {
if ( error . response . status === 401 ) {
setError ( UNAUTHENTICATED _ERROR ) ;
const channelToSend = JSON . parse ( channelForEffect ) ;
analytics . apiLogPublish ( channelToSend ) ;
} else {
setError ( GENERIC _ERROR ) ;
}
2020-03-18 22:14:11 +01:00
setFetchingStats ( false ) ;
} ) ;
}
2020-04-23 22:38:09 +02:00
} , [ claimId , channelForEffect , channelHasClaims , setFetchingStats , setStats ] ) ;
2020-03-18 22:14:11 +01:00
return (
< React.Fragment >
2020-04-22 21:07:38 +02:00
{ ! stats && (
2020-03-18 22:14:11 +01:00
< div className = "main--empty" >
2020-04-22 21:07:38 +02:00
{ fetchingStats ? (
< Spinner delayed / >
) : (
< div >
{ error && (
< Yrbl
type = "sad"
title = { error === GENERIC _ERROR ? _ _ ( 'No Stats Found' ) : _ _ ( 'Error Fetching Stats' ) }
subtitle = {
error === GENERIC _ERROR
? _ _ (
'There are no stats for this channel. Make sure you are signed in with the correct email and have data sharing turned on.'
)
: _ _ (
"You are not able to see this channel's stats. Make sure you are signed in with the correct email and have data sharing turned on."
)
2020-03-18 22:14:11 +01:00
}
2020-04-22 21:07:38 +02:00
/ >
) }
{ ! error && (
< Yrbl
title = {
channelHasClaims
? _ _ ( 'No recent publishes' )
: _ _ ( "You haven't published anything with this channel yet!" )
}
subtitle = {
< Button
button = "primary"
label = { _ _ ( 'Publish Something' ) }
onClick = { ( ) => {
if ( claim ) {
prepareEdit ( claim . name ) ;
history . push ( ` / $ / ${ PAGES . PUBLISH } ` ) ;
}
} }
/ >
}
/ >
) }
< / div >
) }
< / div >
2020-03-18 22:14:11 +01:00
) }
2020-04-22 21:07:38 +02:00
2020-03-18 22:14:11 +01:00
{ stats && (
< div className = "section" >
< div className = "columns" >
< Card
iconColor
2020-04-22 21:07:38 +02:00
title = { < span > { _ _ ( '%follower_count% followers' , { follower _count : stats . ChannelSubs } ) } < / span > }
2020-03-18 22:14:11 +01:00
icon = { ICONS . SUBSCRIBE }
subtitle = {
< div className = "card__data-subtitle" >
< span >
2020-03-19 17:42:43 +01:00
{ 0 > - 1 && '+' } { ' ' }
2020-03-19 15:38:20 +01:00
{ _ _ ( '%follower_count_weekly_change% this week' , {
follower _count _weekly _change : stats . ChannelSubChange || 0 ,
} ) }
2020-03-18 22:14:11 +01:00
< / span >
{ stats . ChannelSubChange > 0 && < Icon icon = { ICONS . SUPPORT } iconColor = "green" size = { 18 } / > }
< / div >
}
/ >
< Card
icon = { ICONS . EYE }
2020-03-19 15:38:20 +01:00
title = { < span > { _ _ ( '%all_content_views% views' , { all _content _views : stats . AllContentViews } ) } < / span > }
subtitle = {
2020-03-19 17:42:43 +01:00
< div className = "card__data-subtitle" >
< span >
{ _ _ ( '+ %all_content_views_weekly_change% this week' , {
all _content _views _weekly _change : stats . AllContentViewChange || 0 ,
} ) }
< / span >
{ stats . AllContentViewChange > 0 && < Icon icon = { ICONS . SUPPORT } iconColor = "green" size = { 18 } / > }
< / div >
2020-03-19 15:38:20 +01:00
}
2020-03-18 22:14:11 +01:00
/ >
< / div >
2020-04-23 23:40:58 +02:00
{ / * < C a r d
2020-04-22 21:07:38 +02:00
iconColor
className = "section"
title = { < span > { _ _ ( '%lbc_received% LBC Earned' , { lbc _received : stats . AllLBCReceived } ) } < / span > }
2020-04-23 18:15:51 +02:00
icon = { ICONS . REWARDS }
2020-04-22 21:07:38 +02:00
subtitle = {
< React.Fragment >
< div className = "card__data-subtitle" >
< span >
{ '+' } { ' ' }
{ _ _ ( '%lbc_received_changed% this week' , {
lbc _received _changed : stats . LBCReceivedChange || 0 ,
} ) }
< / span >
{ stats . LBCReceivedChange > 0 && < Icon icon = { ICONS . SUPPORT } iconColor = "green" size = { 18 } / > }
< / div >
< p className = "help" >
{ _ _ (
"Earnings may also include any LBC you've sent yourself or added as support. We are working on making this more accurate. Check your wallet page for the correct total balance."
) }
< / p >
< / React.Fragment >
2020-03-18 22:14:11 +01:00
}
2020-04-23 23:40:58 +02:00
/> */ }
2020-04-22 21:07:38 +02:00
{ stats . VideoURITopNew ? (
< Card
className = "section"
title = { _ _ ( 'Your Recent Content' ) }
body = {
< React.Fragment >
< div className = "card--inline" >
2020-04-23 22:38:09 +02:00
< ClaimPreview uri = { stats . VideoURITopNew } / >
2020-04-22 21:07:38 +02:00
< / div >
< div className = "section__subtitle card__data-subtitle" >
< span >
{ _ _ ( '%view_count% %views%' , {
view _count : stats . VideoViewsTopNew ,
views : stats . VideoViewsTopNew === 1 ? 'view' : 'views' ,
} ) }
< / span >
{ stats . VideoViewsTopNew > 0 && < Icon icon = { ICONS . SUPPORT } iconColor = "green" size = { 18 } / > }
< / div >
< / React.Fragment >
}
/ >
) : (
< Card
className = "section"
title = { _ _ ( 'Your Recent Content' ) }
subtitle = {
! stats . VideoURITopNew &&
_ _ (
"No recent publishes found for this channel. Publish something new and track how it's performing here."
)
}
actions = {
< div className = "section__actions" >
< Button
button = "primary"
label = { _ _ ( 'New Publish' ) }
onClick = { ( ) => {
if ( claim ) {
prepareEdit ( claim . name ) ;
history . push ( ` / $ / ${ PAGES . PUBLISH } ` ) ;
}
} }
/ >
< / div >
}
/ >
) }
< Card
className = "section"
title = { _ _ ( 'Most Viewed Content' ) }
2020-03-18 22:14:11 +01:00
body = {
< React.Fragment >
< div className = "card--inline" >
2020-03-19 15:38:20 +01:00
< ClaimPreview uri = { stats . VideoURITopAllTime } / >
< / div >
< div className = "section__subtitle card__data-subtitle" >
< span >
{ _ _ ( '%all_time_top_views% views - %all_time_views_weekly_change% this week' , {
all _time _top _views : stats . VideoViewsTopAllTime ,
all _time _views _weekly _change : stats . VideoViewChangeTopAllTime ,
} ) }
< / span >
{ stats . VideoViewChangeTopAllTime > 0 && < Icon icon = { ICONS . SUPPORT } iconColor = "green" size = { 18 } / > }
2020-03-18 22:14:11 +01:00
< / div >
< / React.Fragment >
}
/ >
< / div >
) }
< / React.Fragment >
) ;
}