2022-08-08 08:00:14 -07:00
|
|
|
// Displays a find/replace dialog
|
|
|
|
|
|
|
|
const React = require('react');
|
|
|
|
const { useMemo, useState, useEffect } = require('react');
|
|
|
|
|
2023-09-21 01:12:40 -07:00
|
|
|
import { EditorSettings } from './types';
|
2022-08-08 08:00:14 -07:00
|
|
|
import { _ } from '@joplin/lib/locale';
|
2022-08-21 14:03:41 -07:00
|
|
|
import { BackHandler, TextInput, View, Text, StyleSheet, ViewStyle } from 'react-native';
|
2022-08-08 08:00:14 -07:00
|
|
|
import { Theme } from '@joplin/lib/themes/type';
|
2024-05-25 06:41:27 -07:00
|
|
|
import IconButton from '../IconButton';
|
2023-09-21 01:12:40 -07:00
|
|
|
import { SearchState } from '@joplin/editor/types';
|
|
|
|
import { SearchControl } from './types';
|
2022-08-08 08:00:14 -07:00
|
|
|
|
|
|
|
const buttonSize = 48;
|
|
|
|
|
|
|
|
type OnChangeCallback = (text: string)=> void;
|
|
|
|
type Callback = ()=> void;
|
|
|
|
|
|
|
|
export const defaultSearchState: SearchState = {
|
|
|
|
useRegex: false,
|
|
|
|
caseSensitive: false,
|
|
|
|
|
|
|
|
searchText: '',
|
|
|
|
replaceText: '',
|
|
|
|
dialogVisible: false,
|
|
|
|
};
|
|
|
|
|
|
|
|
export interface SearchPanelProps {
|
2023-07-06 11:17:41 -07:00
|
|
|
searchControl: SearchControl;
|
|
|
|
searchState: SearchState;
|
|
|
|
editorSettings: EditorSettings;
|
2022-08-08 08:00:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
interface ActionButtonProps {
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2022-08-08 08:00:14 -07:00
|
|
|
styles: any;
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId: number;
|
2022-08-08 08:00:14 -07:00
|
|
|
iconName: string;
|
|
|
|
title: string;
|
|
|
|
onPress: Callback;
|
|
|
|
}
|
|
|
|
|
2023-11-26 12:37:45 +01:00
|
|
|
const ActionButton = (props: ActionButtonProps) => {
|
2022-08-08 08:00:14 -07:00
|
|
|
return (
|
2024-05-25 06:41:27 -07:00
|
|
|
<IconButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={props.themeId}
|
2024-05-25 06:41:27 -07:00
|
|
|
containerStyle={props.styles.button}
|
2022-08-08 08:00:14 -07:00
|
|
|
onPress={props.onPress}
|
2022-08-21 14:03:41 -07:00
|
|
|
description={props.title}
|
2024-05-25 06:41:27 -07:00
|
|
|
iconName={`material ${props.iconName}`}
|
|
|
|
iconStyle={props.styles.buttonText}
|
|
|
|
/>
|
2022-08-08 08:00:14 -07:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
interface ToggleButtonProps {
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2022-08-08 08:00:14 -07:00
|
|
|
styles: any;
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId: number;
|
2022-08-08 08:00:14 -07:00
|
|
|
iconName: string;
|
|
|
|
title: string;
|
|
|
|
active: boolean;
|
|
|
|
onToggle: Callback;
|
|
|
|
}
|
2022-08-21 14:03:41 -07:00
|
|
|
|
2022-08-08 08:00:14 -07:00
|
|
|
const ToggleButton = (props: ToggleButtonProps) => {
|
|
|
|
const active = props.active;
|
|
|
|
|
|
|
|
return (
|
2024-05-25 06:41:27 -07:00
|
|
|
<IconButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={props.themeId}
|
2024-05-25 06:41:27 -07:00
|
|
|
containerStyle={{
|
2022-08-08 08:00:14 -07:00
|
|
|
...props.styles.toggleButton,
|
|
|
|
...(active ? props.styles.toggleButtonActive : {}),
|
|
|
|
}}
|
|
|
|
onPress={props.onToggle}
|
|
|
|
|
|
|
|
accessibilityState={{
|
|
|
|
checked: props.active,
|
|
|
|
}}
|
2022-08-21 14:03:41 -07:00
|
|
|
description={props.title}
|
2022-08-08 08:00:14 -07:00
|
|
|
accessibilityRole='switch'
|
2024-05-25 06:41:27 -07:00
|
|
|
|
|
|
|
iconName={`material ${props.iconName}`}
|
|
|
|
iconStyle={
|
2022-08-08 08:00:14 -07:00
|
|
|
active ? props.styles.activeButtonText : props.styles.buttonText
|
2024-05-25 06:41:27 -07:00
|
|
|
}
|
|
|
|
/>
|
2022-08-08 08:00:14 -07:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const useStyles = (theme: Theme) => {
|
|
|
|
return useMemo(() => {
|
2022-08-21 14:03:41 -07:00
|
|
|
const buttonStyle: ViewStyle = {
|
2022-08-08 08:00:14 -07:00
|
|
|
width: buttonSize,
|
|
|
|
height: buttonSize,
|
|
|
|
backgroundColor: theme.backgroundColor4,
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'center',
|
|
|
|
flexShrink: 1,
|
|
|
|
};
|
|
|
|
const buttonTextStyle = {
|
|
|
|
color: theme.color4,
|
|
|
|
fontSize: 30,
|
|
|
|
};
|
|
|
|
|
|
|
|
return StyleSheet.create({
|
|
|
|
button: buttonStyle,
|
|
|
|
toggleButton: {
|
|
|
|
...buttonStyle,
|
|
|
|
},
|
|
|
|
toggleButtonActive: {
|
|
|
|
...buttonStyle,
|
|
|
|
backgroundColor: theme.backgroundColor3,
|
|
|
|
},
|
|
|
|
input: {
|
|
|
|
flexGrow: 1,
|
|
|
|
height: buttonSize,
|
|
|
|
backgroundColor: theme.backgroundColor4,
|
|
|
|
color: theme.color4,
|
|
|
|
},
|
|
|
|
buttonText: buttonTextStyle,
|
|
|
|
activeButtonText: {
|
|
|
|
...buttonTextStyle,
|
|
|
|
color: theme.color4,
|
|
|
|
},
|
|
|
|
text: {
|
|
|
|
color: theme.color,
|
|
|
|
},
|
|
|
|
labeledInput: {
|
|
|
|
flexGrow: 1,
|
|
|
|
flexDirection: 'row',
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'center',
|
|
|
|
marginLeft: 10,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}, [theme]);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const SearchPanel = (props: SearchPanelProps) => {
|
2022-08-21 14:03:41 -07:00
|
|
|
const theme = props.editorSettings.themeData;
|
|
|
|
const placeholderColor = theme.color3;
|
|
|
|
const styles = useStyles(theme);
|
2022-08-08 08:00:14 -07:00
|
|
|
|
|
|
|
const [showingAdvanced, setShowAdvanced] = useState(false);
|
|
|
|
|
|
|
|
const state = props.searchState;
|
|
|
|
const control = props.searchControl;
|
|
|
|
|
2024-04-05 12:16:49 +01:00
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
2022-08-08 08:00:14 -07:00
|
|
|
const updateSearchState = (changedData: any) => {
|
2023-06-01 12:02:36 +01:00
|
|
|
const newState = { ...state, ...changedData };
|
2022-08-08 08:00:14 -07:00
|
|
|
control.setSearchState(newState);
|
|
|
|
};
|
|
|
|
|
2024-02-26 10:16:23 +00:00
|
|
|
// Creates a TextInput with the given parameters
|
2022-08-08 08:00:14 -07:00
|
|
|
const createInput = (
|
2023-08-22 11:58:53 +01:00
|
|
|
placeholder: string, value: string, onChange: OnChangeCallback, autoFocus: boolean,
|
2022-08-08 08:00:14 -07:00
|
|
|
) => {
|
|
|
|
return (
|
|
|
|
<TextInput
|
|
|
|
style={styles.input}
|
|
|
|
autoFocus={autoFocus}
|
|
|
|
onChangeText={onChange}
|
|
|
|
value={value}
|
|
|
|
placeholder={placeholder}
|
|
|
|
placeholderTextColor={placeholderColor}
|
|
|
|
returnKeyType='search'
|
|
|
|
blurOnSubmit={false}
|
|
|
|
onSubmitEditing={control.findNext}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Close the search dialog on back button press
|
|
|
|
useEffect(() => {
|
|
|
|
// Only register the listener if the dialog is visible
|
|
|
|
if (!state.dialogVisible) {
|
|
|
|
return () => {};
|
|
|
|
}
|
|
|
|
|
|
|
|
const backListener = BackHandler.addEventListener('hardwareBackPress', () => {
|
|
|
|
control.hideSearch();
|
|
|
|
return true;
|
|
|
|
});
|
|
|
|
|
|
|
|
return () => backListener.remove();
|
2022-08-19 12:10:04 +01:00
|
|
|
// eslint-disable-next-line @seiyab/react-hooks/exhaustive-deps -- Old code before rule was applied
|
2022-08-08 08:00:14 -07:00
|
|
|
}, [state.dialogVisible]);
|
|
|
|
|
|
|
|
|
2022-08-21 14:03:41 -07:00
|
|
|
const themeId = props.editorSettings.themeId;
|
2022-08-08 08:00:14 -07:00
|
|
|
const closeButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="close"
|
|
|
|
onPress={control.hideSearch}
|
2022-10-30 18:37:58 +00:00
|
|
|
title={_('Close')}
|
2022-08-08 08:00:14 -07:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const showDetailsButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="menu-down"
|
|
|
|
onPress={() => setShowAdvanced(true)}
|
|
|
|
title={_('Show advanced')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const hideDetailsButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="menu-up"
|
|
|
|
onPress={() => setShowAdvanced(false)}
|
|
|
|
title={_('Hide advanced')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const searchTextInput = createInput(
|
|
|
|
_('Search for...'),
|
|
|
|
state.searchText,
|
|
|
|
(newText: string) => {
|
|
|
|
updateSearchState({
|
|
|
|
searchText: newText,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
// Autofocus
|
2023-08-22 11:58:53 +01:00
|
|
|
true,
|
2022-08-08 08:00:14 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
const replaceTextInput = createInput(
|
|
|
|
_('Replace with...'),
|
|
|
|
state.replaceText,
|
|
|
|
(newText: string) => {
|
|
|
|
updateSearchState({
|
|
|
|
replaceText: newText,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
// Don't autofocus
|
2023-08-22 11:58:53 +01:00
|
|
|
false,
|
2022-08-08 08:00:14 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
const labeledSearchInput = (
|
|
|
|
<View style={styles.labeledInput} accessible>
|
|
|
|
<Text style={styles.text}>{_('Find: ')}</Text>
|
|
|
|
{ searchTextInput }
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
|
|
|
|
const labeledReplaceInput = (
|
|
|
|
<View style={styles.labeledInput} accessible>
|
|
|
|
<Text style={styles.text}>{_('Replace: ')}</Text>
|
|
|
|
{ replaceTextInput }
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
|
|
|
|
const toNextButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="menu-right"
|
|
|
|
onPress={control.findNext}
|
|
|
|
title={_('Next match')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const toPrevButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="menu-left"
|
|
|
|
onPress={control.findPrevious}
|
|
|
|
title={_('Previous match')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const replaceButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="swap-horizontal"
|
2023-09-21 01:12:40 -07:00
|
|
|
onPress={control.replaceNext}
|
2022-08-08 08:00:14 -07:00
|
|
|
title={_('Replace')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const replaceAllButton = (
|
|
|
|
<ActionButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="reply-all"
|
|
|
|
onPress={control.replaceAll}
|
|
|
|
title={_('Replace all')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const regexpButton = (
|
|
|
|
<ToggleButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="regex"
|
|
|
|
onToggle={() => {
|
|
|
|
updateSearchState({
|
|
|
|
useRegex: !state.useRegex,
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
active={state.useRegex}
|
|
|
|
title={_('Regular expression')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const caseSensitiveButton = (
|
|
|
|
<ToggleButton
|
2022-08-21 14:03:41 -07:00
|
|
|
themeId={themeId}
|
2022-08-08 08:00:14 -07:00
|
|
|
styles={styles}
|
|
|
|
iconName="format-letter-case"
|
|
|
|
onToggle={() => {
|
|
|
|
updateSearchState({
|
|
|
|
caseSensitive: !state.caseSensitive,
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
active={state.caseSensitive}
|
|
|
|
title={_('Case sensitive')}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
const simpleLayout = (
|
|
|
|
<View style={{ flexDirection: 'row' }}>
|
|
|
|
{ closeButton }
|
|
|
|
{ searchTextInput }
|
|
|
|
{ showDetailsButton }
|
|
|
|
{ toPrevButton }
|
|
|
|
{ toNextButton }
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
|
|
|
|
const advancedLayout = (
|
|
|
|
<View style={{ flexDirection: 'column', alignItems: 'center' }}>
|
|
|
|
<View style={{ flexDirection: 'row' }}>
|
|
|
|
{ closeButton }
|
|
|
|
{ labeledSearchInput }
|
|
|
|
{ hideDetailsButton }
|
|
|
|
{ toPrevButton }
|
|
|
|
{ toNextButton }
|
|
|
|
</View>
|
|
|
|
<View style={{ flexDirection: 'row' }}>
|
|
|
|
{ regexpButton }
|
|
|
|
{ caseSensitiveButton }
|
|
|
|
{ labeledReplaceInput }
|
|
|
|
{ replaceButton }
|
|
|
|
{ replaceAllButton }
|
|
|
|
</View>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!state.dialogVisible) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return showingAdvanced ? advancedLayout : simpleLayout;
|
|
|
|
};
|
|
|
|
|
|
|
|
export default SearchPanel;
|