2016-04-01 06:52:03 +08:00
|
|
|
import React from 'react';
|
2017-09-27 02:33:08 +08:00
|
|
|
import PropTypes from 'prop-types';
|
2017-09-27 02:46:00 +08:00
|
|
|
import { RetinaImg, Flexbox } from 'mailspring-component-kit';
|
2016-04-01 06:52:03 +08:00
|
|
|
|
2017-12-27 23:34:40 +08:00
|
|
|
class AppearanceScaleSlider extends React.Component {
|
|
|
|
static displayName = 'AppearanceScaleSlider';
|
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
id: PropTypes.string,
|
|
|
|
config: PropTypes.object.isRequired,
|
|
|
|
};
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super();
|
|
|
|
this.kp = `core.workspace.interfaceZoom`;
|
|
|
|
this.state = { value: props.config.get(this.kp) };
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillReceiveProps(nextProps) {
|
|
|
|
this.setState({ value: nextProps.config.get(this.kp) });
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<div id={this.props.id} className="appearance-scale-slider">
|
|
|
|
<div className="ruler">
|
|
|
|
<div style={{ flex: 1.02 }}>
|
|
|
|
<RetinaImg name="appearance-scale-small.png" mode={RetinaImg.Mode.ContentDark} />
|
|
|
|
</div>
|
|
|
|
<div className="midpoint" />
|
|
|
|
<div style={{ flex: 2, textAlign: 'right' }}>
|
|
|
|
<RetinaImg name="appearance-scale-big.png" mode={RetinaImg.Mode.ContentDark} />
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
<input
|
|
|
|
type="range"
|
|
|
|
min={0.8}
|
|
|
|
max={1.4}
|
|
|
|
step={0.05}
|
|
|
|
value={this.state.value}
|
|
|
|
onChange={e => this.props.config.set(this.kp, e.target.value)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-02 02:00:17 +08:00
|
|
|
class MenubarStylePicker extends React.Component {
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
this.kp = 'core.workspace.menubarStyle';
|
|
|
|
}
|
|
|
|
|
|
|
|
onChangeMenubarStyle = e => {
|
|
|
|
this.props.config.set(this.kp, e.target.value);
|
|
|
|
};
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const val = this.props.config.get(this.kp) || 'default';
|
|
|
|
|
|
|
|
const options = [
|
|
|
|
['default', 'Default Window Controls and Menubar', ''],
|
|
|
|
[
|
|
|
|
'autohide',
|
|
|
|
'Default Window Controls and Auto-hiding Menubar',
|
|
|
|
'(Requires supported window manager. Press `Alt` to show menu.)',
|
|
|
|
],
|
|
|
|
['hamburger', 'Custom Window Frame and Right-hand Menu', ''],
|
|
|
|
];
|
|
|
|
|
|
|
|
return (
|
|
|
|
<section className="platform-linux-only">
|
|
|
|
<h6 htmlFor="change-layout">Window Controls and Menus</h6>
|
|
|
|
{options.map(([enumValue, description, comment], idx) => (
|
|
|
|
<div key={enumValue} style={{ marginBottom: 10 }}>
|
|
|
|
<label htmlFor={`radio${idx}`}>
|
|
|
|
<input
|
|
|
|
id={`radio${idx}`}
|
|
|
|
type="radio"
|
|
|
|
value={enumValue}
|
|
|
|
name="menubarStyle"
|
|
|
|
checked={val === enumValue}
|
2018-01-23 14:18:46 +08:00
|
|
|
onChange={this.onChangeMenubarStyle}
|
2018-01-02 02:00:17 +08:00
|
|
|
/>
|
|
|
|
{` ${description}`}
|
|
|
|
{comment && (
|
|
|
|
<div style={{ paddingLeft: 24, fontSize: '0.9em', opacity: 0.7 }}>{comment}</div>
|
|
|
|
)}
|
|
|
|
</label>
|
|
|
|
</div>
|
|
|
|
))}
|
|
|
|
<div className="platform-note" style={{ lineHeight: '23px' }}>
|
|
|
|
<div
|
|
|
|
className="btn btn-small"
|
|
|
|
style={{ float: 'right' }}
|
|
|
|
onClick={() => {
|
|
|
|
require('electron').remote.app.relaunch();
|
|
|
|
require('electron').remote.app.quit();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
Relaunch
|
|
|
|
</div>
|
|
|
|
Relaunch to apply window changes.
|
|
|
|
</div>
|
|
|
|
</section>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-01 06:52:03 +08:00
|
|
|
class AppearanceModeSwitch extends React.Component {
|
|
|
|
static displayName = 'AppearanceModeSwitch';
|
|
|
|
|
|
|
|
static propTypes = {
|
2017-12-27 23:34:40 +08:00
|
|
|
id: PropTypes.string,
|
2017-09-27 02:33:08 +08:00
|
|
|
config: PropTypes.object.isRequired,
|
2016-04-01 06:52:03 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
constructor(props) {
|
|
|
|
super();
|
|
|
|
this.state = {
|
|
|
|
value: props.config.get('core.workspace.mode'),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillReceiveProps(nextProps) {
|
|
|
|
this.setState({
|
|
|
|
value: nextProps.config.get('core.workspace.mode'),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
_onApplyChanges = () => {
|
2017-09-27 02:36:58 +08:00
|
|
|
AppEnv.commands.dispatch(`application:select-${this.state.value}-mode`);
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2016-04-01 06:52:03 +08:00
|
|
|
|
|
|
|
_renderModeOptions() {
|
2017-09-27 02:33:08 +08:00
|
|
|
return ['list', 'split'].map(mode => (
|
2016-04-01 06:52:03 +08:00
|
|
|
<AppearanceModeOption
|
|
|
|
mode={mode}
|
|
|
|
key={mode}
|
|
|
|
active={this.state.value === mode}
|
2017-09-27 02:33:08 +08:00
|
|
|
onClick={() => this.setState({ value: mode })}
|
2016-05-07 07:24:40 +08:00
|
|
|
/>
|
2017-09-27 02:33:08 +08:00
|
|
|
));
|
2016-04-01 06:52:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const hasChanges = this.state.value !== this.props.config.get('core.workspace.mode');
|
2017-09-27 02:33:08 +08:00
|
|
|
let applyChangesClass = 'btn';
|
|
|
|
if (!hasChanges) applyChangesClass += ' btn-disabled';
|
2016-04-01 06:52:03 +08:00
|
|
|
|
|
|
|
return (
|
2017-12-27 23:34:40 +08:00
|
|
|
<div id={this.props.id} className="appearance-mode-switch">
|
2017-09-27 02:33:08 +08:00
|
|
|
<Flexbox direction="row" style={{ alignItems: 'center' }} className="item">
|
2016-04-01 06:52:03 +08:00
|
|
|
{this._renderModeOptions()}
|
|
|
|
</Flexbox>
|
2017-09-27 02:33:08 +08:00
|
|
|
<div className={applyChangesClass} onClick={this._onApplyChanges}>
|
|
|
|
Apply Layout
|
|
|
|
</div>
|
2016-04-01 06:52:03 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-07 07:24:40 +08:00
|
|
|
const AppearanceModeOption = function AppearanceModeOption(props) {
|
2017-09-27 02:33:08 +08:00
|
|
|
let classname = 'appearance-mode';
|
|
|
|
if (props.active) classname += ' active';
|
2016-05-07 07:24:40 +08:00
|
|
|
|
|
|
|
const label = {
|
|
|
|
list: 'Single Panel',
|
|
|
|
split: 'Two Panel',
|
|
|
|
}[props.mode];
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className={classname} onClick={props.onClick}>
|
|
|
|
<RetinaImg name={`appearance-mode-${props.mode}.png`} mode={RetinaImg.Mode.ContentIsMask} />
|
|
|
|
<div>{label}</div>
|
|
|
|
</div>
|
|
|
|
);
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2016-05-07 07:24:40 +08:00
|
|
|
AppearanceModeOption.propTypes = {
|
2017-09-27 02:33:08 +08:00
|
|
|
mode: PropTypes.string.isRequired,
|
|
|
|
active: PropTypes.bool,
|
|
|
|
onClick: PropTypes.func,
|
|
|
|
};
|
2016-04-01 06:52:03 +08:00
|
|
|
|
|
|
|
class PreferencesAppearance extends React.Component {
|
|
|
|
static displayName = 'PreferencesAppearance';
|
|
|
|
|
|
|
|
static propTypes = {
|
2017-09-27 02:33:08 +08:00
|
|
|
config: PropTypes.object,
|
|
|
|
configSchema: PropTypes.object,
|
|
|
|
};
|
2016-04-01 06:52:03 +08:00
|
|
|
|
2018-01-02 02:00:17 +08:00
|
|
|
onPickTheme = () => {
|
2017-09-27 02:36:58 +08:00
|
|
|
AppEnv.commands.dispatch('window:launch-theme-picker');
|
2017-09-27 02:33:08 +08:00
|
|
|
};
|
2016-04-20 08:14:25 +08:00
|
|
|
|
2016-04-01 06:52:03 +08:00
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<div className="container-appearance">
|
2018-01-02 02:00:17 +08:00
|
|
|
<section>
|
|
|
|
<h6 htmlFor="change-layout">Layout</h6>
|
|
|
|
<AppearanceModeSwitch id="change-layout" config={this.props.config} />
|
|
|
|
</section>
|
|
|
|
<section>
|
|
|
|
<h6 htmlFor="change-layout" style={{ marginTop: 10 }}>
|
|
|
|
Theme and Style
|
|
|
|
</h6>
|
|
|
|
<div>
|
|
|
|
<button className="btn btn-large" onClick={this.onPickTheme}>
|
|
|
|
Change theme...
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
</section>
|
|
|
|
<MenubarStylePicker config={this.props.config} />
|
|
|
|
<section>
|
|
|
|
<h6 htmlFor="change-scale">Scaling</h6>
|
|
|
|
<AppearanceScaleSlider id="change-scale" config={this.props.config} />
|
|
|
|
<div className="platform-note">
|
|
|
|
Scaling adjusts the entire UI, including icons, dividers, and text. Messages you send
|
|
|
|
will still have the same font size. Decreasing scale significantly may make dividers and
|
|
|
|
icons too small to click.
|
|
|
|
</div>
|
|
|
|
</section>
|
2016-04-01 06:52:03 +08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default PreferencesAppearance;
|