1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-30 10:36:35 +02:00
joplin/packages/app-mobile/components/NotesBar.tsx

294 lines
7.2 KiB
TypeScript
Raw Normal View History

2022-09-09 16:06:03 +02:00
import * as React from 'react';
import { View, Text, TextInput, TouchableOpacity, FlatList, StyleSheet } from 'react-native';
import { State } from '@joplin/lib/reducer';
import { themeStyle } from './global-style';
const { connect } = require('react-redux');
const Icon = require('react-native-vector-icons/Ionicons').default;
const { _ } = require('@joplin/lib/locale');
import { Style } from './global-style';
import Note from '@joplin/lib/models/Note';
import NotesBarListItem from './NotesBarListItem';
import Folder from '@joplin/lib/models/Folder';
import searchNotes from './searchNotes';
interface Props {
themeId: number;
notes: any[];
todoCheckbox_change: (checked: boolean)=> void;
selectedFolderId: string;
activeFolderId: string;
dispatch: any;
selectedNoteId: string;
settings: any;
}
function NotesBarComponent(props: Props) {
const [query, setQuery] = React.useState<string>('');
const [notes, setNotes] = React.useState<any[]>(props.notes);
const theme: Style = React.useMemo(() => themeStyle(props.themeId), [props.themeId]);
const styles = (): Style => {
const styles: Style = {
container: {
flex: 1,
width: '100%',
backgroundColor: theme.backgroundColor3,
},
horizontalFlex: {
flexDirection: 'row',
},
title: {
alignItems: 'center',
},
titleText: {
fontSize: 16,
},
closeIcon: {
fontSize: 30,
paddingTop: 8,
paddingBottom: 8,
paddingRight: theme.marginRight,
paddingLeft: theme.marginLeft,
},
top: {
color: theme.color,
},
topContainer: {
width: '100%',
justifyContent: 'space-between',
paddingLeft: theme.marginLeft,
},
padding: {
paddingLeft: theme.marginLeft,
paddingRight: theme.marginRight,
paddingTop: 12,
paddingBottom: 12,
},
titleIcon: {
fontSize: 22,
marginRight: 4,
},
divider: {
backgroundColor: theme.dividerColor,
height: 1,
width: '100%',
},
nativeInput: {
fontSize: theme.fontSize,
flex: 1,
paddingRight: 8,
},
searchIcon: {
fontSize: 22,
},
searchInput: {
alignItems: 'center',
backgroundColor: theme.backgroundColor,
paddingLeft: 8,
borderRadius: 4,
borderWidth: 1,
borderColor: theme.dividerColor,
height: 42,
flex: 1,
},
button: {
height: 42,
width: 42,
backgroundColor: theme.color4,
alignItems: 'center',
justifyContent: 'center',
borderRadius: 4,
flex: 0.5,
marginLeft: 8,
},
buttonIcon: {
color: theme.backgroundColor,
fontSize: 22,
},
inputGroup: {
justifyContent: 'space-between',
},
};
return StyleSheet.create(styles);
};
const titleComp = (
<View style={[styles().title, styles().horizontalFlex]}>
<Icon name='md-document'style={[styles().top, styles().titleIcon]} />
<Text style={[styles().top, styles().titleText]}>{_('Notes')}</Text>
</View>
);
const dividerComp = (
<View style={styles().divider}></View>
);
const handleNotesBarClose = () => {
props.dispatch({ type: 'NOTES_BAR_CLOSE' });
};
const closeButtonComp = (
<TouchableOpacity
onPress={handleNotesBarClose}
accessibilityLabel={_('Toggle note list')}
accessibilityRole="button"
>
<Icon name="close" style={[styles().top, styles().closeIcon]}/>
</TouchableOpacity>
);
const renderIconButton = (icon: JSX.Element, onPress: ()=> Promise<void>, label: string) => {
return (
<TouchableOpacity
style={styles().button}
activeOpacity={0.8}
onPress={onPress}
accessibilityLabel={label}
accessibilityRole="button"
>
{icon}
</TouchableOpacity>
);
};
const handleNewNote = async (isTodo: boolean) => {
let folderId = props.selectedFolderId !== Folder.conflictFolderId() ? props.selectedFolderId : null;
if (!folderId) folderId = props.activeFolderId;
props.dispatch({
type: 'NAV_BACK',
});
const newNote = await Note.save({
parent_id: folderId,
is_todo: isTodo ? 1 : 0,
}, { provisional: true });
props.dispatch({
type: 'NAV_GO',
routeName: 'Note',
noteId: newNote.id,
});
};
const addNoteButtonComp = renderIconButton(<Icon name='document-text-outline' style={styles().buttonIcon} />, () => handleNewNote(false), _('New note'));
const addTodoButtonComp = renderIconButton(<Icon name='checkbox-outline' style={styles().buttonIcon} />, () => handleNewNote(true), _('New to-do'));
const topComp = (
<View>
<View style={[styles().topContainer, styles().horizontalFlex]}>
{titleComp}
{closeButtonComp}
</View>
{dividerComp}
</View>
);
const refreshSearch = async () => {
const notes = await searchNotes(query, props.settings['db.ftsEnabled'], props.dispatch);
setNotes(notes);
};
const handleQuerySubmit = async () => {
const trimmedQuery = query.trim();
if (!trimmedQuery) {
props.dispatch({
type: 'SEARCH_QUERY',
query: '',
});
} else {
props.dispatch({
type: 'SEARCH_QUERY',
query: trimmedQuery,
});
}
setQuery(trimmedQuery);
await refreshSearch();
};
const searchInputComp = (
<View style={[styles().horizontalFlex, styles().searchInput]}>
<Icon name='search' style={[styles().top, styles().searchIcon]}/>
<TextInput style={[styles().top, styles().nativeInput]} placeholder='Search' onChangeText={setQuery} value={query} onSubmitEditing={handleQuerySubmit} placeholderTextColor={theme.dividerColor} />
</View>
);
const inputGroupComp = (
<View style={{ width: '100%' }}>
<View style={[styles().padding, styles().horizontalFlex, styles().inputGroup]}>
{searchInputComp}
{addNoteButtonComp}
{addTodoButtonComp}
</View>
{dividerComp}
</View>
);
let flatListRef: any = React.useRef(null);
const onRenderItem = React.useCallback(({ item }: { item: any }) => {
if (item.is_todo) {
return <NotesBarListItem note={item} todoCheckbox_change={props.todoCheckbox_change} />;
} else {
return <NotesBarListItem note={item} />;
}
}, [props.todoCheckbox_change]);
const NotesBarListComp = (
<FlatList
data={notes}
renderItem={onRenderItem}
keyExtractor={(item: any) => item.id}
getItemLayout={(data, index) => (
{
length: data.length,
offset: (theme.fontSize + styles().padding.paddingTop + styles().padding.paddingBottom) * index,
viewOffset: (theme.fontSize + styles().padding.paddingTop + styles().padding.paddingBottom),
index,
}
)}
ref={(ref: any) => { flatListRef = ref; }}
/>
);
// Scroll the notesbar to selected note item after rendering
React.useEffect(() => {
const selectedItemIndex = notes.findIndex(item => item.id === props.selectedNoteId);
if (selectedItemIndex >= 0) {
flatListRef.scrollToIndex({ index: selectedItemIndex });
}
});
// Update the notesbar when a note item changes
React.useEffect(() => {
setNotes(props.notes);
}, [props.notes]);
return (
<View style={styles().container}>
{topComp}
{inputGroupComp}
{ NotesBarListComp }
</View>
);
}
const NotesBar = connect((state: State) => {
return {
themeId: state.settings.theme,
notes: state.notes,
activeFolderId: state.settings.activeFolderId,
selectedFolderId: state.selectedFolderId,
selectedNoteId: state.selectedNoteIds[0],
settings: state.settings,
};
})(NotesBarComponent);
export default NotesBar;