mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-18 09:35:20 +02:00
86 lines
2.8 KiB
TypeScript
86 lines
2.8 KiB
TypeScript
import * as React from 'react';
|
|
|
|
import shim from '@joplin/lib/shim';
|
|
import { FunctionComponent, useCallback, useEffect, useState } from 'react';
|
|
import { ConfigScreenStyles } from './configScreenStyles';
|
|
import { View, Text } from 'react-native';
|
|
import Setting, { SettingItem } from '@joplin/lib/models/Setting';
|
|
import { openDocumentTree } from '@joplin/react-native-saf-x';
|
|
import { UpdateSettingValueCallback } from './types';
|
|
import { reg } from '@joplin/lib/registry';
|
|
import type FsDriverWeb from '../../../utils/fs-driver/fs-driver-rn.web';
|
|
import { TouchableRipple } from 'react-native-paper';
|
|
|
|
interface Props {
|
|
styles: ConfigScreenStyles;
|
|
settingMetadata: SettingItem;
|
|
mode: 'read'|'readwrite';
|
|
updateSettingValue: UpdateSettingValueCallback;
|
|
}
|
|
|
|
type ExtendedSelf = (typeof window.self) & {
|
|
showDirectoryPicker: (options: { id: string; mode: string })=> Promise<FileSystemDirectoryHandle>;
|
|
};
|
|
declare const self: ExtendedSelf;
|
|
|
|
const FileSystemPathSelector: FunctionComponent<Props> = props => {
|
|
const [fileSystemPath, setFileSystemPath] = useState<string>('');
|
|
|
|
const settingId = props.settingMetadata.key;
|
|
|
|
useEffect(() => {
|
|
setFileSystemPath(Setting.value(settingId));
|
|
}, [settingId]);
|
|
|
|
const selectDirectoryButtonPress = useCallback(async () => {
|
|
if (shim.mobilePlatform() === 'web') {
|
|
// Directory picker IDs can't include certain characters.
|
|
const pickerId = `setting-${settingId}`.replace(/[^a-zA-Z]/g, '_');
|
|
const handle = await self.showDirectoryPicker({ id: pickerId, mode: props.mode });
|
|
const fsDriver = shim.fsDriver() as FsDriverWeb;
|
|
const uri = await fsDriver.mountExternalDirectory(handle, pickerId, props.mode);
|
|
await props.updateSettingValue(settingId, uri);
|
|
setFileSystemPath(uri);
|
|
} else {
|
|
try {
|
|
const doc = await openDocumentTree(true);
|
|
if (doc?.uri) {
|
|
setFileSystemPath(doc.uri);
|
|
await props.updateSettingValue(settingId, doc.uri);
|
|
} else {
|
|
throw new Error('User cancelled operation');
|
|
}
|
|
} catch (e) {
|
|
reg.logger().info('Didn\'t pick sync dir: ', e);
|
|
}
|
|
}
|
|
}, [props.updateSettingValue, settingId, props.mode]);
|
|
|
|
// Supported on Android and some versions of Chrome
|
|
const supported = shim.fsDriver().isUsingAndroidSAF() || (shim.mobilePlatform() === 'web' && 'showDirectoryPicker' in self);
|
|
if (!supported) {
|
|
return null;
|
|
}
|
|
|
|
const styleSheet = props.styles.styleSheet;
|
|
|
|
return (
|
|
<TouchableRipple
|
|
onPress={selectDirectoryButtonPress}
|
|
style={styleSheet.settingContainer}
|
|
role='button'
|
|
>
|
|
<View style={styleSheet.settingContainer}>
|
|
<Text key="label" style={styleSheet.settingText}>
|
|
{props.settingMetadata.label()}
|
|
</Text>
|
|
<Text style={styleSheet.settingControl}>
|
|
{fileSystemPath}
|
|
</Text>
|
|
</View>
|
|
</TouchableRipple>
|
|
);
|
|
};
|
|
|
|
export default FileSystemPathSelector;
|