mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-30 10:36:35 +02:00
634 lines
18 KiB
JavaScript
634 lines
18 KiB
JavaScript
const React = require('react');
|
|
const { connect } = require('react-redux');
|
|
const Setting = require('lib/models/Setting.js');
|
|
const { bridge } = require('electron').remote.require('./bridge');
|
|
const { themeStyle } = require('../theme.js');
|
|
const pathUtils = require('lib/path-utils.js');
|
|
const { _ } = require('lib/locale.js');
|
|
const SyncTargetRegistry = require('lib/SyncTargetRegistry');
|
|
const shared = require('lib/components/shared/config-shared.js');
|
|
const ConfigMenuBar = require('./ConfigMenuBar.min.js');
|
|
const { EncryptionConfigScreen } = require('./EncryptionConfigScreen.min');
|
|
const { ClipperConfigScreen } = require('./ClipperConfigScreen.min');
|
|
|
|
class ConfigScreenComponent extends React.Component {
|
|
constructor() {
|
|
super();
|
|
|
|
shared.init(this);
|
|
|
|
this.state.selectedSectionName = 'general';
|
|
this.state.screenName = '';
|
|
|
|
this.checkSyncConfig_ = async () => {
|
|
await shared.checkSyncConfig(this, this.state.settings);
|
|
};
|
|
|
|
this.checkNextcloudAppButton_click = async () => {
|
|
this.setState({ showNextcloudAppLog: true });
|
|
await shared.checkNextcloudApp(this, this.state.settings);
|
|
};
|
|
|
|
this.showLogButton_click = () => {
|
|
this.setState({ showNextcloudAppLog: true });
|
|
};
|
|
|
|
this.nextcloudAppHelpLink_click = () => {
|
|
bridge().openExternal('https://joplinapp.org/nextcloud_app');
|
|
};
|
|
|
|
this.rowStyle_ = {
|
|
marginBottom: 10,
|
|
};
|
|
|
|
this.configMenuBar_selectionChange = this.configMenuBar_selectionChange.bind(this);
|
|
}
|
|
|
|
UNSAFE_componentWillMount() {
|
|
this.setState({ settings: this.props.settings });
|
|
}
|
|
|
|
componentDidMount() {
|
|
if (this.props.defaultSection) {
|
|
this.setState({ selectedSectionName: this.props.defaultSection }, () => {
|
|
this.switchSection(this.props.defaultSection);
|
|
});
|
|
}
|
|
}
|
|
|
|
sectionByName(name) {
|
|
const sections = shared.settingsSections({ device: 'desktop', settings: this.state.settings });
|
|
for (const section of sections) {
|
|
if (section.name === name) return section;
|
|
}
|
|
|
|
throw new Error(`Invalid section name: ${name}`);
|
|
}
|
|
|
|
screenFromName(screenName) {
|
|
if (screenName === 'encryption') return <EncryptionConfigScreen theme={this.props.theme}/>;
|
|
if (screenName === 'server') return <ClipperConfigScreen theme={this.props.theme}/>;
|
|
|
|
throw new Error(`Invalid screen name: ${screenName}`);
|
|
}
|
|
|
|
switchSection(name) {
|
|
const section = this.sectionByName(name);
|
|
let screenName = '';
|
|
if (section.isScreen) {
|
|
screenName = section.name;
|
|
|
|
if (this.hasChanges()) {
|
|
const ok = confirm(_('This will open a new screen. Save your current changes?'));
|
|
if (ok) shared.saveSettings(this);
|
|
}
|
|
}
|
|
|
|
this.setState({ selectedSectionName: section.name, screenName: screenName });
|
|
}
|
|
|
|
configMenuBar_selectionChange(event) {
|
|
this.switchSection(event.section.name);
|
|
}
|
|
|
|
keyValueToArray(kv) {
|
|
const output = [];
|
|
for (const k in kv) {
|
|
if (!kv.hasOwnProperty(k)) continue;
|
|
output.push({
|
|
key: k,
|
|
label: kv[k],
|
|
});
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
renderSectionDescription(section) {
|
|
const description = Setting.sectionDescription(section.name);
|
|
if (!description) return null;
|
|
|
|
const theme = themeStyle(this.props.theme);
|
|
return (
|
|
<div style={Object.assign({}, theme.textStyle, { marginBottom: 15 })}>
|
|
{description}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
sectionToComponent(key, section, settings, selected) {
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
const createSettingComponents = (advanced) => {
|
|
const output = [];
|
|
for (let i = 0; i < section.metadatas.length; i++) {
|
|
const md = section.metadatas[i];
|
|
if (!!md.advanced !== advanced) continue;
|
|
const settingComp = this.settingToComponent(md.key, settings[md.key]);
|
|
output.push(settingComp);
|
|
}
|
|
return output;
|
|
};
|
|
|
|
const settingComps = createSettingComponents(false);
|
|
const advancedSettingComps = createSettingComponents(true);
|
|
|
|
const sectionStyle = {
|
|
marginTop: 20,
|
|
marginBottom: 20,
|
|
};
|
|
|
|
if (!selected) sectionStyle.display = 'none';
|
|
|
|
if (section.name === 'general') {
|
|
sectionStyle.borderTopWidth = 0;
|
|
}
|
|
|
|
if (section.name === 'sync') {
|
|
const syncTargetMd = SyncTargetRegistry.idToMetadata(settings['sync.target']);
|
|
const statusStyle = Object.assign({}, theme.textStyle, { marginTop: 10 });
|
|
|
|
if (syncTargetMd.supportsConfigCheck) {
|
|
const messages = shared.checkSyncConfigMessages(this);
|
|
const statusComp = !messages.length ? null : (
|
|
<div style={statusStyle}>
|
|
{messages[0]}
|
|
{messages.length >= 1 ? <p>{messages[1]}</p> : null}
|
|
</div>
|
|
);
|
|
|
|
settingComps.push(
|
|
<div key="check_sync_config_button" style={this.rowStyle_}>
|
|
<button disabled={this.state.checkSyncConfigResult === 'checking'} style={theme.buttonStyle} onClick={this.checkSyncConfig_}>
|
|
{_('Check synchronisation configuration')}
|
|
</button>
|
|
{statusComp}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
if (syncTargetMd.name === 'nextcloud') {
|
|
const syncTarget = settings['sync.5.syncTargets'][settings['sync.5.path']];
|
|
|
|
let status = _('Unknown');
|
|
let errorMessage = null;
|
|
|
|
if (this.state.checkNextcloudAppResult === 'checking') {
|
|
status = _('Checking...');
|
|
} else if (syncTarget) {
|
|
if (syncTarget.uuid) status = _('OK');
|
|
if (syncTarget.error) {
|
|
status = _('Error');
|
|
errorMessage = syncTarget.error;
|
|
}
|
|
}
|
|
|
|
const statusComp = !errorMessage || this.state.checkNextcloudAppResult === 'checking' || !this.state.showNextcloudAppLog ? null : (
|
|
<div style={statusStyle}>
|
|
<p style={theme.textStyle}>{_('The Joplin Nextcloud App is either not installed or misconfigured. Please see the full error message below:')}</p>
|
|
<pre>{errorMessage}</pre>
|
|
</div>
|
|
);
|
|
|
|
const showLogButton = !errorMessage || this.state.showNextcloudAppLog ? null : (
|
|
<a style={theme.urlStyle} href="#" onClick={this.showLogButton_click}>[{_('Show Log')}]</a>
|
|
);
|
|
|
|
const appStatusStyle = Object.assign({}, theme.textStyle, { fontWeight: 'bold' });
|
|
|
|
settingComps.push(
|
|
<div key="nextcloud_app_check" style={this.rowStyle_}>
|
|
<span style={theme.textStyle}>Beta: {_('Joplin Nextcloud App status:')} </span><span style={appStatusStyle}>{status}</span>
|
|
|
|
{showLogButton}
|
|
|
|
<button disabled={this.state.checkNextcloudAppResult === 'checking'} style={theme.buttonStyle} onClick={this.checkNextcloudAppButton_click}>
|
|
{_('Check Status')}
|
|
</button>
|
|
|
|
<a style={theme.urlStyle} href="#" onClick={this.nextcloudAppHelpLink_click}>[{_('Help')}]</a>
|
|
{statusComp}
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
let advancedSettingsButton = null;
|
|
const advancedSettingsSectionStyle = { display: 'none' };
|
|
|
|
if (advancedSettingComps.length) {
|
|
const iconName = this.state.showAdvancedSettings ? 'fa fa-angle-down' : 'fa fa-angle-right';
|
|
const advancedSettingsButtonStyle = Object.assign({}, theme.buttonStyle, { marginBottom: 10 });
|
|
advancedSettingsButton = <button onClick={() => shared.advancedSettingsButton_click(this)} style={advancedSettingsButtonStyle}><i style={{ fontSize: 14 }} className={iconName}></i> {_('Show Advanced Settings')}</button>;
|
|
advancedSettingsSectionStyle.display = this.state.showAdvancedSettings ? 'block' : 'none';
|
|
}
|
|
|
|
return (
|
|
<div key={key} style={sectionStyle}>
|
|
{this.renderSectionDescription(section)}
|
|
<div>{settingComps}</div>
|
|
{advancedSettingsButton}
|
|
<div style={advancedSettingsSectionStyle}>{advancedSettingComps}</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
settingToComponent(key, value) {
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
const output = null;
|
|
|
|
const rowStyle = this.rowStyle_;
|
|
|
|
const labelStyle = Object.assign({}, theme.textStyle, {
|
|
display: 'inline-block',
|
|
marginRight: 10,
|
|
color: theme.color,
|
|
});
|
|
|
|
const subLabel = Object.assign({}, labelStyle, {
|
|
opacity: 0.7,
|
|
marginBottom: Math.round(rowStyle.marginBottom * 0.7),
|
|
});
|
|
|
|
const invisibleLabel = Object.assign({}, labelStyle, {
|
|
opacity: 0,
|
|
});
|
|
|
|
const checkboxLabelStyle = Object.assign({}, labelStyle, {
|
|
marginLeft: 8,
|
|
});
|
|
|
|
const controlStyle = {
|
|
display: 'inline-block',
|
|
color: theme.color,
|
|
backgroundColor: theme.backgroundColor,
|
|
};
|
|
|
|
const descriptionStyle = Object.assign({}, theme.textStyle, {
|
|
color: theme.colorFaded,
|
|
marginTop: 5,
|
|
fontStyle: 'italic',
|
|
maxWidth: '70em',
|
|
});
|
|
|
|
const textInputBaseStyle = Object.assign({}, controlStyle, {
|
|
border: '1px solid',
|
|
padding: '4px 6px',
|
|
borderColor: theme.dividerColor,
|
|
borderRadius: 4,
|
|
});
|
|
|
|
const updateSettingValue = (key, value) => {
|
|
// console.info(key + ' = ' + value);
|
|
return shared.updateSettingValue(this, key, value);
|
|
};
|
|
|
|
// Component key needs to be key+value otherwise it doesn't update when the settings change.
|
|
|
|
const md = Setting.settingMetadata(key);
|
|
|
|
const descriptionText = Setting.keyDescription(key, 'desktop');
|
|
const descriptionComp = descriptionText ? <div style={descriptionStyle}>{descriptionText}</div> : null;
|
|
|
|
if (md.isEnum) {
|
|
const items = [];
|
|
const settingOptions = md.options();
|
|
const array = this.keyValueToArray(settingOptions);
|
|
for (let i = 0; i < array.length; i++) {
|
|
const e = array[i];
|
|
items.push(
|
|
<option value={e.key.toString()} key={e.key}>
|
|
{settingOptions[e.key]}
|
|
</option>
|
|
);
|
|
}
|
|
|
|
const selectStyle = Object.assign({}, controlStyle, { height: 22, borderColor: theme.dividerColor });
|
|
|
|
return (
|
|
<div key={key} style={rowStyle}>
|
|
<div style={labelStyle}>
|
|
<label>{md.label()}</label>
|
|
</div>
|
|
<select
|
|
value={value}
|
|
style={selectStyle}
|
|
onChange={event => {
|
|
updateSettingValue(key, event.target.value);
|
|
}}
|
|
>
|
|
{items}
|
|
</select>
|
|
{descriptionComp}
|
|
</div>
|
|
);
|
|
} else if (md.type === Setting.TYPE_BOOL) {
|
|
const onCheckboxClick = () => {
|
|
updateSettingValue(key, !value);
|
|
};
|
|
|
|
// Hack: The {key+value.toString()} is needed as otherwise the checkbox doesn't update when the state changes.
|
|
// There's probably a better way to do this but can't figure it out.
|
|
|
|
return (
|
|
<div key={key + value.toString()} style={rowStyle}>
|
|
<div style={controlStyle}>
|
|
<input
|
|
id={`setting_checkbox_${key}`}
|
|
type="checkbox"
|
|
checked={!!value}
|
|
onChange={event => {
|
|
onCheckboxClick(event);
|
|
}}
|
|
/>
|
|
<label
|
|
onClick={event => {
|
|
onCheckboxClick(event);
|
|
}}
|
|
style={checkboxLabelStyle}
|
|
htmlFor={`setting_checkbox_${key}`}
|
|
>
|
|
{md.label()}
|
|
</label>
|
|
{descriptionComp}
|
|
</div>
|
|
</div>
|
|
);
|
|
} else if (md.type === Setting.TYPE_STRING) {
|
|
const inputStyle = Object.assign({}, textInputBaseStyle, {
|
|
width: '50%',
|
|
minWidth: '20em',
|
|
});
|
|
const inputType = md.secure === true ? 'password' : 'text';
|
|
|
|
if (md.subType === 'file_path_and_args') {
|
|
inputStyle.marginBottom = subLabel.marginBottom;
|
|
|
|
const splitCmd = cmdString => {
|
|
const path = pathUtils.extractExecutablePath(cmdString);
|
|
const args = cmdString.substr(path.length + 1);
|
|
return [pathUtils.unquotePath(path), args];
|
|
};
|
|
|
|
const joinCmd = cmdArray => {
|
|
if (!cmdArray[0] && !cmdArray[1]) return '';
|
|
let cmdString = pathUtils.quotePath(cmdArray[0]);
|
|
if (!cmdString) cmdString = '""';
|
|
if (cmdArray[1]) cmdString += ` ${cmdArray[1]}`;
|
|
return cmdString;
|
|
};
|
|
|
|
const onPathChange = event => {
|
|
const cmd = splitCmd(this.state.settings[key]);
|
|
cmd[0] = event.target.value;
|
|
updateSettingValue(key, joinCmd(cmd));
|
|
};
|
|
|
|
const onArgsChange = event => {
|
|
const cmd = splitCmd(this.state.settings[key]);
|
|
cmd[1] = event.target.value;
|
|
updateSettingValue(key, joinCmd(cmd));
|
|
};
|
|
|
|
const browseButtonClick = () => {
|
|
const paths = bridge().showOpenDialog();
|
|
if (!paths || !paths.length) return;
|
|
const cmd = splitCmd(this.state.settings[key]);
|
|
cmd[0] = paths[0];
|
|
updateSettingValue(key, joinCmd(cmd));
|
|
};
|
|
|
|
const cmd = splitCmd(this.state.settings[key]);
|
|
|
|
return (
|
|
<div key={key} style={rowStyle}>
|
|
<div style={{ display: 'flex' }}>
|
|
<div style={{ flex: 0, whiteSpace: 'nowrap' }}>
|
|
<div style={labelStyle}>
|
|
<label>{md.label()}</label>
|
|
</div>
|
|
</div>
|
|
<div style={{ flex: 0 }}>
|
|
<div style={subLabel}>Path:</div>
|
|
<div style={subLabel}>Arguments:</div>
|
|
</div>
|
|
<div style={{ flex: 1 }}>
|
|
<div style={{ display: 'flex', flexDirection: 'row', alignItems: 'center', marginBottom: inputStyle.marginBottom }}>
|
|
<input
|
|
type={inputType}
|
|
style={Object.assign({}, inputStyle, { marginBottom: 0 })}
|
|
onChange={event => {
|
|
onPathChange(event);
|
|
}}
|
|
value={cmd[0]}
|
|
/>
|
|
<button onClick={browseButtonClick} style={Object.assign({}, theme.buttonStyle, { marginLeft: 5 })}>
|
|
{_('Browse...')}
|
|
</button>
|
|
</div>
|
|
<input
|
|
type={inputType}
|
|
style={inputStyle}
|
|
onChange={event => {
|
|
onArgsChange(event);
|
|
}}
|
|
value={cmd[1]}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
<div style={{ display: 'flex' }}>
|
|
<div style={{ flex: 0, whiteSpace: 'nowrap' }}>
|
|
<div style={invisibleLabel}>
|
|
<label>{md.label()}</label>
|
|
</div>
|
|
</div>
|
|
<div style={{ flex: 1 }}>{descriptionComp}</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
} else {
|
|
const onTextChange = event => {
|
|
updateSettingValue(key, event.target.value);
|
|
};
|
|
|
|
return (
|
|
<div key={key} style={rowStyle}>
|
|
<div style={labelStyle}>
|
|
<label>{md.label()}</label>
|
|
</div>
|
|
<input
|
|
type={inputType}
|
|
style={inputStyle}
|
|
value={this.state.settings[key]}
|
|
onChange={event => {
|
|
onTextChange(event);
|
|
}}
|
|
/>
|
|
{descriptionComp}
|
|
</div>
|
|
);
|
|
}
|
|
} else if (md.type === Setting.TYPE_INT) {
|
|
const onNumChange = event => {
|
|
updateSettingValue(key, event.target.value);
|
|
};
|
|
|
|
const label = [md.label()];
|
|
if (md.unitLabel) label.push(`(${md.unitLabel()})`);
|
|
|
|
const inputStyle = Object.assign({}, textInputBaseStyle);
|
|
|
|
return (
|
|
<div key={key} style={rowStyle}>
|
|
<div style={labelStyle}>
|
|
<label>{label.join(' ')}</label>
|
|
</div>
|
|
<input
|
|
type="number"
|
|
style={inputStyle}
|
|
value={this.state.settings[key]}
|
|
onChange={event => {
|
|
onNumChange(event);
|
|
}}
|
|
min={md.minimum}
|
|
max={md.maximum}
|
|
step={md.step}
|
|
/>
|
|
{descriptionComp}
|
|
</div>
|
|
);
|
|
} else if (md.type === Setting.TYPE_BUTTON) {
|
|
const theme = themeStyle(this.props.theme);
|
|
const buttonStyle = Object.assign({}, theme.buttonStyle, {
|
|
display: 'inline-block',
|
|
marginRight: 10,
|
|
});
|
|
|
|
return (
|
|
<div key={key} style={rowStyle}>
|
|
<div style={labelStyle}>
|
|
<label>{md.label()}</label>
|
|
</div>
|
|
<button style={buttonStyle} onClick={md.onClick}>
|
|
{_('Edit')}
|
|
</button>
|
|
{descriptionComp}
|
|
</div>
|
|
);
|
|
} else {
|
|
console.warn(`Type not implemented: ${key}`);
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
onApplyClick() {
|
|
shared.saveSettings(this);
|
|
}
|
|
|
|
onSaveClick() {
|
|
shared.saveSettings(this);
|
|
this.props.dispatch({ type: 'NAV_BACK' });
|
|
}
|
|
|
|
onCancelClick() {
|
|
this.props.dispatch({ type: 'NAV_BACK' });
|
|
}
|
|
|
|
hasChanges() {
|
|
return !!this.state.changedSettingKeys.length;
|
|
}
|
|
|
|
render() {
|
|
const theme = themeStyle(this.props.theme);
|
|
|
|
const style = Object.assign(
|
|
{
|
|
backgroundColor: theme.backgroundColor,
|
|
},
|
|
this.props.style,
|
|
{
|
|
overflow: 'hidden',
|
|
display: 'flex',
|
|
flexDirection: 'column',
|
|
}
|
|
);
|
|
|
|
const settings = this.state.settings;
|
|
|
|
const containerStyle = Object.assign({}, theme.containerStyle, { padding: 10, paddingTop: 0, display: 'flex', flex: 1 });
|
|
|
|
const hasChanges = this.hasChanges();
|
|
|
|
const buttonStyle = Object.assign({}, theme.buttonStyle, {
|
|
display: 'inline-block',
|
|
marginRight: 10,
|
|
});
|
|
|
|
const buttonStyleApprove = Object.assign({}, buttonStyle, {
|
|
opacity: hasChanges ? 1 : theme.disabledOpacity,
|
|
});
|
|
|
|
const settingComps = shared.settingsToComponents2(this, 'desktop', settings, this.state.selectedSectionName);
|
|
|
|
const buttonBarStyle = {
|
|
display: 'flex',
|
|
alignItems: 'center',
|
|
padding: 10,
|
|
borderTopWidth: 1,
|
|
borderTopStyle: 'solid',
|
|
borderTopColor: theme.dividerColor,
|
|
};
|
|
|
|
const screenComp = this.state.screenName ? <div style={{ overflow: 'scroll', flex: 1 }}>{this.screenFromName(this.state.screenName)}</div> : null;
|
|
|
|
if (screenComp) containerStyle.display = 'none';
|
|
|
|
const sections = shared.settingsSections({ device: 'desktop', settings });
|
|
|
|
return (
|
|
<div style={style}>
|
|
<ConfigMenuBar
|
|
selection={this.state.selectedSectionName}
|
|
onSelectionChange={this.configMenuBar_selectionChange}
|
|
sections={sections}
|
|
theme={this.props.theme}
|
|
/>
|
|
{screenComp}
|
|
<div style={containerStyle}>{settingComps}</div>
|
|
<div style={buttonBarStyle}>
|
|
<button
|
|
onClick={() => {
|
|
this.onCancelClick();
|
|
}}
|
|
style={buttonStyle}
|
|
>
|
|
<i style={theme.buttonIconStyle} className={'fa fa-chevron-left'}></i>
|
|
{hasChanges && !screenComp ? _('Cancel') : _('Back')}
|
|
</button>
|
|
{ !screenComp && (
|
|
<div>
|
|
<button disabled={!hasChanges} onClick={() => { this.onSaveClick(); }} style={buttonStyleApprove}>{_('OK')}</button>
|
|
<button disabled={!hasChanges} onClick={() => { this.onApplyClick(); }} style={buttonStyleApprove}>{_('Apply')}</button>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
}
|
|
|
|
const mapStateToProps = state => {
|
|
return {
|
|
theme: state.settings.theme,
|
|
settings: state.settings,
|
|
locale: state.settings.locale,
|
|
};
|
|
};
|
|
|
|
const ConfigScreen = connect(mapStateToProps)(ConfigScreenComponent);
|
|
|
|
module.exports = { ConfigScreen };
|