1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-12 08:54:00 +02:00
joplin/ReactNativeClient/lib/components/side-menu-content.js

254 lines
7.6 KiB
JavaScript
Raw Normal View History

2017-07-22 17:55:09 +02:00
import React, { Component } from 'react';
import { TouchableOpacity , Button, Text, Image, StyleSheet, ScrollView, View } from 'react-native';
2017-05-24 21:27:13 +02:00
import { connect } from 'react-redux'
2017-07-22 17:55:09 +02:00
import Icon from 'react-native-vector-icons/Ionicons';
2017-06-24 20:06:28 +02:00
import { Log } from 'lib/log.js';
2017-07-25 20:36:52 +02:00
import { Tag } from 'lib/models/tag.js';
2017-06-24 20:06:28 +02:00
import { Note } from 'lib/models/note.js';
2017-07-24 23:52:30 +02:00
import { Setting } from 'lib/models/setting.js';
2017-07-15 18:14:15 +02:00
import { FoldersScreenUtils } from 'lib/components/screens/folders-utils.js'
import { Synchronizer } from 'lib/synchronizer.js';
2017-07-06 21:48:17 +02:00
import { reg } from 'lib/registry.js';
import { _ } from 'lib/locale.js';
2017-08-01 19:59:01 +02:00
import { globalStyle, themeStyle } from 'lib/components/global-style.js';
2017-05-24 21:27:13 +02:00
class SideMenuContentComponent extends Component {
2017-07-06 21:48:17 +02:00
constructor() {
super();
2017-07-25 20:36:52 +02:00
this.state = { syncReportText: '',
//width: 0,
};
2017-08-01 19:59:01 +02:00
this.styles_ = {};
}
styles() {
const theme = themeStyle(this.props.theme);
if (this.styles_[this.props.theme]) return this.styles_[this.props.theme];
this.styles_ = {};
let styles = {
menu: {
flex: 1,
backgroundColor: theme.backgroundColor,
borderTopWidth: 1,
borderTopColor: theme.dividerColor,
},
button: {
flex: 1,
flexDirection: 'row',
height: 36,
alignItems: 'center',
paddingLeft: theme.marginLeft,
paddingRight: theme.marginRight,
},
buttonText: {
flex: 1,
color: theme.color,
paddingLeft: 10,
fontSize: theme.fontSize,
},
syncStatus: {
paddingLeft: theme.marginLeft,
paddingRight: theme.marginRight,
color: theme.colorFaded,
fontSize: theme.fontSizeSmaller,
},
tagItemList: {
flex: 1,
flexDirection: 'row',
flexWrap: 'wrap'
},
};
styles.folderButton = Object.assign({}, styles.button);
styles.folderButtonText = Object.assign({}, styles.buttonText);
styles.folderButtonSelected = Object.assign({}, styles.folderButton);
styles.folderButtonSelected.backgroundColor = theme.selectedColor;
styles.folderIcon = Object.assign({}, theme.icon);
styles.folderIcon.color = '#0072d5';
styles.tagButton = Object.assign({}, styles.button);
styles.tagButtonSelected = Object.assign({}, styles.tagButton);
styles.tagButtonSelected.backgroundColor = theme.selectedColor;
styles.tagButtonSelected.borderRadius = 1000;
styles.tagButtonText = Object.assign({}, styles.buttonText);
styles.tagButtonText.flex = 0;
styles.syncButton = Object.assign({}, styles.button);
styles.syncButtonText = Object.assign({}, styles.buttonText);
this.styles_[this.props.theme] = StyleSheet.create(styles);
return this.styles_[this.props.theme];
2017-07-06 21:48:17 +02:00
}
2017-05-24 21:27:13 +02:00
folder_press(folder) {
2017-07-08 00:25:03 +02:00
this.props.dispatch({ type: 'SIDE_MENU_CLOSE' });
2017-05-24 22:11:37 +02:00
2017-07-25 20:09:01 +02:00
this.props.dispatch({
type: 'NAV_GO',
2017-07-25 20:09:01 +02:00
routeName: 'Notes',
folderId: folder.id,
});
2017-05-24 21:27:13 +02:00
}
2017-07-25 20:36:52 +02:00
tag_press(tag) {
this.props.dispatch({ type: 'SIDE_MENU_CLOSE' });
this.props.dispatch({
type: 'NAV_GO',
routeName: 'Notes',
tagId: tag.id,
});
}
2017-07-06 21:48:17 +02:00
async synchronize_press() {
2017-07-31 20:32:51 +02:00
const action = this.props.syncStarted ? 'cancel' : 'start';
2017-07-24 23:52:30 +02:00
if (Setting.value('sync.target') == Setting.SYNC_TARGET_ONEDRIVE && !reg.oneDriveApi().auth()) {
this.props.dispatch({ type: 'SIDE_MENU_CLOSE' });
this.props.dispatch({
type: 'NAV_GO',
2017-07-24 23:52:30 +02:00
routeName: 'OneDriveLogin',
});
return;
}
2017-07-30 22:22:57 +02:00
let sync = null;
try {
sync = await reg.synchronizer(Setting.value('sync.target'))
} catch (error) {
reg.logger().info('Could not acquire synchroniser:');
reg.logger().info(error);
return;
}
2017-07-31 20:32:51 +02:00
if (action == 'cancel') {
sync.cancel();
2017-07-06 21:48:17 +02:00
} else {
2017-07-24 23:52:30 +02:00
reg.scheduleSync(0);
2017-07-06 21:48:17 +02:00
}
}
2017-07-22 17:55:09 +02:00
folderItem(folder, selected) {
2017-08-01 19:59:01 +02:00
const iconComp = selected ? <Icon name='md-folder-open' style={this.styles().folderIcon} /> : <Icon name='md-folder' style={this.styles().folderIcon} />;
const folderButtonStyle = selected ? this.styles().folderButtonSelected : this.styles().folderButton;
2017-07-22 17:55:09 +02:00
return (
<TouchableOpacity key={folder.id} onPress={() => { this.folder_press(folder) }}>
<View style={folderButtonStyle}>
{ iconComp }
2017-08-01 19:59:01 +02:00
<Text numberOfLines={1} style={this.styles().folderButtonText}>{folder.title}</Text>
2017-07-22 17:55:09 +02:00
</View>
</TouchableOpacity>
);
}
2017-07-25 20:36:52 +02:00
tagItem(tag, selected) {
2017-08-01 19:59:01 +02:00
const iconComp = <Icon name='md-pricetag' style={this.styles().folderIcon} />
const tagButtonStyle = selected ? this.styles().tagButtonSelected : this.styles().tagButton;
2017-07-25 20:36:52 +02:00
return (
<TouchableOpacity key={tag.id} onPress={() => { this.tag_press(tag) }}>
<View style={tagButtonStyle}>
{ iconComp }
2017-08-01 19:59:01 +02:00
<Text numberOfLines={1} style={this.styles().tagButtonText}>{tag.title}</Text>
2017-07-25 20:36:52 +02:00
</View>
</TouchableOpacity>
);
}
2017-07-22 17:55:09 +02:00
synchronizeButton(state) {
2017-07-26 23:27:03 +02:00
const title = state == 'sync' ? _('Synchronise') : _('Cancel synchronisation');
2017-07-22 18:36:55 +02:00
const iconComp = state == 'sync' ? <Icon name='md-sync' style={globalStyle.icon} /> : <Icon name='md-close' style={globalStyle.icon} />;
2017-07-22 17:55:09 +02:00
return (
<TouchableOpacity key={'synchronize_button'} onPress={() => { this.synchronize_press() }}>
2017-08-01 19:59:01 +02:00
<View style={this.styles().syncButton}>
2017-07-22 17:55:09 +02:00
{ iconComp }
2017-08-01 19:59:01 +02:00
<Text style={this.styles().syncButtonText}>{title}</Text>
2017-07-22 17:55:09 +02:00
</View>
</TouchableOpacity>
);
}
2017-07-25 20:36:52 +02:00
makeDivider(key) {
return <View style={{ marginTop: 15, marginBottom: 15, flex: -1, borderBottomWidth: 1, borderBottomColor: globalStyle.dividerColor }} key={key}></View>
}
2017-05-24 21:27:13 +02:00
render() {
2017-07-06 21:48:17 +02:00
let items = [];
2017-07-22 17:55:09 +02:00
// HACK: inner height of ScrollView doesn't appear to be calculated correctly when
// using padding. So instead creating blank elements for padding bottom and top.
items.push(<View style={{ height: globalStyle.marginTop }} key='bottom_top_hack'/>);
2017-07-28 19:57:01 +02:00
if (this.props.folders.length) {
for (let i = 0; i < this.props.folders.length; i++) {
let folder = this.props.folders[i];
items.push(this.folderItem(folder, this.props.selectedFolderId == folder.id && this.props.notesParentType == 'Folder'));
}
2017-07-25 20:36:52 +02:00
2017-07-28 19:57:01 +02:00
if (items.length) items.push(this.makeDivider('divider_1'));
2017-05-24 21:27:13 +02:00
}
2017-07-28 19:57:01 +02:00
if (this.props.tags.length) {
2017-08-22 19:57:35 +02:00
let tags = this.props.tags.slice();
tags.sort((a, b) => { return a.title < b.title ? -1 : +1; });
2017-07-28 19:57:01 +02:00
let tagItems = [];
2017-08-22 19:57:35 +02:00
for (let i = 0; i < tags.length; i++) {
const tag = tags[i];
2017-07-28 19:57:01 +02:00
tagItems.push(this.tagItem(tag, this.props.selectedTagId == tag.id && this.props.notesParentType == 'Tag'));
}
2017-07-25 20:36:52 +02:00
2017-07-28 19:57:01 +02:00
items.push(
2017-08-01 19:59:01 +02:00
<View style={this.styles().tagItemList} key="tag_items">
2017-07-28 19:57:01 +02:00
{tagItems}
</View>
);
if (tagItems.length) items.push(this.makeDivider('divider_2'));
}
2017-07-06 21:48:17 +02:00
let lines = Synchronizer.reportToLines(this.props.syncReport);
while (lines.length < 10) lines.push(''); // Add blank lines so that height of report text is fixed and doesn't affect scrolling
const syncReportText = lines.join("\n");
2017-07-22 17:55:09 +02:00
items.push(this.synchronizeButton(this.props.syncStarted ? 'cancel' : 'sync'));
2017-08-01 19:59:01 +02:00
items.push(<Text key='sync_report' style={this.styles().syncStatus}>{syncReportText}</Text>);
2017-07-06 21:48:17 +02:00
2017-07-22 17:55:09 +02:00
items.push(<View style={{ height: globalStyle.marginBottom }} key='bottom_padding_hack'/>);
2017-07-06 21:48:17 +02:00
2017-05-24 21:27:13 +02:00
return (
2017-07-25 22:24:30 +02:00
<View style={{flex:1, borderRightWidth: 1, borderRightColor: globalStyle.dividerColor }}>
2017-07-22 17:55:09 +02:00
<View style={{flexDirection:'row'}}>
2017-07-26 23:27:03 +02:00
<Image style={{flex:1, height: 100}} source={require('../images/SideMenuHeader.png')} />
2017-07-22 17:55:09 +02:00
</View>
2017-08-01 19:59:01 +02:00
<ScrollView scrollsToTop={false} style={this.styles().menu}>
2017-07-22 17:55:09 +02:00
{ items }
</ScrollView>
</View>
2017-05-24 21:27:13 +02:00
);
}
};
const SideMenuContent = connect(
(state) => {
return {
folders: state.folders,
2017-07-25 20:36:52 +02:00
tags: state.tags,
syncStarted: state.syncStarted,
syncReport: state.syncReport,
2017-07-22 17:55:09 +02:00
selectedFolderId: state.selectedFolderId,
2017-07-25 20:36:52 +02:00
selectedTagId: state.selectedTagId,
notesParentType: state.notesParentType,
locale: state.settings.locale,
2017-08-01 19:59:01 +02:00
theme: state.settings.theme,
2017-05-24 21:27:13 +02:00
};
}
)(SideMenuContentComponent)
export { SideMenuContent };