mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-05 12:50:29 +02:00
118 lines
3.4 KiB
TypeScript
118 lines
3.4 KiB
TypeScript
|
import * as React from 'react';
|
||
|
import { AppState } from '../../utils/types';
|
||
|
import { connect } from 'react-redux';
|
||
|
import { ScrollView, StyleSheet, View } from 'react-native';
|
||
|
import { ToolbarButtonInfo, ToolbarItem } from '@joplin/lib/services/commands/ToolbarButtonUtils';
|
||
|
import toolbarButtonsFromState from './utils/toolbarButtonsFromState';
|
||
|
import { useCallback, useMemo, useRef, useState } from 'react';
|
||
|
import { themeStyle } from '../global-style';
|
||
|
import ToggleSpaceButton from '../ToggleSpaceButton';
|
||
|
import ToolbarEditorDialog from './ToolbarEditorDialog';
|
||
|
import { EditorState } from './types';
|
||
|
import ToolbarButton from './ToolbarButton';
|
||
|
import isSelected from './utils/isSelected';
|
||
|
import { _ } from '@joplin/lib/locale';
|
||
|
|
||
|
interface Props {
|
||
|
themeId: number;
|
||
|
toolbarButtonInfos: ToolbarItem[];
|
||
|
editorState: EditorState;
|
||
|
}
|
||
|
|
||
|
const useStyles = (themeId: number) => {
|
||
|
return useMemo(() => {
|
||
|
const theme = themeStyle(themeId);
|
||
|
return StyleSheet.create({
|
||
|
content: {
|
||
|
flexGrow: 0,
|
||
|
backgroundColor: theme.backgroundColor3,
|
||
|
},
|
||
|
contentContainer: {
|
||
|
flexGrow: 1,
|
||
|
paddingVertical: 0,
|
||
|
flexDirection: 'row',
|
||
|
},
|
||
|
spacer: {
|
||
|
flexGrow: 1,
|
||
|
},
|
||
|
});
|
||
|
}, [themeId]);
|
||
|
};
|
||
|
|
||
|
type SetSettingsVisible = React.Dispatch<React.SetStateAction<boolean>>;
|
||
|
const useSettingButtonInfo = (setSettingsVisible: SetSettingsVisible) => {
|
||
|
return useMemo((): ToolbarButtonInfo => ({
|
||
|
type: 'button',
|
||
|
name: 'showToolbarSettings',
|
||
|
tooltip: _('Settings'),
|
||
|
iconName: 'material cogs',
|
||
|
enabled: true,
|
||
|
onClick: () => setSettingsVisible(true),
|
||
|
title: '',
|
||
|
}), [setSettingsVisible]);
|
||
|
};
|
||
|
|
||
|
const EditorToolbar: React.FC<Props> = props => {
|
||
|
const styles = useStyles(props.themeId);
|
||
|
|
||
|
const buttonInfos: ToolbarButtonInfo[] = [];
|
||
|
|
||
|
for (const info of props.toolbarButtonInfos) {
|
||
|
if (info.type !== 'separator') {
|
||
|
buttonInfos.push(info);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
const renderButton = (info: ToolbarButtonInfo) => {
|
||
|
return <ToolbarButton
|
||
|
key={`command-${info.name}`}
|
||
|
buttonInfo={info}
|
||
|
themeId={props.themeId}
|
||
|
selected={isSelected(info.name, props.editorState)}
|
||
|
/>;
|
||
|
};
|
||
|
|
||
|
const [settingsVisible, setSettingsVisible] = useState(false);
|
||
|
const scrollViewRef = useRef<ScrollView|null>(null);
|
||
|
const onDismissSettingsDialog = useCallback(() => {
|
||
|
setSettingsVisible(false);
|
||
|
|
||
|
// On Android, if the ScrollView isn't manually scrolled to the end,
|
||
|
// all items can be invisible in some cases. This causes issues with
|
||
|
// TalkBack on Android.
|
||
|
// In particular, if 1) the toolbar initially has many items on a device
|
||
|
// with a small screen, and 2) the user removes most items, then most/all
|
||
|
// items are scrolled offscreen. Calling .scrollToEnd corrects this:
|
||
|
scrollViewRef.current?.scrollToEnd();
|
||
|
}, []);
|
||
|
|
||
|
const settingsButtonInfo = useSettingButtonInfo(setSettingsVisible);
|
||
|
const settingsButton = <ToolbarButton
|
||
|
buttonInfo={settingsButtonInfo}
|
||
|
themeId={props.themeId}
|
||
|
/>;
|
||
|
|
||
|
return <>
|
||
|
<ToggleSpaceButton themeId={props.themeId}>
|
||
|
<ScrollView
|
||
|
ref={scrollViewRef}
|
||
|
horizontal={true}
|
||
|
style={styles.content}
|
||
|
contentContainerStyle={styles.contentContainer}
|
||
|
>
|
||
|
{buttonInfos.map(renderButton)}
|
||
|
<View style={styles.spacer}/>
|
||
|
{settingsButton}
|
||
|
</ScrollView>
|
||
|
</ToggleSpaceButton>
|
||
|
<ToolbarEditorDialog visible={settingsVisible} onDismiss={onDismissSettingsDialog} />
|
||
|
</>;
|
||
|
};
|
||
|
|
||
|
export default connect((state: AppState) => {
|
||
|
return {
|
||
|
themeId: state.settings.theme,
|
||
|
toolbarButtonInfos: toolbarButtonsFromState(state),
|
||
|
};
|
||
|
})(EditorToolbar);
|