* Get home screen's swipable pager working with the drawer * Add tab bar to pager * Implement popular & following views on home screen * Visual tune-up * Move the feed selector to the footer * Fix to 'new posts' poll * Add the view header as a feed item * Use the native driver on the tabbar indicator to improve perf * Reduce home polling to the currently active page; also reuse some code * Add soft reset on tap selected in tab bar * Remove explicit 'onboarding' flow * Choose good stuff based on service * Add foaf-based follow discovery * Fall back to who to follow * Fix backgrounds * Switch to the off-spec goodstuff route * 1.8 * Fix for dev & staging * Swap the tab bar items and rename suggested to what's hot * Go to whats-hot by default if you have no follows * Implement pager and tabbar for desktop web * Pin deps to make expo happy * Add language filtering to goodstuff
203 lines
5.1 KiB
TypeScript
203 lines
5.1 KiB
TypeScript
import React from 'react'
|
|
import {StyleSheet, StyleProp, View, ViewStyle} from 'react-native'
|
|
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
|
|
import {HeartIcon} from 'lib/icons'
|
|
import {s} from 'lib/styles'
|
|
import {useTheme} from 'lib/ThemeContext'
|
|
import {usePalette} from 'lib/hooks/usePalette'
|
|
|
|
export function LoadingPlaceholder({
|
|
width,
|
|
height,
|
|
style,
|
|
}: {
|
|
width: string | number
|
|
height: string | number
|
|
style?: StyleProp<ViewStyle>
|
|
}) {
|
|
const theme = useTheme()
|
|
return (
|
|
<View
|
|
style={[
|
|
styles.loadingPlaceholder,
|
|
{
|
|
width,
|
|
height,
|
|
backgroundColor: theme.palette.default.backgroundLight,
|
|
},
|
|
style,
|
|
]}
|
|
/>
|
|
)
|
|
}
|
|
|
|
export function PostLoadingPlaceholder({
|
|
style,
|
|
}: {
|
|
style?: StyleProp<ViewStyle>
|
|
}) {
|
|
const theme = useTheme()
|
|
const pal = usePalette('default')
|
|
return (
|
|
<View style={[styles.post, pal.view, style]}>
|
|
<LoadingPlaceholder width={52} height={52} style={styles.avatar} />
|
|
<View style={[s.flex1]}>
|
|
<LoadingPlaceholder width={100} height={8} style={[s.mb10]} />
|
|
<LoadingPlaceholder width={200} height={8} style={[s.mb5]} />
|
|
<LoadingPlaceholder width={200} height={8} style={[s.mb5]} />
|
|
<LoadingPlaceholder width={120} height={8} style={[s.mb10]} />
|
|
<View style={s.flexRow}>
|
|
<View style={s.flex1}>
|
|
<FontAwesomeIcon
|
|
style={{color: theme.palette.default.icon}}
|
|
icon={['far', 'comment']}
|
|
size={14}
|
|
/>
|
|
</View>
|
|
<View style={s.flex1}>
|
|
<FontAwesomeIcon
|
|
style={{color: theme.palette.default.icon}}
|
|
icon="retweet"
|
|
size={18}
|
|
/>
|
|
</View>
|
|
<View style={s.flex1}>
|
|
<HeartIcon
|
|
style={{color: theme.palette.default.icon} as ViewStyle}
|
|
size={17}
|
|
strokeWidth={1.7}
|
|
/>
|
|
</View>
|
|
<View style={s.flex1} />
|
|
</View>
|
|
</View>
|
|
</View>
|
|
)
|
|
}
|
|
|
|
export function PostFeedLoadingPlaceholder() {
|
|
return (
|
|
<>
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
<PostLoadingPlaceholder />
|
|
</>
|
|
)
|
|
}
|
|
|
|
export function NotificationLoadingPlaceholder({
|
|
style,
|
|
}: {
|
|
style?: StyleProp<ViewStyle>
|
|
}) {
|
|
const pal = usePalette('default')
|
|
return (
|
|
<View style={[styles.notification, pal.view, style]}>
|
|
<View style={[s.flexRow, s.mb10]}>
|
|
<LoadingPlaceholder width={30} height={30} style={styles.smallAvatar} />
|
|
</View>
|
|
<LoadingPlaceholder width={200} height={8} style={[s.mb5]} />
|
|
<LoadingPlaceholder width={120} height={8} style={[s.mb5]} />
|
|
</View>
|
|
)
|
|
}
|
|
|
|
export function NotificationFeedLoadingPlaceholder() {
|
|
return (
|
|
<>
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
<NotificationLoadingPlaceholder />
|
|
</>
|
|
)
|
|
}
|
|
|
|
export function ProfileCardLoadingPlaceholder({
|
|
style,
|
|
}: {
|
|
style?: StyleProp<ViewStyle>
|
|
}) {
|
|
const pal = usePalette('default')
|
|
return (
|
|
<View style={[styles.profileCard, pal.view, style]}>
|
|
<LoadingPlaceholder
|
|
width={40}
|
|
height={40}
|
|
style={styles.profileCardAvi}
|
|
/>
|
|
<View>
|
|
<LoadingPlaceholder width={140} height={8} style={[s.mb5]} />
|
|
<LoadingPlaceholder width={120} height={8} style={[s.mb10]} />
|
|
<LoadingPlaceholder width={220} height={8} style={[s.mb5]} />
|
|
</View>
|
|
</View>
|
|
)
|
|
}
|
|
|
|
export function ProfileCardFeedLoadingPlaceholder() {
|
|
return (
|
|
<>
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
<ProfileCardLoadingPlaceholder />
|
|
</>
|
|
)
|
|
}
|
|
|
|
const styles = StyleSheet.create({
|
|
loadingPlaceholder: {
|
|
borderRadius: 6,
|
|
},
|
|
post: {
|
|
flexDirection: 'row',
|
|
padding: 10,
|
|
margin: 1,
|
|
},
|
|
avatar: {
|
|
borderRadius: 26,
|
|
marginRight: 10,
|
|
marginLeft: 6,
|
|
},
|
|
notification: {
|
|
padding: 10,
|
|
paddingLeft: 46,
|
|
margin: 1,
|
|
},
|
|
profileCard: {
|
|
flexDirection: 'row',
|
|
padding: 10,
|
|
margin: 1,
|
|
},
|
|
profileCardAvi: {
|
|
borderRadius: 20,
|
|
marginRight: 10,
|
|
},
|
|
smallAvatar: {
|
|
borderRadius: 15,
|
|
marginRight: 10,
|
|
},
|
|
})
|