Add Connect to server
form
This commit is contained in:
parent
ab8f7d698c
commit
4ab936986f
|
@ -14,6 +14,7 @@ const ref = memoizeWith(autoFocus => String(Boolean(autoFocus)), autoFocus => in
|
||||||
const Button = props => r.button({
|
const Button = props => r.button({
|
||||||
ref,
|
ref,
|
||||||
className: 'button',
|
className: 'button',
|
||||||
|
type: 'button',
|
||||||
...props,
|
...props,
|
||||||
}, props.children);
|
}, props.children);
|
||||||
|
|
||||||
|
|
|
@ -11,15 +11,23 @@ const {
|
||||||
Provider,
|
Provider,
|
||||||
} = require('@futpib/react-electron-menu');
|
} = require('@futpib/react-electron-menu');
|
||||||
|
|
||||||
const MenuProvider = ({ children }) => r(Provider, { electron }, r(React.Fragment, {}, [
|
const MenuProvider = ({ children, ...props }) => r(Provider, { electron }, r(React.Fragment, {}, [
|
||||||
r(WindowMenu),
|
r(WindowMenu, props),
|
||||||
...[].concat(children),
|
...[].concat(children),
|
||||||
]));
|
]));
|
||||||
|
|
||||||
const WindowMenu = () => r(WindowMenuBase, [
|
const WindowMenu = ({ openConnectToServerModal }) => r(WindowMenuBase, [
|
||||||
r(MenuItem, {
|
r(MenuItem, {
|
||||||
label: 'App',
|
label: 'File',
|
||||||
}, [
|
}, [
|
||||||
|
r(MenuItem, {
|
||||||
|
label: 'Connect to server...',
|
||||||
|
accelerator: 'CommandOrControl+N',
|
||||||
|
onClick: openConnectToServerModal,
|
||||||
|
}),
|
||||||
|
|
||||||
|
r(MenuItem.Separator),
|
||||||
|
|
||||||
r(MenuItem, {
|
r(MenuItem, {
|
||||||
label: 'Quit',
|
label: 'Quit',
|
||||||
role: 'quit',
|
role: 'quit',
|
||||||
|
|
51
components/modals/confirmation.js
Normal file
51
components/modals/confirmation.js
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
|
||||||
|
const r = require('r-dom');
|
||||||
|
|
||||||
|
const React = require('react');
|
||||||
|
|
||||||
|
const Modal = require('react-modal');
|
||||||
|
|
||||||
|
const Checkbox = require('../checkbox');
|
||||||
|
const Button = require('../button');
|
||||||
|
|
||||||
|
class ConfirmationModal extends React.PureComponent {
|
||||||
|
render() {
|
||||||
|
const { target, confirmation, onConfirm, onCancel } = this.props;
|
||||||
|
|
||||||
|
return r(Modal, {
|
||||||
|
isOpen: Boolean(confirmation),
|
||||||
|
onRequestClose: onCancel,
|
||||||
|
}, [
|
||||||
|
confirmation === 'unloadModuleByIndex' && r(React.Fragment, [
|
||||||
|
r.h3('Module unload confirmation'),
|
||||||
|
|
||||||
|
target && r.p([
|
||||||
|
'You are about to unload ',
|
||||||
|
r.code(target.name),
|
||||||
|
'.',
|
||||||
|
'This may not be easily undoable and may impair sound playback on your system.',
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
|
||||||
|
r(Checkbox, {
|
||||||
|
checked: this.props.preferences.doNotAskForConfirmations,
|
||||||
|
onChange: () => this.props.toggle('doNotAskForConfirmations'),
|
||||||
|
}, 'Do not ask for confirmations'),
|
||||||
|
|
||||||
|
r.div({
|
||||||
|
className: 'button-group',
|
||||||
|
}, [
|
||||||
|
r(Button, {
|
||||||
|
onClick: onCancel,
|
||||||
|
}, 'Cancel'),
|
||||||
|
|
||||||
|
r(Button, {
|
||||||
|
onClick: onConfirm,
|
||||||
|
autoFocus: true,
|
||||||
|
}, 'Confirm'),
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = ConfirmationModal;
|
87
components/modals/connect-to-server.js
Normal file
87
components/modals/connect-to-server.js
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
|
||||||
|
const { spawn } = require('child_process');
|
||||||
|
|
||||||
|
const {
|
||||||
|
merge,
|
||||||
|
} = require('ramda');
|
||||||
|
|
||||||
|
const r = require('r-dom');
|
||||||
|
|
||||||
|
const React = require('react');
|
||||||
|
|
||||||
|
const Modal = require('react-modal');
|
||||||
|
|
||||||
|
const Button = require('../button');
|
||||||
|
const Label = require('../label');
|
||||||
|
const Input = require('../input');
|
||||||
|
|
||||||
|
class ConnectToServerModal extends React.PureComponent {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {
|
||||||
|
value: 'tcp:remote-computer.lan',
|
||||||
|
};
|
||||||
|
|
||||||
|
this.handleSubmit = this.handleSubmit.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleSubmit(e) {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const subprocess = spawn('pagraphcontrol', [], {
|
||||||
|
detached: true,
|
||||||
|
stdio: 'ignore',
|
||||||
|
env: merge(process.env, {
|
||||||
|
PULSE_SERVER: this.state.value,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
subprocess.unref();
|
||||||
|
|
||||||
|
this.props.onRequestClose();
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
const { isOpen, onRequestClose } = this.props;
|
||||||
|
|
||||||
|
return r(Modal, {
|
||||||
|
isOpen,
|
||||||
|
onRequestClose,
|
||||||
|
}, [
|
||||||
|
r.h3('Connect to PulseAudio server'),
|
||||||
|
|
||||||
|
r.form({
|
||||||
|
onSubmit: this.handleSubmit,
|
||||||
|
}, [
|
||||||
|
r(Label, [
|
||||||
|
r.div({
|
||||||
|
title: 'Same format as PULSE_SERVER',
|
||||||
|
}, 'Specify the server to connect to:'),
|
||||||
|
r.p([
|
||||||
|
r(Input, {
|
||||||
|
style: { width: '100%' },
|
||||||
|
autoFocus: true,
|
||||||
|
value: this.state.value,
|
||||||
|
onChange: e => this.setState({ value: e.target.value }),
|
||||||
|
}),
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
|
||||||
|
r.div({
|
||||||
|
className: 'button-group',
|
||||||
|
}, [
|
||||||
|
r(Button, {
|
||||||
|
onClick: onRequestClose,
|
||||||
|
}, 'Cancel'),
|
||||||
|
|
||||||
|
r(Button, {
|
||||||
|
type: 'submit',
|
||||||
|
}, 'Connect'),
|
||||||
|
]),
|
||||||
|
]),
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = ConnectToServerModal;
|
|
@ -26,8 +26,8 @@ const {
|
||||||
|
|
||||||
const { modules } = require('../../constants/pulse');
|
const { modules } = require('../../constants/pulse');
|
||||||
|
|
||||||
const Checkbox = require('../checkbox');
|
const ConnectToServerModal = require('./connect-to-server');
|
||||||
const Button = require('../button');
|
const ConfirmationModal = require('./confirmation');
|
||||||
|
|
||||||
Modal.setAppElement('#root');
|
Modal.setAppElement('#root');
|
||||||
|
|
||||||
|
@ -36,46 +36,6 @@ Modal.defaultStyles = {
|
||||||
content: {},
|
content: {},
|
||||||
};
|
};
|
||||||
|
|
||||||
class ConfirmationModal extends React.PureComponent {
|
|
||||||
render() {
|
|
||||||
const { target, confirmation, onConfirm, onCancel } = this.props;
|
|
||||||
|
|
||||||
return r(Modal, {
|
|
||||||
isOpen: Boolean(confirmation),
|
|
||||||
onRequestClose: onCancel,
|
|
||||||
}, [
|
|
||||||
confirmation === 'unloadModuleByIndex' && r(React.Fragment, [
|
|
||||||
r.h3('Module unload confirmation'),
|
|
||||||
|
|
||||||
target && r.p([
|
|
||||||
'You are about to unload ',
|
|
||||||
r.code(target.name),
|
|
||||||
'.',
|
|
||||||
'This may not be easily undoable and may impair sound playback on your system.',
|
|
||||||
]),
|
|
||||||
]),
|
|
||||||
|
|
||||||
r(Checkbox, {
|
|
||||||
checked: this.props.preferences.doNotAskForConfirmations,
|
|
||||||
onChange: () => this.props.toggle('doNotAskForConfirmations'),
|
|
||||||
}, 'Do not ask for confirmations'),
|
|
||||||
|
|
||||||
r.div({
|
|
||||||
className: 'button-group',
|
|
||||||
}, [
|
|
||||||
r(Button, {
|
|
||||||
onClick: onCancel,
|
|
||||||
}, 'Cancel'),
|
|
||||||
|
|
||||||
r(Button, {
|
|
||||||
onClick: onConfirm,
|
|
||||||
autoFocus: true,
|
|
||||||
}, 'Confirm'),
|
|
||||||
]),
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class Modals extends React.PureComponent {
|
class Modals extends React.PureComponent {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
@ -84,15 +44,21 @@ class Modals extends React.PureComponent {
|
||||||
target: null,
|
target: null,
|
||||||
confirmation: null,
|
confirmation: null,
|
||||||
continuation: null,
|
continuation: null,
|
||||||
|
|
||||||
|
connectToServerModalOpen: false,
|
||||||
|
|
||||||
|
actions: {
|
||||||
|
openConnectToServerModal: this.openConnectToServerModal.bind(this),
|
||||||
|
},
|
||||||
};
|
};
|
||||||
this.state = this.initialState;
|
this.state = this.initialState;
|
||||||
|
|
||||||
this.handleCancel = this.handleCancel.bind(this);
|
this.handleCancel = this.handleCancel.bind(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
static getDerivedStateFromProps(props) {
|
static getDerivedStateFromProps(props, state) {
|
||||||
return {
|
return {
|
||||||
actions: mapObjIndexed((f, name) => function (...args) {
|
actions: merge(state.actions, mapObjIndexed((f, name) => function (...args) {
|
||||||
const continuation = () => {
|
const continuation = () => {
|
||||||
props[name](...args);
|
props[name](...args);
|
||||||
this.setState(this.initialState);
|
this.setState(this.initialState);
|
||||||
|
@ -123,10 +89,14 @@ class Modals extends React.PureComponent {
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
}),
|
})),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
openConnectToServerModal() {
|
||||||
|
this.setState({ connectToServerModalOpen: true });
|
||||||
|
}
|
||||||
|
|
||||||
handleCancel() {
|
handleCancel() {
|
||||||
this.setState(this.initialState);
|
this.setState(this.initialState);
|
||||||
}
|
}
|
||||||
|
@ -136,7 +106,7 @@ class Modals extends React.PureComponent {
|
||||||
const { actions, target, confirmation, continuation } = this.state;
|
const { actions, target, confirmation, continuation } = this.state;
|
||||||
|
|
||||||
return r(React.Fragment, [
|
return r(React.Fragment, [
|
||||||
...children({ actions: map(a => a.bind(this), actions) }),
|
...[].concat(children({ actions: map(a => a.bind(this), actions) })),
|
||||||
|
|
||||||
r(ConfirmationModal, {
|
r(ConfirmationModal, {
|
||||||
target,
|
target,
|
||||||
|
@ -147,6 +117,11 @@ class Modals extends React.PureComponent {
|
||||||
preferences,
|
preferences,
|
||||||
toggle,
|
toggle,
|
||||||
}),
|
}),
|
||||||
|
|
||||||
|
r(ConnectToServerModal, {
|
||||||
|
isOpen: this.state.connectToServerModalOpen,
|
||||||
|
onRequestClose: this.handleCancel,
|
||||||
|
}),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,10 +20,11 @@ const theme = require('./utils/theme');
|
||||||
|
|
||||||
const Root = () => r(ReduxProvider, {
|
const Root = () => r(ReduxProvider, {
|
||||||
store: createStore(),
|
store: createStore(),
|
||||||
}, r(MenuProvider, {
|
|
||||||
}, r(HotKeys, {
|
}, r(HotKeys, {
|
||||||
}, ({ graphRef, cardsRef, preferencesRef }) => r(Modals, {
|
}, ({ graphRef, cardsRef, preferencesRef }) => r(Modals, {
|
||||||
}, ({ actions }) => [
|
}, ({ actions }) => r(MenuProvider, {
|
||||||
|
...actions,
|
||||||
|
}, [
|
||||||
r(Graph, { ref: graphRef, ...actions }),
|
r(Graph, { ref: graphRef, ...actions }),
|
||||||
r(Cards, { ref: cardsRef }),
|
r(Cards, { ref: cardsRef }),
|
||||||
r(Preferences, { ref: preferencesRef }),
|
r(Preferences, { ref: preferencesRef }),
|
||||||
|
|
|
@ -119,7 +119,7 @@ module.exports = store => {
|
||||||
store.dispatch(pulseActions.remove({ type, index }));
|
store.dispatch(pulseActions.remove({ type, index }));
|
||||||
})
|
})
|
||||||
.on('error', error => {
|
.on('error', error => {
|
||||||
console.error(error);
|
handleError(error);
|
||||||
});
|
});
|
||||||
|
|
||||||
const reconnect = () => new Bluebird((resolve, reject) => {
|
const reconnect = () => new Bluebird((resolve, reject) => {
|
||||||
|
|
Loading…
Reference in New Issue
Block a user