Show volumes
This commit is contained in:
parent
a507e1a6c3
commit
947469300b
|
@ -26,6 +26,10 @@ const {
|
||||||
|
|
||||||
const { getPaiByTypeAndIndex } = require('../../selectors');
|
const { getPaiByTypeAndIndex } = require('../../selectors');
|
||||||
|
|
||||||
|
const {
|
||||||
|
PA_VOLUME_NORM,
|
||||||
|
} = require('../../constants/pulse');
|
||||||
|
|
||||||
const {
|
const {
|
||||||
GraphView,
|
GraphView,
|
||||||
} = require('./satellites-graph');
|
} = require('./satellites-graph');
|
||||||
|
@ -207,6 +211,51 @@ const renderNode = (nodeRef, data, key, selected, hovered) => r({
|
||||||
hovered,
|
hovered,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const VolumeThumbnail = ({ pai, state }) => {
|
||||||
|
if (state.preferences.hideVolumeThumbnails) {
|
||||||
|
return r(React.Fragment);
|
||||||
|
}
|
||||||
|
|
||||||
|
const volumes = (pai && pai.channelVolumes) || [];
|
||||||
|
const muted = !pai || pai.muted;
|
||||||
|
|
||||||
|
const step = size / 32;
|
||||||
|
const padding = 2;
|
||||||
|
const width = size - 8;
|
||||||
|
const height = ((1 + volumes.length) * step);
|
||||||
|
|
||||||
|
return r.svg({
|
||||||
|
classSet: {
|
||||||
|
'volume-thumbnail': true,
|
||||||
|
'volume-thumbnail-muted': muted,
|
||||||
|
},
|
||||||
|
}, [
|
||||||
|
r.line({
|
||||||
|
className: 'volume-thumbnail-ruler-line',
|
||||||
|
x1: padding,
|
||||||
|
x2: padding,
|
||||||
|
y1: padding,
|
||||||
|
y2: padding + height,
|
||||||
|
}),
|
||||||
|
|
||||||
|
r.line({
|
||||||
|
className: 'volume-thumbnail-ruler-line',
|
||||||
|
x1: padding + width,
|
||||||
|
x2: padding + width,
|
||||||
|
y1: padding,
|
||||||
|
y2: padding + height,
|
||||||
|
}),
|
||||||
|
|
||||||
|
...volumes.map((v, i) => r.line({
|
||||||
|
className: 'volume-thumbnail-volume-line',
|
||||||
|
x1: padding,
|
||||||
|
x2: padding + ((v / PA_VOLUME_NORM) * width),
|
||||||
|
y1: padding + ((1 + i) * step),
|
||||||
|
y2: padding + ((1 + i) * step),
|
||||||
|
})),
|
||||||
|
]);
|
||||||
|
};
|
||||||
|
|
||||||
const DebugText = ({ dgo, pai, state }) => r.div({
|
const DebugText = ({ dgo, pai, state }) => r.div({
|
||||||
style: {
|
style: {
|
||||||
fontSize: '50%',
|
fontSize: '50%',
|
||||||
|
@ -218,20 +267,25 @@ const DebugText = ({ dgo, pai, state }) => r.div({
|
||||||
|
|
||||||
const SinkText = ({ dgo, pai, state }) => r.div([
|
const SinkText = ({ dgo, pai, state }) => r.div([
|
||||||
r.div({
|
r.div({
|
||||||
|
className: 'node-name',
|
||||||
title: pai.name,
|
title: pai.name,
|
||||||
}, pai.description),
|
}, pai.description),
|
||||||
|
r(VolumeThumbnail, { pai, state }),
|
||||||
r(DebugText, { dgo, pai, state }),
|
r(DebugText, { dgo, pai, state }),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const SourceText = ({ dgo, pai, state }) => r.div([
|
const SourceText = ({ dgo, pai, state }) => r.div([
|
||||||
r.div({
|
r.div({
|
||||||
|
className: 'node-name',
|
||||||
title: pai.name,
|
title: pai.name,
|
||||||
}, pai.description),
|
}, pai.description),
|
||||||
|
r(VolumeThumbnail, { pai, state }),
|
||||||
r(DebugText, { dgo, pai, state }),
|
r(DebugText, { dgo, pai, state }),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const ClientText = ({ dgo, pai, state }) => r.div([
|
const ClientText = ({ dgo, pai, state }) => r.div([
|
||||||
r.div({
|
r.div({
|
||||||
|
className: 'node-name',
|
||||||
title: path('properties.application.process.binary'.split('.'), pai),
|
title: path('properties.application.process.binary'.split('.'), pai),
|
||||||
}, pai.name),
|
}, pai.name),
|
||||||
r(DebugText, { dgo, pai, state }),
|
r(DebugText, { dgo, pai, state }),
|
||||||
|
@ -239,6 +293,7 @@ const ClientText = ({ dgo, pai, state }) => r.div([
|
||||||
|
|
||||||
const ModuleText = ({ dgo, pai, state }) => r.div([
|
const ModuleText = ({ dgo, pai, state }) => r.div([
|
||||||
r.div({
|
r.div({
|
||||||
|
className: 'node-name',
|
||||||
title: pai.properties.module.description,
|
title: pai.properties.module.description,
|
||||||
}, pai.name),
|
}, pai.name),
|
||||||
r(DebugText, { dgo, pai, state }),
|
r(DebugText, { dgo, pai, state }),
|
||||||
|
@ -248,17 +303,11 @@ const renderNodeText = state => dgo => r('foreignObject', {
|
||||||
x: -s2,
|
x: -s2,
|
||||||
y: -s2,
|
y: -s2,
|
||||||
}, r.div({
|
}, r.div({
|
||||||
|
className: 'node-text',
|
||||||
style: {
|
style: {
|
||||||
width: size,
|
width: size,
|
||||||
height: size,
|
height: size,
|
||||||
|
|
||||||
padding: 2,
|
|
||||||
|
|
||||||
whiteSpace: 'pre',
|
|
||||||
|
|
||||||
backgroundRepeat: 'no-repeat',
|
|
||||||
backgroundSize: '60%',
|
|
||||||
backgroundPosition: 'center',
|
|
||||||
backgroundImage: (icon => icon && `url(${icon})`)(state.icons[getPaiIcon(dgoToPai.get(dgo))]),
|
backgroundImage: (icon => icon && `url(${icon})`)(state.icons[getPaiIcon(dgoToPai.get(dgo))]),
|
||||||
},
|
},
|
||||||
}, r({
|
}, r({
|
||||||
|
@ -279,28 +328,22 @@ const renderEdge = props => r(Edge, {
|
||||||
...props,
|
...props,
|
||||||
});
|
});
|
||||||
|
|
||||||
const renderEdgeText = state => ({ data, transform }) => r('foreignObject', {
|
const renderEdgeText = state => ({ data: dgo, transform }) => {
|
||||||
transform,
|
const pai = dgo.type && getPaiByTypeAndIndex(dgo.type, dgo.index)({ pulse: state });
|
||||||
}, r.div({
|
|
||||||
style: {
|
|
||||||
width: size,
|
|
||||||
height: size,
|
|
||||||
|
|
||||||
padding: 2,
|
return r('foreignObject', {
|
||||||
|
transform,
|
||||||
whiteSpace: 'pre',
|
}, r.div({
|
||||||
|
className: 'edge-text',
|
||||||
backgroundRepeat: 'no-repeat',
|
style: {
|
||||||
backgroundSize: '60%',
|
width: size,
|
||||||
backgroundPosition: 'center',
|
height: size,
|
||||||
},
|
},
|
||||||
}, [
|
}, [
|
||||||
r(DebugText, {
|
pai && r(VolumeThumbnail, { pai, state }),
|
||||||
dgo: data,
|
r(DebugText, { dgo, pai, state }),
|
||||||
pai: data.type && getPaiByTypeAndIndex(data.type, data.index)({ pulse: state }),
|
]));
|
||||||
state,
|
};
|
||||||
}),
|
|
||||||
]));
|
|
||||||
|
|
||||||
class Graph extends React.Component {
|
class Graph extends React.Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
|
@ -334,16 +377,16 @@ class Graph extends React.Component {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
this.getIconPath('audio-volume-muted');
|
||||||
|
}
|
||||||
|
|
||||||
componentDidUpdate() {
|
componentDidUpdate() {
|
||||||
forEach(pai => {
|
forEach(pai => {
|
||||||
const icon = getPaiIcon(pai);
|
const icon = getPaiIcon(pai);
|
||||||
if (!icon) {
|
if (icon) {
|
||||||
return;
|
this.getIconPath(icon);
|
||||||
}
|
}
|
||||||
if (!this._requestedIcons.has(icon) && !this.props.icons[icon]) {
|
|
||||||
this.props.getIconPath(icon, 128);
|
|
||||||
}
|
|
||||||
this._requestedIcons.add(icon);
|
|
||||||
}, flatten(map(values, [
|
}, flatten(map(values, [
|
||||||
this.props.infos.sinks,
|
this.props.infos.sinks,
|
||||||
this.props.infos.sources,
|
this.props.infos.sources,
|
||||||
|
@ -352,6 +395,13 @@ class Graph extends React.Component {
|
||||||
])));
|
])));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getIconPath(icon) {
|
||||||
|
if (!this._requestedIcons.has(icon) && !this.props.icons[icon]) {
|
||||||
|
this.props.getIconPath(icon, 128);
|
||||||
|
}
|
||||||
|
this._requestedIcons.add(icon);
|
||||||
|
}
|
||||||
|
|
||||||
onSelectNode(selected) {
|
onSelectNode(selected) {
|
||||||
this.setState({ selected });
|
this.setState({ selected });
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,6 +77,13 @@ const Preferences = withStateHandlers(
|
||||||
}, 'Hide pulseaudio applications'),
|
}, 'Hide pulseaudio applications'),
|
||||||
]),
|
]),
|
||||||
|
|
||||||
|
r.div([
|
||||||
|
r(Checkbox, {
|
||||||
|
checked: props.preferences.hideVolumeThumbnails,
|
||||||
|
onChange: () => props.actions.toggle('hideVolumeThumbnails'),
|
||||||
|
}, 'Hide volume thumbnails'),
|
||||||
|
]),
|
||||||
|
|
||||||
r.div([
|
r.div([
|
||||||
r(Checkbox, {
|
r(Checkbox, {
|
||||||
checked: props.preferences.showDebugInfo,
|
checked: props.preferences.showDebugInfo,
|
||||||
|
|
|
@ -1,4 +1,6 @@
|
||||||
|
|
||||||
|
const PA_VOLUME_NORM = 0x10000;
|
||||||
|
|
||||||
const things = [ {
|
const things = [ {
|
||||||
method: 'getModules',
|
method: 'getModules',
|
||||||
type: 'module',
|
type: 'module',
|
||||||
|
@ -30,5 +32,7 @@ const things = [ {
|
||||||
} ];
|
} ];
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
|
PA_VOLUME_NORM,
|
||||||
|
|
||||||
things,
|
things,
|
||||||
};
|
};
|
||||||
|
|
61
index.css
61
index.css
|
@ -71,30 +71,33 @@ button:active {
|
||||||
fill: var(--themeSelectedBgColor);
|
fill: var(--themeSelectedBgColor);
|
||||||
}
|
}
|
||||||
|
|
||||||
.view-wrapper .sourceOutput .edge {
|
.view-wrapper .edge.edge {
|
||||||
|
marker-end: none;
|
||||||
|
}
|
||||||
|
.view-wrapper .sourceOutput .edge-path {
|
||||||
marker-end: url(#my-source-arrow);
|
marker-end: url(#my-source-arrow);
|
||||||
}
|
}
|
||||||
.view-wrapper .sourceOutput .edge.selected {
|
.view-wrapper .sourceOutput .selected .edge-path {
|
||||||
marker-end: url(#my-source-arrow-selected);
|
marker-end: url(#my-source-arrow-selected);
|
||||||
}
|
}
|
||||||
|
|
||||||
.view-wrapper .sinkInput .edge {
|
.view-wrapper .sinkInput .edge-path {
|
||||||
marker-end: url(#my-sink-arrow);
|
marker-end: url(#my-sink-arrow);
|
||||||
}
|
}
|
||||||
.view-wrapper .sinkInput .edge.selected {
|
.view-wrapper .sinkInput .selected .edge-path {
|
||||||
marker-end: url(#my-sink-arrow-selected);
|
marker-end: url(#my-sink-arrow-selected);
|
||||||
}
|
}
|
||||||
|
|
||||||
#edge-custom-container .edge {
|
#edge-custom-container .edge-path {
|
||||||
marker-end: none;
|
marker-end: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.view-wrapper .graph .edge {
|
.view-wrapper .graph .edge {
|
||||||
stroke: var(--successColor);
|
stroke: var(--borders);
|
||||||
}
|
}
|
||||||
|
|
||||||
.view-wrapper .graph .arrow {
|
.view-wrapper .graph .arrow {
|
||||||
fill: var(--successColor);
|
fill: var(--borders);
|
||||||
}
|
}
|
||||||
|
|
||||||
.preferences {
|
.preferences {
|
||||||
|
@ -125,3 +128,47 @@ button:active {
|
||||||
.view-wrapper .graph .edge-mouse-handler {
|
.view-wrapper .graph .edge-mouse-handler {
|
||||||
stroke-width: 30px;
|
stroke-width: 30px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.node {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-text, .edge-text {
|
||||||
|
pointer-events: none;
|
||||||
|
|
||||||
|
padding: 2;
|
||||||
|
|
||||||
|
white-space: pre;
|
||||||
|
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: 60%;
|
||||||
|
background-position: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.node-name {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.volume-thumbnail-ruler-line {
|
||||||
|
stroke-width: 2px;
|
||||||
|
stroke: var(--borders);
|
||||||
|
}
|
||||||
|
.node.selected .volume-thumbnail-ruler-line {
|
||||||
|
stroke: var(--themeSelectedFgColor)
|
||||||
|
}
|
||||||
|
.volume-thumbnail-volume-line {
|
||||||
|
stroke-width: 2px;
|
||||||
|
stroke: var(--successColor);
|
||||||
|
}
|
||||||
|
.node.selected .volume-thumbnail-volume-line {
|
||||||
|
stroke: var(--themeSelectedFgColor)
|
||||||
|
}
|
||||||
|
|
||||||
|
.volume-thumbnail-muted .volume-thumbnail-volume-line {
|
||||||
|
stroke: var(--borders)
|
||||||
|
}
|
||||||
|
|
||||||
|
.edge-text {
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
|
@ -16,6 +16,8 @@ const initialState = {
|
||||||
hideMonitors: false,
|
hideMonitors: false,
|
||||||
hidePulseaudioApps: true,
|
hidePulseaudioApps: true,
|
||||||
|
|
||||||
|
hideVolumeThumbnails: false,
|
||||||
|
|
||||||
showDebugInfo: false,
|
showDebugInfo: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue
Block a user