bsky-app/src/view/shell/mobile/TabsSelector.tsx
Paul Frazee 7e31645e9a Add a design system (#34)
* Add theming system

* Add standard Button control and update RadioButtons

* Unify radiobutton with design system

* Update debug screen to have multiple views

* Add ToggleButton

* Update error controls to use design system

* Add typography to <Text> element

* Move DropdownButton into the design system

* Clean out old code

* Move Text into design system

* Add 'inverted' color palette

* Move LoadingPlaceholder into the design system
2022-12-28 14:06:01 -06:00

314 lines
9 KiB
TypeScript

import React, {createRef, useRef, useMemo, useState} from 'react'
import {observer} from 'mobx-react-lite'
import {
Animated,
ScrollView,
Share,
StyleSheet,
TouchableWithoutFeedback,
View,
} from 'react-native'
import {useSafeAreaInsets} from 'react-native-safe-area-context'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {Text} from '../../com/util/text/Text'
import Swipeable from 'react-native-gesture-handler/Swipeable'
import {useStores} from '../../../state'
import {s, colors} from '../../lib/styles'
import {toShareUrl} from '../../../lib/strings'
import {match} from '../../routes'
import {useAnimatedValue} from '../../lib/hooks/useAnimatedValue'
const TAB_HEIGHT = 42
export const TabsSelector = observer(
({
active,
tabMenuInterp,
onClose,
}: {
active: boolean
tabMenuInterp: Animated.Value
onClose: () => void
}) => {
const store = useStores()
const insets = useSafeAreaInsets()
const [closingTabIndex, setClosingTabIndex] = useState<number | undefined>(
undefined,
)
const closeInterp = useAnimatedValue(0)
const tabsRef = useRef<ScrollView>(null)
const tabRefs = useMemo(
() =>
Array.from({length: store.nav.tabs.length}).map(() =>
createRef<Animated.View>(),
),
[store.nav.tabs.length],
)
const wrapperAnimStyle = {
transform: [
{
translateY: tabMenuInterp.interpolate({
inputRange: [0, 1.0],
outputRange: [320, 0],
}),
},
],
}
// events
// =
const onPressNewTab = () => {
store.nav.newTab('/')
onClose()
}
const onPressCloneTab = () => {
store.nav.newTab(store.nav.tab.current.url)
onClose()
}
const onPressShareTab = () => {
onClose()
Share.share({url: toShareUrl(store.nav.tab.current.url)})
}
const onPressChangeTab = (tabIndex: number) => {
store.nav.setActiveTab(tabIndex)
onClose()
}
const onCloseTab = (tabIndex: number) => {
setClosingTabIndex(tabIndex)
closeInterp.setValue(0)
Animated.timing(closeInterp, {
toValue: 1,
duration: 300,
useNativeDriver: false,
}).start(() => {
setClosingTabIndex(undefined)
store.nav.closeTab(tabIndex)
})
}
const onLayout = () => {
// focus the current tab
const targetTab = tabRefs[store.nav.tabIndex]
if (tabsRef.current && targetTab.current) {
targetTab.current.measureLayout?.(
tabsRef.current,
(_left: number, top: number) => {
tabsRef.current?.scrollTo({y: top, animated: false})
},
() => {},
)
}
}
// rendering
// =
const renderSwipeActions = () => {
return <View style={[s.p2]} />
}
const currentTabIndex = store.nav.tabIndex
const closingTabAnimStyle = {
height: Animated.multiply(TAB_HEIGHT, Animated.subtract(1, closeInterp)),
opacity: Animated.subtract(1, closeInterp),
marginBottom: Animated.multiply(4, Animated.subtract(1, closeInterp)),
}
if (!active) {
return <View />
}
return (
<Animated.View
style={[
styles.wrapper,
{bottom: insets.bottom + 55},
wrapperAnimStyle,
]}>
<View onLayout={onLayout}>
<View style={[s.p10, styles.section]}>
<View style={styles.btns}>
<TouchableWithoutFeedback onPress={onPressShareTab}>
<View style={[styles.btn]}>
<View style={styles.btnIcon}>
<FontAwesomeIcon size={16} icon="share" />
</View>
<Text style={styles.btnText}>Share</Text>
</View>
</TouchableWithoutFeedback>
<TouchableWithoutFeedback onPress={onPressCloneTab}>
<View style={[styles.btn]}>
<View style={styles.btnIcon}>
<FontAwesomeIcon size={16} icon={['far', 'clone']} />
</View>
<Text style={styles.btnText}>Clone tab</Text>
</View>
</TouchableWithoutFeedback>
<TouchableWithoutFeedback onPress={onPressNewTab}>
<View style={[styles.btn]}>
<View style={styles.btnIcon}>
<FontAwesomeIcon size={16} icon="plus" />
</View>
<Text style={styles.btnText}>New tab</Text>
</View>
</TouchableWithoutFeedback>
</View>
</View>
<View style={[s.p10, styles.section, styles.sectionGrayBg]}>
<ScrollView ref={tabsRef} style={styles.tabs}>
{store.nav.tabs.map((tab, tabIndex) => {
const {icon} = match(tab.current.url)
const isActive = tabIndex === currentTabIndex
const isClosing = closingTabIndex === tabIndex
return (
<Swipeable
key={tab.id}
renderLeftActions={renderSwipeActions}
renderRightActions={renderSwipeActions}
leftThreshold={100}
rightThreshold={100}
onSwipeableWillOpen={() => onCloseTab(tabIndex)}>
<Animated.View
style={[
styles.tabOuter,
isClosing ? closingTabAnimStyle : undefined,
]}>
<Animated.View
// HOTFIX
// TabsSelector.test.tsx snapshot fails if the
// ref was set like this: ref={tabRefs[tabIndex]}
ref={(ref: any) => (tabRefs[tabIndex] = ref)}
style={[
styles.tab,
styles.existing,
isActive && styles.active,
]}>
<TouchableWithoutFeedback
onPress={() => onPressChangeTab(tabIndex)}>
<View style={styles.tabInner}>
<View style={styles.tabIcon}>
<FontAwesomeIcon size={20} icon={icon} />
</View>
<Text
ellipsizeMode="tail"
numberOfLines={1}
suppressHighlighting={true}
style={[
styles.tabText,
isActive && styles.tabTextActive,
]}>
{tab.current.title || tab.current.url}
</Text>
</View>
</TouchableWithoutFeedback>
<TouchableWithoutFeedback
onPress={() => onCloseTab(tabIndex)}>
<View style={styles.tabClose}>
<FontAwesomeIcon
size={14}
icon="x"
style={styles.tabCloseIcon}
/>
</View>
</TouchableWithoutFeedback>
</Animated.View>
</Animated.View>
</Swipeable>
)
})}
</ScrollView>
</View>
</View>
</Animated.View>
)
},
)
const styles = StyleSheet.create({
wrapper: {
position: 'absolute',
width: '100%',
height: 320,
borderTopColor: colors.gray2,
borderTopWidth: 1,
backgroundColor: '#fff',
opacity: 1,
},
section: {
borderBottomColor: colors.gray2,
borderBottomWidth: 1,
},
sectionGrayBg: {
backgroundColor: colors.gray1,
},
tabs: {
height: 240,
},
tabOuter: {
height: TAB_HEIGHT + 4,
overflow: 'hidden',
},
tab: {
flexDirection: 'row',
height: TAB_HEIGHT,
backgroundColor: colors.gray1,
alignItems: 'center',
borderRadius: 4,
},
tabInner: {
flexDirection: 'row',
flex: 1,
alignItems: 'center',
paddingLeft: 12,
paddingVertical: 12,
},
existing: {
borderColor: colors.gray4,
borderWidth: 1,
},
active: {
backgroundColor: colors.white,
borderColor: colors.black,
borderWidth: 1,
},
tabIcon: {},
tabText: {
flex: 1,
paddingHorizontal: 10,
fontSize: 16,
},
tabTextActive: {
fontWeight: '500',
},
tabClose: {
paddingVertical: 16,
paddingRight: 16,
},
tabCloseIcon: {
color: '#655',
},
btns: {
flexDirection: 'row',
paddingTop: 2,
},
btn: {
flexDirection: 'row',
flex: 1,
alignItems: 'center',
justifyContent: 'center',
backgroundColor: colors.gray1,
borderRadius: 4,
marginRight: 5,
paddingLeft: 12,
paddingRight: 16,
paddingVertical: 10,
},
btnIcon: {
marginRight: 8,
},
btnText: {
fontWeight: '500',
fontSize: 16,
},
})