2018-03-26 14:32:43 -07:00
// @flow
2021-12-21 09:42:28 -03:00
import 'scss/component/_header.scss' ;
2021-12-21 10:21:14 -03:00
import { formatCredits } from 'util/format-credits' ;
2021-12-20 09:29:51 -03:00
import { useIsMobile } from 'effects/use-screensize' ;
import { withRouter } from 'react-router' ;
2018-11-25 20:21:25 -05:00
import * as ICONS from 'constants/icons' ;
2019-08-21 16:54:44 -04:00
import * as PAGES from 'constants/pages' ;
2018-03-26 14:32:43 -07:00
import Button from 'component/button' ;
2021-12-20 09:29:51 -03:00
import classnames from 'classnames' ;
import HeaderMenuButtons from 'component/headerMenuButtons' ;
import HeaderProfileMenuButton from 'component/headerProfileMenuButton' ;
import Logo from 'component/logo' ;
2020-08-10 16:47:39 -04:00
import NotificationBubble from 'component/notificationBubble' ;
2021-12-20 09:29:51 -03:00
import React from 'react' ;
2021-12-21 10:21:14 -03:00
import Skeleton from '@mui/material/Skeleton' ;
2021-07-20 19:31:26 -05:00
import SkipNavigationButton from 'component/skipNavigationButton' ;
2021-12-20 09:38:12 -03:00
import Tooltip from 'component/common/tooltip' ;
2021-12-20 09:29:51 -03:00
import WunderBar from 'component/wunderbar' ;
2019-10-09 12:34:18 -04:00
2018-03-26 14:32:43 -07:00
type Props = {
2020-01-03 14:34:17 -05:00
authenticated : boolean ,
2019-10-28 10:04:37 -04:00
authHeader : boolean ,
2020-06-30 01:51:15 -04:00
backout : {
2020-07-10 17:04:36 -04:00
backLabel ? : string ,
backNavDefault ? : string ,
2020-06-30 01:51:15 -04:00
title : string ,
simpleTitle : string , // Just use the same value as `title` if `title` is already short (~< 10 chars), unless you have a better idea for title overlfow on mobile
} ,
2021-12-20 09:29:51 -03:00
balance : number ,
2020-04-13 15:16:07 -04:00
emailToVerify ? : string ,
2020-07-10 17:04:36 -04:00
hasNavigated : boolean ,
2021-12-20 09:29:51 -03:00
hideBalance : boolean ,
hideCancel : boolean ,
history : {
goBack : ( ) => void ,
location : { pathname : string } ,
push : ( string ) => void ,
replace : ( string ) => void ,
} ,
isAbsoluteSideNavHidden : boolean ,
2020-08-10 16:47:39 -04:00
sidebarOpen : boolean ,
2021-12-20 09:29:51 -03:00
syncError : ? string ,
2021-12-21 10:21:14 -03:00
totalBalance ? : number ,
2021-12-20 09:29:51 -03:00
user : ? User ,
clearEmailEntry : ( ) => void ,
clearPasswordEntry : ( ) => void ,
openChangelog : ( { } ) => void ,
2021-03-03 13:50:16 -05:00
setSidebarOpen : ( boolean ) => void ,
2021-12-20 09:29:51 -03:00
signOut : ( ) => void ,
2018-03-26 14:32:43 -07:00
} ;
const Header = ( props : Props ) => {
2019-08-21 16:54:44 -04:00
const {
2021-12-20 09:29:51 -03:00
authenticated ,
authHeader ,
backout ,
2021-01-25 13:34:21 -05:00
balance ,
2021-12-20 09:29:51 -03:00
emailToVerify ,
hideBalance ,
hideCancel ,
history ,
isAbsoluteSideNavHidden ,
sidebarOpen ,
2019-11-01 12:19:28 -04:00
syncError ,
2021-12-21 10:21:14 -03:00
totalBalance ,
2021-12-20 09:29:51 -03:00
user ,
2020-04-13 15:16:07 -04:00
clearEmailEntry ,
clearPasswordEntry ,
2021-12-20 09:29:51 -03:00
openChangelog ,
2020-08-10 16:47:39 -04:00
setSidebarOpen ,
2021-12-20 09:29:51 -03:00
signOut ,
2019-08-21 16:54:44 -04:00
} = props ;
2021-12-20 09:29:51 -03:00
const {
location : { pathname } ,
goBack ,
push ,
} = history ;
2020-06-30 01:51:15 -04:00
const isMobile = useIsMobile ( ) ;
2021-12-20 09:29:51 -03:00
2019-11-22 16:13:00 -05:00
// on the verify page don't let anyone escape other than by closing the tab to keep session data consistent
2021-12-20 09:29:51 -03:00
const isVerifyPage = pathname . includes ( PAGES . AUTH _VERIFY ) ;
const isSignUpPage = pathname . includes ( PAGES . AUTH ) ;
const isSignInPage = pathname . includes ( PAGES . AUTH _SIGNIN ) ;
const isPwdResetPage = pathname . includes ( PAGES . AUTH _PASSWORD _RESET ) ;
2021-12-21 16:01:27 -03:00
const iYTSyncPage = pathname . includes ( PAGES . YOUTUBE _SYNC ) ;
2021-12-20 09:29:51 -03:00
// For pages that allow for "backing out", shows a backout option instead of the Home logo
const canBackout = Boolean ( backout ) ;
2020-07-30 15:14:22 -04:00
const { backLabel , backNavDefault , title : backTitle , simpleTitle : simpleBackTitle } = backout || { } ;
2021-02-09 11:05:56 -05:00
2021-12-21 10:21:14 -03:00
const balanceLoading = totalBalance === undefined ;
const roundedSpendableBalance = formatCredits ( balance , 2 , true ) ;
const roundedTotalBalance = formatCredits ( totalBalance , 2 , true ) ;
2019-11-01 12:19:28 -04:00
// Sign out if they click the "x" when they are on the password prompt
2021-12-20 09:29:51 -03:00
const authHeaderAction = syncError && { onClick : signOut } ;
const homeButtonNavigationProps = ( isVerifyPage && { } ) || ( authHeader && authHeaderAction ) || { navigate : '/' } ;
const sidebarLabel = sidebarOpen
? _ _ ( 'Close sidebar - hide channels you are following.' )
: _ _ ( 'Expand sidebar - view channels you are following.' ) ;
2018-03-26 14:32:43 -07:00
2021-12-20 09:29:51 -03:00
const onBackout = React . useCallback (
( e : any ) => {
const { hasNavigated } = props ;
const { replace } = history ;
2020-07-10 17:04:36 -04:00
2021-12-20 09:29:51 -03:00
window . removeEventListener ( 'popstate' , onBackout ) ;
2020-07-10 17:04:36 -04:00
2021-12-20 09:29:51 -03:00
if ( e . type !== 'popstate' ) {
// if not initiated by pop (back button)
if ( hasNavigated && ! backNavDefault ) {
goBack ( ) ;
} else {
replace ( backNavDefault || ` / ` ) ;
}
2020-07-10 17:04:36 -04:00
}
2021-12-20 09:29:51 -03:00
} ,
[ backNavDefault , goBack , history , props ]
) ;
2020-07-10 17:04:36 -04:00
2020-07-27 10:43:55 -04:00
React . useEffect ( ( ) => {
2021-12-20 09:29:51 -03:00
if ( canBackout ) {
2020-07-27 10:43:55 -04:00
window . addEventListener ( 'popstate' , onBackout ) ;
return ( ) => window . removeEventListener ( 'popstate' , onBackout ) ;
}
2021-12-20 09:29:51 -03:00
} , [ canBackout , onBackout ] ) ;
2020-07-27 10:43:55 -04:00
2021-12-21 09:42:28 -03:00
const userButtons = ( hideWallet ? : boolean , hideProfile ? : boolean ) => (
< div className = "header__menu--right" >
{ isMobile && ! authHeader && ! canBackout && < WunderBar / > }
2021-12-20 09:29:51 -03:00
{ authenticated ? (
< >
2021-12-21 09:42:28 -03:00
{ ! hideWallet && (
< Tooltip
title = {
balance > 0
? _ _ ( 'Immediately spendable: %spendable_balance%' , { spendable _balance : roundedSpendableBalance } )
: _ _ ( 'Your Wallet' )
}
>
< div >
2021-12-21 10:21:14 -03:00
{ balanceLoading ? (
< Skeleton variant = "text" animation = "wave" className = "header__navigationItem--balanceLoading" / >
) : (
< Button
navigate = { ` / $ / ${ PAGES . WALLET } ` }
className = "button--file-action header__navigationItem--balance"
2021-12-21 15:49:18 -03:00
label = {
hideBalance || Number ( roundedTotalBalance ) === 0
? isMobile
? _ _ ( 'Wallet' )
: _ _ ( 'Your Wallet' )
: roundedTotalBalance
}
2021-12-21 10:21:14 -03:00
icon = { ICONS . LBC }
/ >
) }
2021-12-21 09:42:28 -03:00
< / div >
< / Tooltip >
) }
2018-03-26 14:32:43 -07:00
2021-12-21 09:42:28 -03:00
{ ! hideProfile && < HeaderProfileMenuButton / > }
2021-12-20 09:29:51 -03:00
< / >
2021-12-21 12:09:19 -03:00
) : ! isMobile ? (
2021-12-21 09:42:28 -03:00
< div className = "header__authButtons" >
2021-12-21 12:09:19 -03:00
< Button navigate = { ` / $ / ${ PAGES . AUTH _SIGNIN } ` } button = "link" label = { _ _ ( 'Log In' ) } disabled = { user === null } / >
2021-12-20 09:29:51 -03:00
< Button navigate = { ` / $ / ${ PAGES . AUTH } ` } button = "primary" label = { _ _ ( 'Sign Up' ) } disabled = { user === null } / >
< / div >
2021-12-21 12:09:19 -03:00
) : (
< HeaderProfileMenuButton / >
2021-12-20 09:29:51 -03:00
) }
2020-11-10 00:21:04 -05:00
< / div >
) ;
2017-06-06 17:19:12 -04:00
return (
2021-12-20 09:29:51 -03:00
< header className = { classnames ( 'header' , { 'header--minimal' : authHeader } ) } >
2021-12-21 09:42:28 -03:00
{ ! authHeader && canBackout ? (
< div className = "card__actions--between header__contents" >
< div className = "header__menu--left" >
2021-12-20 09:29:51 -03:00
< Button onClick = { onBackout } button = "link" label = { backLabel || _ _ ( 'Cancel' ) } icon = { ICONS . ARROW _LEFT } / >
2021-12-21 09:42:28 -03:00
< / div >
2021-12-20 09:29:51 -03:00
2021-12-21 09:42:28 -03:00
{ backTitle && < h1 className = "header__authTitle" > { ( isMobile && simpleBackTitle ) || backTitle } < / h1 > }
2021-12-20 09:29:51 -03:00
2021-12-21 09:42:28 -03:00
{ userButtons ( false , isMobile ) }
< / div >
) : (
< >
< div className = "header__navigation" >
< div className = "header__menu--left" >
2021-07-15 13:32:58 -05:00
< SkipNavigationButton / >
2021-12-20 09:29:51 -03:00
2020-08-11 16:32:03 -04:00
{ ! authHeader && (
< span style = { { position : 'relative' } } >
< Button
2021-12-20 09:29:51 -03:00
aria - label = { sidebarLabel }
2021-12-21 09:42:28 -03:00
className = "header__navigationItem--icon"
2020-08-11 16:32:03 -04:00
icon = { ICONS . MENU }
2021-07-13 16:23:29 -05:00
aria - expanded = { sidebarOpen }
2020-08-11 16:32:03 -04:00
onClick = { ( ) => setSidebarOpen ( ! sidebarOpen ) }
>
2021-12-20 09:29:51 -03:00
{ isAbsoluteSideNavHidden && isMobile && < NotificationBubble / > }
2020-08-11 16:32:03 -04:00
< / Button >
< / span >
) }
2021-12-20 09:29:51 -03:00
2020-06-29 15:54:07 -04:00
< Button
2021-07-14 22:08:30 -05:00
aria - label = { _ _ ( 'Home' ) }
2021-12-21 09:42:28 -03:00
className = "header__navigationItem--logo"
2021-12-20 09:29:51 -03:00
onClick = { ( ) => pathname === '/' && window . location . reload ( ) }
2020-06-29 15:54:07 -04:00
{ ... homeButtonNavigationProps }
2021-07-21 11:33:28 -04:00
>
< Logo / >
< / Button >
2021-12-13 16:17:30 +08:00
{ /* @if process.env.DEV_CHANGELOG */ }
{ history . location . pathname === '/' && (
< Button
2021-12-20 09:29:51 -03:00
title = "Changelog"
2021-12-13 16:17:30 +08:00
className = "badge--alert"
2021-12-20 09:29:51 -03:00
label = "Changelog"
2021-12-13 16:17:30 +08:00
icon = { ICONS . FEEDBACK }
2021-12-20 09:29:51 -03:00
onClick = { ( ) =>
openChangelog ( {
2021-12-13 16:17:30 +08:00
title : _ _ ( 'Changelog' ) ,
subtitle : _ _ ( 'Warning: this is a test instance.' ) ,
body : < p style = { { whiteSpace : 'pre-wrap' } } > { process . env . DEV _CHANGELOG } < / p > ,
onConfirm : ( closeModal ) => closeModal ( ) ,
hideCancel : true ,
2021-12-20 09:29:51 -03:00
} )
}
2021-12-13 16:17:30 +08:00
/ >
) }
{ /* @endif */ }
2020-08-10 16:47:39 -04:00
< / div >
2021-12-21 09:42:28 -03:00
{ ! authHeader && ! isMobile && (
< div className = "header__center" >
< WunderBar / >
< HeaderMenuButtons / >
< / div >
) }
2021-12-20 09:29:51 -03:00
{ ! authHeader && ! canBackout
2021-12-21 09:42:28 -03:00
? userButtons ( isMobile )
2021-12-20 09:29:51 -03:00
: ! isVerifyPage &&
! hideCancel && (
2021-12-21 09:42:28 -03:00
< div className = "header__menu--right" >
2021-12-20 09:29:51 -03:00
< Button
title = { _ _ ( 'Go Back' ) }
button = "alt"
// className="button--header-close"
icon = { ICONS . REMOVE }
onClick = { ( ) => {
2021-12-29 17:08:04 -03:00
if ( ! iYTSyncPage && ! isPwdResetPage ) {
2021-12-21 16:01:27 -03:00
clearEmailEntry ( ) ;
clearPasswordEntry ( ) ;
}
2021-03-04 01:03:58 -05:00
2021-12-20 09:29:51 -03:00
if ( syncError ) signOut ( ) ;
2021-03-04 01:03:58 -05:00
2021-12-21 16:01:27 -03:00
if ( ( isSignInPage && ! emailToVerify ) || isSignUpPage || isPwdResetPage || iYTSyncPage ) {
2021-12-20 09:29:51 -03:00
goBack ( ) ;
} else {
push ( '/' ) ;
}
} }
/ >
< / div >
2021-03-04 01:03:58 -05:00
) }
2021-12-21 09:42:28 -03:00
< / div >
< / >
) }
2017-06-06 17:19:12 -04:00
< / header >
) ;
2017-06-05 21:21:55 -07:00
} ;
2016-08-27 10:12:56 -04:00
2019-06-17 16:32:38 -04:00
export default withRouter ( Header ) ;