Skip to content

Commit

Permalink
feat(toolbar): add toolbar component
Browse files Browse the repository at this point in the history
  • Loading branch information
Artur Bien authored and arturbien committed Jan 18, 2021
1 parent cbd93dd commit 7bdc5ff
Show file tree
Hide file tree
Showing 6 changed files with 251 additions and 1 deletion.
129 changes: 129 additions & 0 deletions example/src/examples/ToolbarExample.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,129 @@
import React from 'react';
import { View, StyleSheet, Image } from 'react-native';
import { Divider, Toolbar, AppBar } from 'react95-native';

import { TouchableOpacity } from 'react-native-gesture-handler';
import { notificationService } from '../util/notifications';

const icons = [
{
label: 'Notepad',
uri:
'',
},
{
label: 'PowerOff',
uri:
'',
},
{
label: 'Tree',
uri:
'',
},
{
label: 'Unmute',
uri:
'',
},
{
label: 'Keys',
uri:
'',
},
{
label: 'Defrag',
uri:
'',
},
{
label: 'MediaCD',
uri:
'',
},
{
label: 'Bookmark',
uri:
'',
},
{
label: 'Brush',
uri:
'',
},
{
label: 'Earth',
uri:
'',
},
{
label: 'Dial',
uri:
'',
},
{
label: 'TimeDate',
uri:
'',
},
];

const ToolbarExample = () => {
return (
<View style={{ backgroundColor: 'teal', flex: 1 }}>
<AppBar style={{ padding: 2 }}>
<AppBar.BackAction onPress={() => {}} />
<AppBar.Content title='Office' subtitle='tools' />
<View style={styles.wrapper}>
<Divider orientation='vertical' style={{ marginRight: 4 }} />
<Divider
variant='raised'
orientation='vertical'
style={{ height: '80%' }}
/>
<View style={styles.toolbarWrapper}>
<Toolbar>
{icons.map(icon => (
<TouchableOpacity
key={icon.label}
onPress={() =>
notificationService.send({
message: `You just pressed ${icon.label} icon!`,
closeButtonLabel: 'OK!',
})
}
>
<Image
source={{ uri: icon.uri }}
style={styles.item}
key={icon.label}
/>
</TouchableOpacity>
))}
</Toolbar>
</View>
</View>
</AppBar>
</View>
);
};

const styles = StyleSheet.create({
wrapper: {
flexDirection: 'row',
alignItems: 'center',
height: 40,
width: 240,
},
toolbarWrapper: {
flex: 1,
paddingHorizontal: 4,
},
item: {
marginHorizontal: 8,
width: 32,
height: 32,
},
});

export default ToolbarExample;
6 changes: 6 additions & 0 deletions example/src/examples/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ import SliderExample from './SliderExample';
import SnackbarExample from './SnackbarExample';
import TabsExample from './TabsExample';
import TextInputExample from './TextInputExample';
import ToolbarExample from './ToolbarExample';
import TypographyExample from './TypographyExample';
import WindowExample from './WindowExample';

Expand Down Expand Up @@ -97,6 +98,11 @@ export default [
component: SnackbarExample,
title: 'Snackbar',
},
{
name: 'ToolbarExample',
component: ToolbarExample,
title: 'Toolbar',
},
{ name: 'WindowExample', component: WindowExample, title: 'Window' },
],
},
Expand Down
2 changes: 1 addition & 1 deletion example/src/util/notifications.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -73,7 +73,7 @@ export const NotificationProvider = () => {
label: notification.closeButtonLabel || 'OK',
onPress: () => notificationService.remove(notification),
}}
duration={Snackbar.DURATION_SHORT}
duration={1000}
>
{notification.message}
</Snackbar>
Expand Down
113 changes: 113 additions & 0 deletions src/components/Toolbar/Toolbar.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,113 @@
import React from 'react';
import {
StyleSheet,
View,
ScrollView as RNScrollView,
ViewStyle,
StyleProp,
} from 'react-native';
import type {
LayoutChangeEvent,
NativeScrollEvent,
NativeSyntheticEvent,
} from 'react-native';

import { ChevronIcon } from '../..';

const chevronIconSize = 20;

type ScrollViewProps = React.ComponentProps<typeof View> & {
children: React.ReactNode;
scrollViewProps?: React.ComponentProps<typeof RNScrollView>;
style?: StyleProp<ViewStyle>;
};

// TODO: performance improvements (callbacks, refs ...etc)
const ScrollView = ({
children,
scrollViewProps = {},
style,
...rest
}: ScrollViewProps) => {
const [contentOffset, setContentOffset] = React.useState(0);
const [contentSize, setContentSize] = React.useState(0);
const [scrollViewSize, setScrollViewSize] = React.useState(0);

// TODO: that's a naive approach. if it causes problems
// trigger a callback when last child becomes fully visible
const lastElementVisible =
contentOffset + scrollViewSize < contentSize - chevronIconSize;

const handleScroll = (e: NativeSyntheticEvent<NativeScrollEvent>) => {
scrollViewProps.onScroll?.(e);
setContentOffset(e.nativeEvent.contentOffset.x);
};

const handleContentSizeChange = (width: number, height: number) => {
scrollViewProps.onContentSizeChange?.(width, height);
setContentSize(width);
};

const handleLayout = (e: LayoutChangeEvent) => {
scrollViewProps.onLayout?.(e);
setScrollViewSize(e.nativeEvent.layout.width);
};

return (
<View
style={[
styles.wrapper,
{
flexDirection: 'row',
},
style,
]}
{...rest}
>
<View style={[styles.content]}>
<RNScrollView
{...scrollViewProps}
showsHorizontalScrollIndicator={false}
scrollEventThrottle={100}
onScroll={handleScroll}
onContentSizeChange={handleContentSizeChange}
onLayout={handleLayout}
horizontal
>
{children}
</RNScrollView>
</View>
<View style={styles.chevronIcon}>
{lastElementVisible && (
<>
<ChevronIcon
segments={3}
direction='right'
style={{ marginRight: 1 }}
/>
<ChevronIcon segments={3} direction='right' />
</>
)}
</View>
</View>
);
};

const styles = StyleSheet.create({
wrapper: {
display: 'flex',
position: 'relative',
},
content: {
flexGrow: 1,
flexShrink: 1,
},
chevronIcon: {
justifyContent: 'center',
flexDirection: 'row',
alignSelf: 'flex-start',
width: chevronIconSize,
},
});

export default ScrollView;
1 change: 1 addition & 0 deletions src/components/Toolbar/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export { default } from './Toolbar';
1 change: 1 addition & 0 deletions src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ export { default as Slider } from './components/Slider';
export { default as Snackbar } from './components/Snackbar';
export { default as Tabs } from './components/Tabs';
export { default as TextInput } from './components/TextInput';
export { default as Toolbar } from './components/Toolbar';
export { default as Window } from './components/Window';
export { Select, SelectBox } from './components/Select';
export { Text, Title, Anchor } from './components/Typography';
Expand Down

0 comments on commit 7bdc5ff

Please sign in to comment.