import React, { Component } from 'react'; import { TouchableOpacity , Button, Text, Image, StyleSheet, ScrollView, View } from 'react-native'; import { connect } from 'react-redux' import Icon from 'react-native-vector-icons/Ionicons'; import { Log } from 'lib/log.js'; import { Tag } from 'lib/models/tag.js'; import { Note } from 'lib/models/note.js'; import { Setting } from 'lib/models/setting.js'; import { FoldersScreenUtils } from 'lib/components/screens/folders-utils.js' import { Synchronizer } from 'lib/synchronizer.js'; import { reg } from 'lib/registry.js'; import { _ } from 'lib/locale.js'; import { globalStyle, themeStyle } from 'lib/components/global-style.js'; class SideMenuContentComponent extends Component { constructor() { super(); this.state = { syncReportText: '', //width: 0, }; 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]; } folder_press(folder) { this.props.dispatch({ type: 'SIDE_MENU_CLOSE' }); this.props.dispatch({ type: 'NAV_GO', routeName: 'Notes', folderId: folder.id, }); } tag_press(tag) { this.props.dispatch({ type: 'SIDE_MENU_CLOSE' }); this.props.dispatch({ type: 'NAV_GO', routeName: 'Notes', tagId: tag.id, }); } async synchronize_press() { const action = this.props.syncStarted ? 'cancel' : 'start'; 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', routeName: 'OneDriveLogin', }); return; } 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; } if (action == 'cancel') { sync.cancel(); } else { reg.scheduleSync(0); } } folderItem(folder, selected) { const iconComp = selected ? : ; const folderButtonStyle = selected ? this.styles().folderButtonSelected : this.styles().folderButton; return ( { this.folder_press(folder) }}> { iconComp } {folder.title} ); } tagItem(tag, selected) { const iconComp = const tagButtonStyle = selected ? this.styles().tagButtonSelected : this.styles().tagButton; return ( { this.tag_press(tag) }}> { iconComp } {tag.title} ); } synchronizeButton(state) { const title = state == 'sync' ? _('Synchronise') : _('Cancel synchronisation'); const iconComp = state == 'sync' ? : ; return ( { this.synchronize_press() }}> { iconComp } {title} ); } makeDivider(key) { return } render() { let items = []; // 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(); 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')); } if (items.length) items.push(this.makeDivider('divider_1')); } if (this.props.tags.length) { let tags = this.props.tags.slice(); tags.sort((a, b) => { return a.title < b.title ? -1 : +1; }); let tagItems = []; for (let i = 0; i < tags.length; i++) { const tag = tags[i]; tagItems.push(this.tagItem(tag, this.props.selectedTagId == tag.id && this.props.notesParentType == 'Tag')); } items.push( {tagItems} ); if (tagItems.length) items.push(this.makeDivider('divider_2')); } 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"); items.push(this.synchronizeButton(this.props.syncStarted ? 'cancel' : 'sync')); items.push({syncReportText}); items.push(); return ( { items } ); } }; const SideMenuContent = connect( (state) => { return { folders: state.folders, tags: state.tags, syncStarted: state.syncStarted, syncReport: state.syncReport, selectedFolderId: state.selectedFolderId, selectedTagId: state.selectedTagId, notesParentType: state.notesParentType, locale: state.settings.locale, theme: state.settings.theme, }; } )(SideMenuContentComponent) export { SideMenuContent };