Summary: Added button to reset Android SDK path to default + fixed default SDK path on Windows which I introduced in a previous diff and which before might be wrongly resolved to roaming data instead of local data folder Reviewed By: jknoxville Differential Revision: D19498899 fbshipit-source-id: e10865b528f55ea6b56873bdf9369c4b60dca89f
133 lines
3.4 KiB
TypeScript
133 lines
3.4 KiB
TypeScript
/**
|
|
* Copyright (c) Facebook, Inc. and its affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @format
|
|
*/
|
|
|
|
import electron from 'electron';
|
|
import {FlexColumn, styled, Text, FlexRow, Input, colors, Glyph} from 'flipper';
|
|
import React, {useState} from 'react';
|
|
import {promises as fs} from 'fs';
|
|
import {remote} from 'electron';
|
|
import path from 'path';
|
|
|
|
const ConfigFieldContainer = styled(FlexRow)({
|
|
paddingLeft: 10,
|
|
paddingRight: 10,
|
|
});
|
|
|
|
const InfoText = styled(Text)({
|
|
lineHeight: 1.35,
|
|
paddingTop: 5,
|
|
});
|
|
|
|
const FileInputBox = styled(Input)<{isValid: boolean}>(({isValid}) => ({
|
|
marginRight: 0,
|
|
flexGrow: 1,
|
|
fontFamily: 'monospace',
|
|
color: isValid ? undefined : colors.red,
|
|
marginLeft: 10,
|
|
marginTop: 'auto',
|
|
marginBottom: 'auto',
|
|
}));
|
|
|
|
const CenteredGlyph = styled(Glyph)({
|
|
margin: 'auto',
|
|
marginLeft: 10,
|
|
});
|
|
|
|
const GreyedOutOverlay = styled.div({
|
|
backgroundColor: '#EFEEEF',
|
|
borderRadius: 4,
|
|
opacity: 0.6,
|
|
height: '100%',
|
|
position: 'absolute',
|
|
left: 0,
|
|
right: 0,
|
|
});
|
|
|
|
export function FilePathConfigField(props: {
|
|
label: string;
|
|
resetValue?: string;
|
|
defaultValue: string;
|
|
onChange: (path: string) => void;
|
|
frozen?: boolean;
|
|
}) {
|
|
const [value, setValue] = useState(props.defaultValue);
|
|
const [isValid, setIsValid] = useState(true);
|
|
fs.stat(value)
|
|
.then(stat => stat.isDirectory())
|
|
.then(valid => {
|
|
if (valid !== isValid) {
|
|
setIsValid(valid);
|
|
}
|
|
})
|
|
.catch(_ => setIsValid(false));
|
|
|
|
return (
|
|
<ConfigFieldContainer>
|
|
<InfoText>{props.label}</InfoText>
|
|
<FileInputBox
|
|
placeholder={props.label}
|
|
value={value}
|
|
isValid={isValid}
|
|
onChange={e => {
|
|
setValue(e.target.value);
|
|
props.onChange(e.target.value);
|
|
fs.stat(e.target.value)
|
|
.then(stat => stat.isDirectory())
|
|
.then(valid => {
|
|
if (valid !== isValid) {
|
|
setIsValid(valid);
|
|
}
|
|
})
|
|
.catch(_ => setIsValid(false));
|
|
}}
|
|
/>
|
|
<FlexColumn
|
|
onClick={() =>
|
|
remote.dialog
|
|
.showOpenDialog({
|
|
properties: ['openDirectory', 'showHiddenFiles'],
|
|
defaultPath: path.resolve('/'),
|
|
})
|
|
.then((result: electron.SaveDialogReturnValue) => {
|
|
if (result.filePath) {
|
|
const path: string = result.filePath.toString();
|
|
setValue(path);
|
|
props.onChange(path);
|
|
}
|
|
})
|
|
}>
|
|
<CenteredGlyph name="dots-3-circle" variant="outline" />
|
|
</FlexColumn>
|
|
{props.resetValue && (
|
|
<FlexColumn
|
|
title={`Reset to default path ${props.resetValue}`}
|
|
onClick={() => {
|
|
setValue(props.resetValue!);
|
|
props.onChange(props.resetValue!);
|
|
}}>
|
|
<CenteredGlyph name="undo" variant="outline" />
|
|
</FlexColumn>
|
|
)}
|
|
{isValid ? null : (
|
|
<CenteredGlyph name="caution-triangle" color={colors.yellow} />
|
|
)}
|
|
{props.frozen && <GreyedOutOverlay />}
|
|
</ConfigFieldContainer>
|
|
);
|
|
}
|
|
|
|
export function ConfigText(props: {content: string; frozen?: boolean}) {
|
|
return (
|
|
<ConfigFieldContainer>
|
|
<InfoText>{props.content}</InfoText>
|
|
{props.frozen && <GreyedOutOverlay />}
|
|
</ConfigFieldContainer>
|
|
);
|
|
}
|