Add volume peaks display
This commit is contained in:
parent
12ae7aa7df
commit
e1e66de61d
BIN
assets/trail.png
Normal file
BIN
assets/trail.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.9 KiB |
|
@ -31,6 +31,11 @@ class GraphView extends GraphViewBase {
|
||||||
_super_renderBackground: this.renderBackground,
|
_super_renderBackground: this.renderBackground,
|
||||||
renderBackground: this.constructor.prototype.renderBackground.bind(this),
|
renderBackground: this.constructor.prototype.renderBackground.bind(this),
|
||||||
|
|
||||||
|
_super_handleZoomStart: this.handleZoomStart,
|
||||||
|
handleZoomStart: this.constructor.prototype.handleZoomStart.bind(this),
|
||||||
|
_super_handleZoomEnd: this.handleZoomEnd,
|
||||||
|
handleZoomEnd: this.constructor.prototype.handleZoomEnd.bind(this),
|
||||||
|
|
||||||
_super_handleNodeMove: this.handleNodeMove,
|
_super_handleNodeMove: this.handleNodeMove,
|
||||||
handleNodeMove: this.constructor.prototype.handleNodeMove.bind(this),
|
handleNodeMove: this.constructor.prototype.handleNodeMove.bind(this),
|
||||||
|
|
||||||
|
@ -164,6 +169,8 @@ class GraphView extends GraphViewBase {
|
||||||
onNodeMouseDown: this.props.onNodeMouseDown,
|
onNodeMouseDown: this.props.onNodeMouseDown,
|
||||||
onNodeMouseEnter: this.handleNodeMouseEnter,
|
onNodeMouseEnter: this.handleNodeMouseEnter,
|
||||||
onNodeMouseLeave: this.handleNodeMouseLeave,
|
onNodeMouseLeave: this.handleNodeMouseLeave,
|
||||||
|
onNodeDragStart: this.props.onNodeDragStart,
|
||||||
|
onNodeDragEnd: this.props.onNodeDragEnd,
|
||||||
onNodeMove: this.handleNodeMove,
|
onNodeMove: this.handleNodeMove,
|
||||||
onNodeUpdate: this.handleNodeUpdate,
|
onNodeUpdate: this.handleNodeUpdate,
|
||||||
onNodeSelected: this.handleNodeSelected,
|
onNodeSelected: this.handleNodeSelected,
|
||||||
|
@ -175,6 +182,20 @@ class GraphView extends GraphViewBase {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleZoomStart(...args) {
|
||||||
|
if (this.props.onZoomStart) {
|
||||||
|
this.props.onZoomStart();
|
||||||
|
}
|
||||||
|
return this._super_handleZoomStart(...args);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleZoomEnd(...args) {
|
||||||
|
if (this.props.onZoomEnd) {
|
||||||
|
this.props.onZoomEnd();
|
||||||
|
}
|
||||||
|
return this._super_handleZoomEnd(...args);
|
||||||
|
}
|
||||||
|
|
||||||
handleNodeMove(position, nodeId, shiftKey) {
|
handleNodeMove(position, nodeId, shiftKey) {
|
||||||
this._super_handleNodeMove(position, nodeId, shiftKey);
|
this._super_handleNodeMove(position, nodeId, shiftKey);
|
||||||
if (this.props.onNodeMove) {
|
if (this.props.onNodeMove) {
|
||||||
|
@ -254,6 +275,9 @@ class Node extends NodeBase {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
Object.assign(this, {
|
Object.assign(this, {
|
||||||
|
_super_handleDragStart: this.handleDragStart,
|
||||||
|
handleDragStart: this.constructor.prototype.handleDragStart.bind(this),
|
||||||
|
|
||||||
_super_handleDragEnd: this.handleDragEnd,
|
_super_handleDragEnd: this.handleDragEnd,
|
||||||
handleDragEnd: this.constructor.prototype.handleDragEnd.bind(this),
|
handleDragEnd: this.constructor.prototype.handleDragEnd.bind(this),
|
||||||
|
|
||||||
|
@ -283,7 +307,17 @@ class Node extends NodeBase {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleDragStart(...args) {
|
||||||
|
if (this.props.onNodeDragStart) {
|
||||||
|
this.props.onNodeDragStart(...args);
|
||||||
|
}
|
||||||
|
return this._super_handleDragStart(...args);
|
||||||
|
}
|
||||||
|
|
||||||
handleDragEnd(...args) {
|
handleDragEnd(...args) {
|
||||||
|
if (this.props.onNodeDragEnd) {
|
||||||
|
this.props.onNodeDragEnd(...args);
|
||||||
|
}
|
||||||
this.oldSibling = null;
|
this.oldSibling = null;
|
||||||
return this._super_handleDragEnd(...args);
|
return this._super_handleDragEnd(...args);
|
||||||
}
|
}
|
||||||
|
|
|
@ -78,13 +78,15 @@ const { primaryPulseServer } = require('../../reducers/pulse');
|
||||||
const { keyMap } = require('../hot-keys');
|
const { keyMap } = require('../hot-keys');
|
||||||
|
|
||||||
const {
|
const {
|
||||||
GraphView,
|
SatellitesGraphView,
|
||||||
} = require('./satellites-graph');
|
} = require('./satellites-graph');
|
||||||
|
|
||||||
const {
|
const {
|
||||||
Edge,
|
Edge,
|
||||||
} = require('./base');
|
} = require('./base');
|
||||||
|
|
||||||
|
const Peaks = require('./peaks');
|
||||||
|
|
||||||
const LayoutEngine = require('./layout-engine');
|
const LayoutEngine = require('./layout-engine');
|
||||||
|
|
||||||
const maximum = reduce(max, -Infinity);
|
const maximum = reduce(max, -Infinity);
|
||||||
|
@ -268,7 +270,7 @@ const renderDefs = () => r(React.Fragment, [
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const renderBackground = ({
|
const renderBackground = ({
|
||||||
gridSize = 40960,
|
gridSize = 40960 / 4,
|
||||||
onMouseDown,
|
onMouseDown,
|
||||||
}) => r.rect({
|
}) => r.rect({
|
||||||
className: 'background',
|
className: 'background',
|
||||||
|
@ -717,14 +719,19 @@ class GraphObjectContextMenu extends React.PureComponent {
|
||||||
|
|
||||||
const backgroundSymbol = Symbol('graph.backgroundSymbol');
|
const backgroundSymbol = Symbol('graph.backgroundSymbol');
|
||||||
|
|
||||||
class Graph extends React.Component {
|
class Graph extends React.PureComponent {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
|
this.satellitesGraphViewRef = React.createRef();
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
selected: null,
|
selected: null,
|
||||||
moved: null,
|
moved: null,
|
||||||
contexted: null,
|
contexted: null,
|
||||||
|
|
||||||
|
isDraggingNode: false,
|
||||||
|
isZooming: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
this._requestedIcons = new Set();
|
this._requestedIcons = new Set();
|
||||||
|
@ -732,11 +739,16 @@ class Graph extends React.Component {
|
||||||
Object.assign(this, {
|
Object.assign(this, {
|
||||||
onBackgroundMouseDown: this.onBackgroundMouseDown.bind(this),
|
onBackgroundMouseDown: this.onBackgroundMouseDown.bind(this),
|
||||||
|
|
||||||
|
onZoomStart: this.onZoomStart.bind(this),
|
||||||
|
onZoomEnd: this.onZoomEnd.bind(this),
|
||||||
|
|
||||||
onSelectNode: this.onSelectNode.bind(this),
|
onSelectNode: this.onSelectNode.bind(this),
|
||||||
onCreateNode: this.onCreateNode.bind(this),
|
onCreateNode: this.onCreateNode.bind(this),
|
||||||
onUpdateNode: this.onUpdateNode.bind(this),
|
onUpdateNode: this.onUpdateNode.bind(this),
|
||||||
onDeleteNode: this.onDeleteNode.bind(this),
|
onDeleteNode: this.onDeleteNode.bind(this),
|
||||||
onNodeMouseDown: this.onNodeMouseDown.bind(this),
|
onNodeMouseDown: this.onNodeMouseDown.bind(this),
|
||||||
|
onNodeDragStart: this.onNodeDragStart.bind(this),
|
||||||
|
onNodeDragEnd: this.onNodeDragEnd.bind(this),
|
||||||
|
|
||||||
onSelectEdge: this.onSelectEdge.bind(this),
|
onSelectEdge: this.onSelectEdge.bind(this),
|
||||||
canCreateEdge: this.canCreateEdge.bind(this),
|
canCreateEdge: this.canCreateEdge.bind(this),
|
||||||
|
@ -803,6 +815,7 @@ class Graph extends React.Component {
|
||||||
if (binary.startsWith('pavucontrol') ||
|
if (binary.startsWith('pavucontrol') ||
|
||||||
binary.startsWith('kmix') ||
|
binary.startsWith('kmix') ||
|
||||||
binary === 'pulseaudio' ||
|
binary === 'pulseaudio' ||
|
||||||
|
name === 'papeaks' ||
|
||||||
name === 'paclient.js'
|
name === 'paclient.js'
|
||||||
) {
|
) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -857,19 +870,6 @@ class Graph extends React.Component {
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
shouldComponentUpdate(nextProps, nextState) {
|
|
||||||
return !(
|
|
||||||
(nextProps.serverInfo === this.props.serverInfo) &&
|
|
||||||
(nextProps.objects === this.props.objects) &&
|
|
||||||
(nextProps.infos === this.props.infos) &&
|
|
||||||
(nextProps.preferences === this.props.preferences) &&
|
|
||||||
(nextProps.icons === this.props.icons) &&
|
|
||||||
(nextState.selected === this.state.selected) &&
|
|
||||||
(nextState.contexted === this.state.contexted) &&
|
|
||||||
(nextState.moved === this.state.moved)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
componentDidMount() {
|
componentDidMount() {
|
||||||
this.getIconPath('starred');
|
this.getIconPath('starred');
|
||||||
|
|
||||||
|
@ -942,6 +942,18 @@ class Graph extends React.Component {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onNodeDragStart() {
|
||||||
|
this.setState({
|
||||||
|
isDraggingNode: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onNodeDragEnd() {
|
||||||
|
this.setState({
|
||||||
|
isDraggingNode: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
onSelectEdge(selected) {
|
onSelectEdge(selected) {
|
||||||
this.setState({ selected });
|
this.setState({ selected });
|
||||||
}
|
}
|
||||||
|
@ -1113,6 +1125,18 @@ class Graph extends React.Component {
|
||||||
this.graphViewElement.focus();
|
this.graphViewElement.focus();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onZoomStart() {
|
||||||
|
this.setState({
|
||||||
|
isZooming: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onZoomEnd() {
|
||||||
|
this.setState({
|
||||||
|
isZooming: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
hotKeyEscape() {
|
hotKeyEscape() {
|
||||||
const { moved } = this.state;
|
const { moved } = this.state;
|
||||||
|
|
||||||
|
@ -1397,12 +1421,29 @@ class Graph extends React.Component {
|
||||||
render() {
|
render() {
|
||||||
const { nodes, edges } = this.state;
|
const { nodes, edges } = this.state;
|
||||||
|
|
||||||
|
const satellitesGraphViewState = path(
|
||||||
|
[ 'current', 'state' ],
|
||||||
|
this.satellitesGraphViewRef,
|
||||||
|
);
|
||||||
|
|
||||||
return r(HotKeys, {
|
return r(HotKeys, {
|
||||||
handlers: map(f => bind(f, this), pick(keys(keyMap), this)),
|
handlers: map(f => bind(f, this), pick(keys(keyMap), this)),
|
||||||
}, r.div({
|
}, r.div({
|
||||||
id: 'graph',
|
id: 'graph',
|
||||||
}, [
|
}, [
|
||||||
r(GraphView, {
|
!this.props.preferences.hideLiveVolumePeaks && r(Peaks, {
|
||||||
|
key: 'peaks',
|
||||||
|
nodes: defaultTo([], prop('satelliteNodes', satellitesGraphViewState)),
|
||||||
|
edges: defaultTo([], prop('satelliteEdges', satellitesGraphViewState)),
|
||||||
|
accommodateGraphAnimation: this.state.isDraggingNode || this.state.isZooming,
|
||||||
|
peaks: this.props.peaks,
|
||||||
|
}),
|
||||||
|
|
||||||
|
r(SatellitesGraphView, {
|
||||||
|
key: 'graph',
|
||||||
|
|
||||||
|
ref: this.satellitesGraphViewRef,
|
||||||
|
|
||||||
nodeKey: 'id',
|
nodeKey: 'id',
|
||||||
edgeKey: 'id',
|
edgeKey: 'id',
|
||||||
|
|
||||||
|
@ -1418,11 +1459,16 @@ class Graph extends React.Component {
|
||||||
|
|
||||||
onBackgroundMouseDown: this.onBackgroundMouseDown,
|
onBackgroundMouseDown: this.onBackgroundMouseDown,
|
||||||
|
|
||||||
|
onZoomStart: this.onZoomStart,
|
||||||
|
onZoomEnd: this.onZoomEnd,
|
||||||
|
|
||||||
onSelectNode: this.onSelectNode,
|
onSelectNode: this.onSelectNode,
|
||||||
onCreateNode: this.onCreateNode,
|
onCreateNode: this.onCreateNode,
|
||||||
onUpdateNode: this.onUpdateNode,
|
onUpdateNode: this.onUpdateNode,
|
||||||
onDeleteNode: this.onDeleteNode,
|
onDeleteNode: this.onDeleteNode,
|
||||||
onNodeMouseDown: this.onNodeMouseDown,
|
onNodeMouseDown: this.onNodeMouseDown,
|
||||||
|
onNodeDragStart: this.onNodeDragStart,
|
||||||
|
onNodeDragEnd: this.onNodeDragEnd,
|
||||||
|
|
||||||
onSelectEdge: this.onSelectEdge,
|
onSelectEdge: this.onSelectEdge,
|
||||||
canCreateEdge: this.canCreateEdge,
|
canCreateEdge: this.canCreateEdge,
|
||||||
|
@ -1451,6 +1497,8 @@ class Graph extends React.Component {
|
||||||
this.state.contexted && (
|
this.state.contexted && (
|
||||||
this.state.contexted === backgroundSymbol ?
|
this.state.contexted === backgroundSymbol ?
|
||||||
r(BackgroundContextMenu, {
|
r(BackgroundContextMenu, {
|
||||||
|
key: 'background-context-menu',
|
||||||
|
|
||||||
onClose: this.onContextMenuClose,
|
onClose: this.onContextMenuClose,
|
||||||
|
|
||||||
onLoadModule: this.props.openLoadModuleModal,
|
onLoadModule: this.props.openLoadModuleModal,
|
||||||
|
@ -1460,6 +1508,8 @@ class Graph extends React.Component {
|
||||||
onLoadModuleNullSink: this.onLoadModuleNullSink,
|
onLoadModuleNullSink: this.onLoadModuleNullSink,
|
||||||
}) :
|
}) :
|
||||||
r(GraphObjectContextMenu, {
|
r(GraphObjectContextMenu, {
|
||||||
|
key: 'graph-object-context-menu',
|
||||||
|
|
||||||
onClose: this.onContextMenuClose,
|
onClose: this.onContextMenuClose,
|
||||||
|
|
||||||
canSetAsDefault: this.canContextMenuSetAsDefault,
|
canSetAsDefault: this.canContextMenuSetAsDefault,
|
||||||
|
|
149
components/graph/peaks.js
Normal file
149
components/graph/peaks.js
Normal file
|
@ -0,0 +1,149 @@
|
||||||
|
/* global window, performance */
|
||||||
|
|
||||||
|
const React = require('react');
|
||||||
|
|
||||||
|
const r = require('r-dom');
|
||||||
|
|
||||||
|
const PIXI = require('pixi.js');
|
||||||
|
|
||||||
|
const theme = require('../../utils/theme');
|
||||||
|
|
||||||
|
PIXI.ticker.shared.autoStart = false;
|
||||||
|
|
||||||
|
class Peaks extends React.Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {};
|
||||||
|
|
||||||
|
this.containerRef = React.createRef();
|
||||||
|
|
||||||
|
this.handleTick = this.handleTick.bind(this);
|
||||||
|
this.handlePeak = this.handlePeak.bind(this);
|
||||||
|
this.handleResize = this.handleResize.bind(this);
|
||||||
|
this.handleAnimationFrame = this.handleAnimationFrame.bind(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
this.app = new PIXI.Application(window.innerWidth, window.innerHeight, {
|
||||||
|
autoStart: false,
|
||||||
|
transparent: true,
|
||||||
|
});
|
||||||
|
this.app.ticker.add(this.handleTick);
|
||||||
|
|
||||||
|
this.trailTexture = PIXI.Texture.fromImage('assets/trail.png');
|
||||||
|
this.points = [
|
||||||
|
new PIXI.Point(0, 0),
|
||||||
|
new PIXI.Point(100, 100),
|
||||||
|
];
|
||||||
|
this.rope = new PIXI.mesh.Rope(this.trailTexture, this.points);
|
||||||
|
this.rope.blendmode = PIXI.BLEND_MODES.ADD;
|
||||||
|
this.app.stage.addChild(this.rope);
|
||||||
|
|
||||||
|
this.ropes = {};
|
||||||
|
|
||||||
|
this.containerRef.current.appendChild(this.app.view);
|
||||||
|
|
||||||
|
this.peaks = {};
|
||||||
|
this.props.peaks.on('peak', this.handlePeak);
|
||||||
|
|
||||||
|
this.graph = window.document.querySelector('#graph .graph');
|
||||||
|
this.view = this.graph.querySelector('.view');
|
||||||
|
|
||||||
|
window.addEventListener('resize', this.handleResize);
|
||||||
|
|
||||||
|
this.lastAnimationFrameTimeStamp = 0;
|
||||||
|
this.requestAnimationFrame();
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
this.app.destroy();
|
||||||
|
|
||||||
|
this.props.peaks.off('peak', this.handlePeak);
|
||||||
|
|
||||||
|
window.removeEventListener('resize', this.handleResize);
|
||||||
|
|
||||||
|
window.cancelAnimationFrame(this.animationFrameRequest);
|
||||||
|
}
|
||||||
|
|
||||||
|
requestAnimationFrame() {
|
||||||
|
this.animationFrameRequest = window.requestAnimationFrame(this.handleAnimationFrame);
|
||||||
|
}
|
||||||
|
|
||||||
|
get targetDelay() {
|
||||||
|
if (this.props.accommodateGraphAnimation) {
|
||||||
|
return 1000 / 70;
|
||||||
|
}
|
||||||
|
return 1000 / 25;
|
||||||
|
}
|
||||||
|
|
||||||
|
handleAnimationFrame(timeStamp) {
|
||||||
|
if (timeStamp < this.lastAnimationFrameTimeStamp + this.targetDelay) {
|
||||||
|
this.requestAnimationFrame();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.lastAnimationFrameTimeStamp = timeStamp;
|
||||||
|
|
||||||
|
this.app.ticker.update(timeStamp);
|
||||||
|
|
||||||
|
this.requestAnimationFrame();
|
||||||
|
}
|
||||||
|
|
||||||
|
handleTick() {
|
||||||
|
const matrix = this.view.getScreenCTM();
|
||||||
|
const point = this.graph.createSVGPoint();
|
||||||
|
|
||||||
|
const p = ({ x = 0, y = 0 }) => {
|
||||||
|
point.x = x;
|
||||||
|
point.y = y;
|
||||||
|
|
||||||
|
const p = point.matrixTransform(matrix);
|
||||||
|
|
||||||
|
return new PIXI.Point(p.x, p.y);
|
||||||
|
};
|
||||||
|
|
||||||
|
const ropes = this.props.edges
|
||||||
|
.filter(edge => {
|
||||||
|
return edge.type === 'sinkInput' || edge.type === 'sourceOutput';
|
||||||
|
})
|
||||||
|
.map(edge => {
|
||||||
|
const source = this.props.nodes.find(n => n.id === edge.source);
|
||||||
|
const target = this.props.nodes.find(n => n.id === edge.target);
|
||||||
|
|
||||||
|
const peak = this.peaks[target.target] || this.peaks[target.edge];
|
||||||
|
|
||||||
|
const points = [
|
||||||
|
p(target),
|
||||||
|
p(source),
|
||||||
|
];
|
||||||
|
const rope = new PIXI.mesh.Rope(this.trailTexture, points);
|
||||||
|
rope.blendmode = PIXI.BLEND_MODES.ADD;
|
||||||
|
rope.alpha = peak === undefined ? 0 : peak ** (1 / 3);
|
||||||
|
rope.tint = parseInt(theme.colors.themeSelectedBgColor.replace(/#/g, ''), 16);
|
||||||
|
|
||||||
|
return rope;
|
||||||
|
});
|
||||||
|
|
||||||
|
this.app.stage.removeChildren();
|
||||||
|
ropes.forEach(r => this.app.stage.addChild(r));
|
||||||
|
}
|
||||||
|
|
||||||
|
handlePeak(type, id, peak) {
|
||||||
|
this.peaks[`${type}-${id}`] = peak;
|
||||||
|
}
|
||||||
|
|
||||||
|
handleResize() {
|
||||||
|
this.app.renderer.resize(window.innerWidth, window.innerHeight);
|
||||||
|
this.app.ticker.update(performance.now());
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
return r.div({
|
||||||
|
className: 'peaks',
|
||||||
|
ref: this.containerRef,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = Peaks;
|
|
@ -59,7 +59,7 @@ const satelliteSpread = 36;
|
||||||
|
|
||||||
const satelliteEdgeToOriginalEdge = new WeakMap();
|
const satelliteEdgeToOriginalEdge = new WeakMap();
|
||||||
|
|
||||||
class GraphView extends React.Component {
|
class SatellitesGraphView extends React.Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
|
@ -116,10 +116,17 @@ class GraphView extends React.Component {
|
||||||
return satelliteNode;
|
return satelliteNode;
|
||||||
}, edges), originalEdgesByTargetNodeKey);
|
}, edges), originalEdgesByTargetNodeKey);
|
||||||
|
|
||||||
|
const satelliteNodes = flatten(map(node => {
|
||||||
|
const satelliteNodes = satelliteNodesByTargetNodeKey[node.id] || [];
|
||||||
|
SatellitesGraphView.repositionSatellites(node, satelliteNodes);
|
||||||
|
return satelliteNodes.concat(node);
|
||||||
|
}, props.nodes));
|
||||||
|
|
||||||
return {
|
return {
|
||||||
originalEdgesByTargetNodeKey,
|
originalEdgesByTargetNodeKey,
|
||||||
satelliteNodesByTargetNodeKey,
|
satelliteNodesByTargetNodeKey,
|
||||||
satelliteEdges,
|
satelliteEdges,
|
||||||
|
satelliteNodes,
|
||||||
|
|
||||||
selected,
|
selected,
|
||||||
moved,
|
moved,
|
||||||
|
@ -215,21 +222,14 @@ class GraphView extends React.Component {
|
||||||
}
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { nodeKey } = this.props;
|
|
||||||
const {
|
const {
|
||||||
satelliteNodesByTargetNodeKey,
|
|
||||||
satelliteEdges: edges,
|
satelliteEdges: edges,
|
||||||
|
satelliteNodes: nodes,
|
||||||
|
|
||||||
selected,
|
selected,
|
||||||
moved,
|
moved,
|
||||||
} = this.state;
|
} = this.state;
|
||||||
|
|
||||||
const nodes = flatten(map(node => {
|
|
||||||
const satelliteNodes = satelliteNodesByTargetNodeKey[node[nodeKey]] || [];
|
|
||||||
this.constructor.repositionSatellites(node, satelliteNodes);
|
|
||||||
return satelliteNodes.concat(node);
|
|
||||||
}, this.props.nodes));
|
|
||||||
|
|
||||||
return r(GraphViewBase, {
|
return r(GraphViewBase, {
|
||||||
...this.props,
|
...this.props,
|
||||||
|
|
||||||
|
@ -261,4 +261,4 @@ class GraphView extends React.Component {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = { GraphView };
|
module.exports = { SatellitesGraphView };
|
||||||
|
|
|
@ -73,7 +73,7 @@ class Modals extends React.PureComponent {
|
||||||
|
|
||||||
static getDerivedStateFromProps(props, state) {
|
static getDerivedStateFromProps(props, state) {
|
||||||
return {
|
return {
|
||||||
actions: merge(state.actions, mapObjIndexed((f, name) => function (...args) {
|
actions: merge(state.actions, map(a => a.bind(this), mapObjIndexed((f, name) => function (...args) {
|
||||||
const continuation = () => {
|
const continuation = () => {
|
||||||
props[name](...args);
|
props[name](...args);
|
||||||
this.setState(this.initialState);
|
this.setState(this.initialState);
|
||||||
|
@ -104,7 +104,7 @@ class Modals extends React.PureComponent {
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
})),
|
}))),
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,7 +139,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, [
|
||||||
...[].concat(children({ actions: map(a => a.bind(this), actions) })),
|
...[].concat(children({ actions })),
|
||||||
|
|
||||||
r(ConfirmationModal, {
|
r(ConfirmationModal, {
|
||||||
target,
|
target,
|
||||||
|
|
|
@ -154,6 +154,15 @@ class Preferences extends React.Component {
|
||||||
|
|
||||||
r.hr(),
|
r.hr(),
|
||||||
|
|
||||||
|
r.div([
|
||||||
|
r(Checkbox, {
|
||||||
|
checked: this.props.preferences.hideLiveVolumePeaks,
|
||||||
|
onChange: () => this.props.actions.toggle('hideLiveVolumePeaks'),
|
||||||
|
}, 'Hide live volume peaks'),
|
||||||
|
]),
|
||||||
|
|
||||||
|
r.hr(),
|
||||||
|
|
||||||
r.div([
|
r.div([
|
||||||
r(Checkbox, {
|
r(Checkbox, {
|
||||||
checked: this.props.preferences.hideOnScreenButtons,
|
checked: this.props.preferences.hideOnScreenButtons,
|
||||||
|
|
137
components/volume-peaks-provider/index.js
Normal file
137
components/volume-peaks-provider/index.js
Normal file
|
@ -0,0 +1,137 @@
|
||||||
|
|
||||||
|
const { EventEmitter } = require('events');
|
||||||
|
|
||||||
|
const { spawn } = require('child_process');
|
||||||
|
|
||||||
|
const { connect } = require('react-redux');
|
||||||
|
|
||||||
|
const React = require('react');
|
||||||
|
|
||||||
|
const r = require('r-dom');
|
||||||
|
|
||||||
|
const { primaryPulseServer } = require('../../reducers/pulse');
|
||||||
|
|
||||||
|
const PA_SUBSCRIPTION_EVENT_SOURCE = 0x0001;
|
||||||
|
const PA_SUBSCRIPTION_EVENT_SINK_INPUT = 0x0002;
|
||||||
|
|
||||||
|
const VolumePeaksContext = React.createContext(null);
|
||||||
|
|
||||||
|
function spawnProcess({ onPeak, onExit }) {
|
||||||
|
const process = spawn('/home/futpib/code/papeaks/target/release/papeaks', [
|
||||||
|
'--output',
|
||||||
|
'binary',
|
||||||
|
], {
|
||||||
|
stdio: [ 'ignore', 'pipe', 'inherit' ],
|
||||||
|
});
|
||||||
|
|
||||||
|
let leftover = null;
|
||||||
|
const handleData = data => {
|
||||||
|
if (leftover) {
|
||||||
|
data = Buffer.concat([ leftover, data ]);
|
||||||
|
}
|
||||||
|
|
||||||
|
let p = 0;
|
||||||
|
while (p < data.length) {
|
||||||
|
const left = data.length - p;
|
||||||
|
if (left >= 12) {
|
||||||
|
leftover = null;
|
||||||
|
} else {
|
||||||
|
leftover = data.slice(p);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
const type = data.readInt32LE(p);
|
||||||
|
p += 4;
|
||||||
|
const index = data.readInt32LE(p);
|
||||||
|
p += 4;
|
||||||
|
const peak = data.readFloatLE(p);
|
||||||
|
p += 4;
|
||||||
|
|
||||||
|
const typeStr = type === PA_SUBSCRIPTION_EVENT_SOURCE ?
|
||||||
|
'source' :
|
||||||
|
type === PA_SUBSCRIPTION_EVENT_SINK_INPUT ?
|
||||||
|
'sinkInput' :
|
||||||
|
'unexpected';
|
||||||
|
onPeak(typeStr, index, peak);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleExit = () => {
|
||||||
|
process.off('data', handleData);
|
||||||
|
process.off('exit', handleExit);
|
||||||
|
if (onExit) {
|
||||||
|
onExit();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
process.stdout.on('data', handleData);
|
||||||
|
process.on('exit', handleExit);
|
||||||
|
|
||||||
|
return process;
|
||||||
|
}
|
||||||
|
|
||||||
|
class VolumePeaksProvider extends React.Component {
|
||||||
|
constructor(props) {
|
||||||
|
super(props);
|
||||||
|
|
||||||
|
this.state = {};
|
||||||
|
|
||||||
|
this.emitter = new EventEmitter();
|
||||||
|
}
|
||||||
|
|
||||||
|
static getDerivedStateFromProps(props) {
|
||||||
|
const state = props.hideLiveVolumePeaks ? 'closed' : props.state;
|
||||||
|
return { state };
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidMount() {
|
||||||
|
if (this.state.state === 'ready') {
|
||||||
|
this._spawnProcess();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentDidUpdate(prevProps, prevState) {
|
||||||
|
if (this.state.state !== 'ready' && prevState.state === 'ready') {
|
||||||
|
this._killProcess();
|
||||||
|
} else if (this.state.state === 'ready' && prevState.state !== 'ready') {
|
||||||
|
this._spawnProcess();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
componentWillUnmount() {
|
||||||
|
this._killProcess();
|
||||||
|
this.emitter.removeAllListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
_spawnProcess() {
|
||||||
|
this.process = spawnProcess({
|
||||||
|
onPeak: (type, index, peak) => {
|
||||||
|
this.emitter.emit('peak', type, index, peak);
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_killProcess() {
|
||||||
|
if (this.process && !this.process.killed) {
|
||||||
|
this.process.kill();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
render() {
|
||||||
|
return r(VolumePeaksContext.Provider, {
|
||||||
|
value: this.emitter,
|
||||||
|
}, this.props.children);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
VolumePeaksProvider: connect(
|
||||||
|
state => ({
|
||||||
|
state: state.pulse[primaryPulseServer].state,
|
||||||
|
|
||||||
|
hideLiveVolumePeaks: state.preferences.hideLiveVolumePeaks,
|
||||||
|
}),
|
||||||
|
)(VolumePeaksProvider),
|
||||||
|
|
||||||
|
VolumePeaksConsumer: VolumePeaksContext.Consumer,
|
||||||
|
};
|
19
index.css
19
index.css
|
@ -100,8 +100,23 @@ div[tabindex="-1"]:focus {
|
||||||
border-color: var(--themeSelectedBgColor);
|
border-color: var(--themeSelectedBgColor);
|
||||||
}
|
}
|
||||||
|
|
||||||
.view-wrapper .graph {
|
.peaks {
|
||||||
background: var(--themeBaseColor);
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
bottom: 0;
|
||||||
|
right: 0;
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.view-wrapper.view-wrapper {
|
||||||
|
background: transparent;
|
||||||
|
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.view-wrapper .grid-dot {
|
.view-wrapper .grid-dot {
|
||||||
|
|
|
@ -35,6 +35,7 @@
|
||||||
"freedesktop-icons": "^0.1.0",
|
"freedesktop-icons": "^0.1.0",
|
||||||
"ini": "^1.3.5",
|
"ini": "^1.3.5",
|
||||||
"mathjs": "^5.2.3",
|
"mathjs": "^5.2.3",
|
||||||
|
"pixi.js": "^4.8.2",
|
||||||
"prop-types": "^15.6.2",
|
"prop-types": "^15.6.2",
|
||||||
"r-dom": "^2.4.0",
|
"r-dom": "^2.4.0",
|
||||||
"ramda": "^0.25.0",
|
"ramda": "^0.25.0",
|
||||||
|
|
|
@ -29,6 +29,8 @@ const initialState = {
|
||||||
maxVolume: 1.5,
|
maxVolume: 1.5,
|
||||||
volumeStep: 1 / 20,
|
volumeStep: 1 / 20,
|
||||||
|
|
||||||
|
hideLiveVolumePeaks: false,
|
||||||
|
|
||||||
doNotAskForConfirmations: false,
|
doNotAskForConfirmations: false,
|
||||||
showDebugInfo: false,
|
showDebugInfo: false,
|
||||||
|
|
||||||
|
|
|
@ -18,12 +18,15 @@ const ServerInfo = require('./components/server-info');
|
||||||
const { HotKeys } = require('./components/hot-keys');
|
const { HotKeys } = require('./components/hot-keys');
|
||||||
const { MenuProvider } = require('./components/menu');
|
const { MenuProvider } = require('./components/menu');
|
||||||
const Modals = require('./components/modals');
|
const Modals = require('./components/modals');
|
||||||
|
const { VolumePeaksProvider, VolumePeaksConsumer } = require('./components/volume-peaks-provider');
|
||||||
|
|
||||||
const theme = require('./utils/theme');
|
const theme = require('./utils/theme');
|
||||||
|
|
||||||
const Root = () => r(ReduxProvider, {
|
const Root = () => r(ReduxProvider, {
|
||||||
store: createStore(),
|
store: createStore(),
|
||||||
}, r(HotKeys, {
|
}, r(VolumePeaksProvider, {
|
||||||
|
}, r(VolumePeaksConsumer, {
|
||||||
|
}, peaks => r(HotKeys, {
|
||||||
}, ({
|
}, ({
|
||||||
graphRef,
|
graphRef,
|
||||||
cardsRef,
|
cardsRef,
|
||||||
|
@ -35,14 +38,14 @@ const Root = () => r(ReduxProvider, {
|
||||||
...modalsActions,
|
...modalsActions,
|
||||||
...hotKeysActions,
|
...hotKeysActions,
|
||||||
}, [
|
}, [
|
||||||
|
r(Graph, { ref: graphRef, peaks, ...modalsActions }),
|
||||||
r(TopLeftOnScreenButtonGroup, hotKeysActions),
|
r(TopLeftOnScreenButtonGroup, hotKeysActions),
|
||||||
r(Graph, { ref: graphRef, ...modalsActions }),
|
|
||||||
r(Cards, { ref: cardsRef }),
|
r(Cards, { ref: cardsRef }),
|
||||||
r(Network, { ref: networkRef, ...modalsActions }),
|
r(Network, { ref: networkRef, ...modalsActions }),
|
||||||
r(Preferences, { ref: preferencesRef }),
|
r(Preferences, { ref: preferencesRef }),
|
||||||
r(ServerInfo),
|
r(ServerInfo),
|
||||||
r(Log),
|
r(Log),
|
||||||
]))));
|
]))))));
|
||||||
|
|
||||||
Object.entries(theme.colors).forEach(([ key, value ]) => {
|
Object.entries(theme.colors).forEach(([ key, value ]) => {
|
||||||
document.body.style.setProperty('--' + key, value);
|
document.body.style.setProperty('--' + key, value);
|
||||||
|
|
62
yarn.lock
62
yarn.lock
|
@ -976,6 +976,11 @@ binary@^0.3.0:
|
||||||
buffers "~0.1.1"
|
buffers "~0.1.1"
|
||||||
chainsaw "~0.1.0"
|
chainsaw "~0.1.0"
|
||||||
|
|
||||||
|
bit-twiddle@^1.0.2:
|
||||||
|
version "1.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/bit-twiddle/-/bit-twiddle-1.0.2.tgz#0c6c1fabe2b23d17173d9a61b7b7093eb9e1769e"
|
||||||
|
integrity sha1-DGwfq+KyPRcXPZpht7cJPrnhdp4=
|
||||||
|
|
||||||
bluebird@^3.0.0:
|
bluebird@^3.0.0:
|
||||||
version "3.5.2"
|
version "3.5.2"
|
||||||
resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.5.2.tgz#1be0908e054a751754549c270489c1505d4ab15a"
|
resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.5.2.tgz#1be0908e054a751754549c270489c1505d4ab15a"
|
||||||
|
@ -2130,6 +2135,11 @@ duplexer3@^0.1.4:
|
||||||
resolved "https://registry.yarnpkg.com/duplexer3/-/duplexer3-0.1.4.tgz#ee01dd1cac0ed3cbc7fdbea37dc0a8f1ce002ce2"
|
resolved "https://registry.yarnpkg.com/duplexer3/-/duplexer3-0.1.4.tgz#ee01dd1cac0ed3cbc7fdbea37dc0a8f1ce002ce2"
|
||||||
integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=
|
integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI=
|
||||||
|
|
||||||
|
earcut@^2.1.3:
|
||||||
|
version "2.1.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/earcut/-/earcut-2.1.3.tgz#ca579545f351941af7c3d0df49c9f7d34af99b0c"
|
||||||
|
integrity sha512-AxdCdWUk1zzK/NuZ7e1ljj6IGC+VAdC3Qb7QQDsXpfNrc5IM8tL9nNXUmEGE6jRHTfZ10zhzRhtDmWVsR5pd3A==
|
||||||
|
|
||||||
ecc-jsbn@~0.1.1:
|
ecc-jsbn@~0.1.1:
|
||||||
version "0.1.2"
|
version "0.1.2"
|
||||||
resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9"
|
resolved "https://registry.yarnpkg.com/ecc-jsbn/-/ecc-jsbn-0.1.2.tgz#3a83a904e54353287874c564b7549386849a98c9"
|
||||||
|
@ -2568,6 +2578,11 @@ etag@~1.8.1:
|
||||||
resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
|
resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887"
|
||||||
integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=
|
integrity sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=
|
||||||
|
|
||||||
|
eventemitter3@^2.0.0:
|
||||||
|
version "2.0.3"
|
||||||
|
resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-2.0.3.tgz#b5e1079b59fb5e1ba2771c0a993be060a58c99ba"
|
||||||
|
integrity sha1-teEHm1n7XhuidxwKmTvgYKWMmbo=
|
||||||
|
|
||||||
execa@^0.7.0:
|
execa@^0.7.0:
|
||||||
version "0.7.0"
|
version "0.7.0"
|
||||||
resolved "https://registry.yarnpkg.com/execa/-/execa-0.7.0.tgz#944becd34cc41ee32a63a9faf27ad5a65fc59777"
|
resolved "https://registry.yarnpkg.com/execa/-/execa-0.7.0.tgz#944becd34cc41ee32a63a9faf27ad5a65fc59777"
|
||||||
|
@ -3975,6 +3990,11 @@ isexe@^2.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10"
|
resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10"
|
||||||
integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=
|
integrity sha1-6PvzdNxVb/iUehDcsFctYz8s+hA=
|
||||||
|
|
||||||
|
ismobilejs@^0.4.0:
|
||||||
|
version "0.4.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/ismobilejs/-/ismobilejs-0.4.1.tgz#1a5f126c70fed39c93da380fa62cbae5723e7dc2"
|
||||||
|
integrity sha1-Gl8SbHD+05yT2jgPpiy65XI+fcI=
|
||||||
|
|
||||||
isobject@^2.0.0:
|
isobject@^2.0.0:
|
||||||
version "2.1.0"
|
version "2.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/isobject/-/isobject-2.1.0.tgz#f065561096a3f1da2ef46272f815c840d87e0c89"
|
resolved "https://registry.yarnpkg.com/isobject/-/isobject-2.1.0.tgz#f065561096a3f1da2ef46272f815c840d87e0c89"
|
||||||
|
@ -4695,6 +4715,11 @@ mimic-fn@^1.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-1.2.0.tgz#820c86a39334640e99516928bd03fca88057d022"
|
resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-1.2.0.tgz#820c86a39334640e99516928bd03fca88057d022"
|
||||||
integrity sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==
|
integrity sha512-jf84uxzwiuiIVKiOLpfYk7N46TSy8ubTonmneY9vrpHNAnp0QBt2BxWV9dO3/j+BoVAb+a5G6YDPW3M5HOdMWQ==
|
||||||
|
|
||||||
|
mini-signals@^1.1.1:
|
||||||
|
version "1.2.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/mini-signals/-/mini-signals-1.2.0.tgz#45b08013c5fae51a24aa1a935cd317c9ed721d74"
|
||||||
|
integrity sha1-RbCAE8X65RokqhqTXNMXye1yHXQ=
|
||||||
|
|
||||||
"minimatch@2 || 3", minimatch@^3.0.0, minimatch@^3.0.3, minimatch@^3.0.4:
|
"minimatch@2 || 3", minimatch@^3.0.0, minimatch@^3.0.3, minimatch@^3.0.4:
|
||||||
version "3.0.4"
|
version "3.0.4"
|
||||||
resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083"
|
resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.0.4.tgz#5166e286457f03306064be5497e8dbb0c3d32083"
|
||||||
|
@ -5292,6 +5317,11 @@ parse-ms@^1.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/parse-ms/-/parse-ms-1.0.1.tgz#56346d4749d78f23430ca0c713850aef91aa361d"
|
resolved "https://registry.yarnpkg.com/parse-ms/-/parse-ms-1.0.1.tgz#56346d4749d78f23430ca0c713850aef91aa361d"
|
||||||
integrity sha1-VjRtR0nXjyNDDKDHE4UK75GqNh0=
|
integrity sha1-VjRtR0nXjyNDDKDHE4UK75GqNh0=
|
||||||
|
|
||||||
|
parse-uri@^1.0.0:
|
||||||
|
version "1.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/parse-uri/-/parse-uri-1.0.0.tgz#2872dcc22f1a797acde1583d8a0ac29552ddac20"
|
||||||
|
integrity sha1-KHLcwi8aeXrN4Vg9igrClVLdrCA=
|
||||||
|
|
||||||
parseurl@~1.3.2:
|
parseurl@~1.3.2:
|
||||||
version "1.3.2"
|
version "1.3.2"
|
||||||
resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.2.tgz#fc289d4ed8993119460c156253262cdc8de65bf3"
|
resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.2.tgz#fc289d4ed8993119460c156253262cdc8de65bf3"
|
||||||
|
@ -5411,6 +5441,25 @@ pinkie@^2.0.0:
|
||||||
resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870"
|
resolved "https://registry.yarnpkg.com/pinkie/-/pinkie-2.0.4.tgz#72556b80cfa0d48a974e80e77248e80ed4f7f870"
|
||||||
integrity sha1-clVrgM+g1IqXToDnckjoDtT3+HA=
|
integrity sha1-clVrgM+g1IqXToDnckjoDtT3+HA=
|
||||||
|
|
||||||
|
pixi-gl-core@^1.1.4:
|
||||||
|
version "1.1.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/pixi-gl-core/-/pixi-gl-core-1.1.4.tgz#8b4b5c433b31e419bc379dc565ce1b835a91b372"
|
||||||
|
integrity sha1-i0tcQzsx5Bm8N53FZc4bg1qRs3I=
|
||||||
|
|
||||||
|
pixi.js@^4.8.2:
|
||||||
|
version "4.8.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/pixi.js/-/pixi.js-4.8.2.tgz#c9e6f5f3b6780d2236705a7539e4e5ca3d74151f"
|
||||||
|
integrity sha512-OHA3Q3wwxRJXkVWALVuiUcUqQZd5p0rQF9ikCvOmux3A6Lxb5S61v4PMEAVgR3+1auZekbv/GNHCxDGFCQSi8g==
|
||||||
|
dependencies:
|
||||||
|
bit-twiddle "^1.0.2"
|
||||||
|
earcut "^2.1.3"
|
||||||
|
eventemitter3 "^2.0.0"
|
||||||
|
ismobilejs "^0.4.0"
|
||||||
|
object-assign "^4.0.1"
|
||||||
|
pixi-gl-core "^1.1.4"
|
||||||
|
remove-array-items "^1.0.0"
|
||||||
|
resource-loader "^2.1.1"
|
||||||
|
|
||||||
pkg-conf@^2.0.0, pkg-conf@^2.1.0:
|
pkg-conf@^2.0.0, pkg-conf@^2.1.0:
|
||||||
version "2.1.0"
|
version "2.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/pkg-conf/-/pkg-conf-2.1.0.tgz#2126514ca6f2abfebd168596df18ba57867f0058"
|
resolved "https://registry.yarnpkg.com/pkg-conf/-/pkg-conf-2.1.0.tgz#2126514ca6f2abfebd168596df18ba57867f0058"
|
||||||
|
@ -6107,6 +6156,11 @@ remotedev-utils@^0.1.1:
|
||||||
remotedev-serialize "^0.1.0"
|
remotedev-serialize "^0.1.0"
|
||||||
shortid "^2.2.6"
|
shortid "^2.2.6"
|
||||||
|
|
||||||
|
remove-array-items@^1.0.0:
|
||||||
|
version "1.1.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/remove-array-items/-/remove-array-items-1.1.0.tgz#e3d4aaba795a412479ea89408dee345387da7d3b"
|
||||||
|
integrity sha512-+YAHWd5patqAM/F4uBsto9h8RXDVxPRrKW46AkbI6eH12OFrN9wlGpkNWYxCjCfwtkidTjaaCXqU634V4mysvw==
|
||||||
|
|
||||||
remove-trailing-separator@^1.0.1:
|
remove-trailing-separator@^1.0.1:
|
||||||
version "1.1.0"
|
version "1.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz#c24bce2a283adad5bc3f58e0d48249b92379d8ef"
|
resolved "https://registry.yarnpkg.com/remove-trailing-separator/-/remove-trailing-separator-1.1.0.tgz#c24bce2a283adad5bc3f58e0d48249b92379d8ef"
|
||||||
|
@ -6217,6 +6271,14 @@ resolve@^1.1.6, resolve@^1.5.0, resolve@^1.6.0, resolve@^1.8.1:
|
||||||
dependencies:
|
dependencies:
|
||||||
path-parse "^1.0.5"
|
path-parse "^1.0.5"
|
||||||
|
|
||||||
|
resource-loader@^2.1.1:
|
||||||
|
version "2.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/resource-loader/-/resource-loader-2.1.1.tgz#f03ec08dd26aae0b0dd2a24a6d312aec2b5a004d"
|
||||||
|
integrity sha512-jRMGYUfa4AGk9ib45Wxc93lobhQVoiCUAUkWqsbb/fhGPge97YT1S8aC0xBEQpolMsrdmB3o7SH8VmIEvIDOLA==
|
||||||
|
dependencies:
|
||||||
|
mini-signals "^1.1.1"
|
||||||
|
parse-uri "^1.0.0"
|
||||||
|
|
||||||
restore-cursor@^1.0.1:
|
restore-cursor@^1.0.1:
|
||||||
version "1.0.1"
|
version "1.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-1.0.1.tgz#34661f46886327fed2991479152252df92daa541"
|
resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-1.0.1.tgz#34661f46886327fed2991479152252df92daa541"
|
||||||
|
|
Loading…
Reference in New Issue
Block a user