1 |
iliev |
2192 |
/* |
2 |
|
|
* JSampler - a java front-end for LinuxSampler |
3 |
|
|
* |
4 |
|
|
* Copyright (C) 2011 Grigor Iliev <grigor@grigoriliev.com> |
5 |
|
|
* |
6 |
|
|
* This file is part of JSampler. |
7 |
|
|
* |
8 |
|
|
* JSampler is free software; you can redistribute it and/or modify |
9 |
|
|
* it under the terms of the GNU General Public License version 2 |
10 |
|
|
* as published by the Free Software Foundation. |
11 |
|
|
* |
12 |
|
|
* JSampler is distributed in the hope that it will be useful, |
13 |
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
14 |
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
15 |
|
|
* GNU General Public License for more details. |
16 |
|
|
* |
17 |
|
|
* You should have received a copy of the GNU General Public License |
18 |
|
|
* along with JSampler; if not, write to the Free Software |
19 |
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, |
20 |
|
|
* MA 02111-1307 USA |
21 |
|
|
*/ |
22 |
|
|
package org.jsampler.view; |
23 |
|
|
|
24 |
|
|
import javax.swing.event.TreeModelEvent; |
25 |
|
|
import javax.swing.tree.TreePath; |
26 |
|
|
import java.util.ArrayList; |
27 |
|
|
import java.util.Enumeration; |
28 |
|
|
import java.util.Vector; |
29 |
|
|
import javax.swing.event.TreeModelListener; |
30 |
|
|
import javax.swing.tree.TreeModel; |
31 |
|
|
import javax.swing.tree.TreeNode; |
32 |
|
|
|
33 |
|
|
import org.jsampler.AudioDeviceModel; |
34 |
|
|
import org.jsampler.CC; |
35 |
|
|
import org.jsampler.EffectChain; |
36 |
|
|
import org.jsampler.event.AudioDeviceAdapter; |
37 |
|
|
import org.jsampler.event.AudioDeviceEvent; |
38 |
|
|
import org.jsampler.event.EffectChainEvent; |
39 |
|
|
import org.jsampler.event.EffectChainListener; |
40 |
|
|
import org.jsampler.event.ListEvent; |
41 |
|
|
import org.jsampler.event.ListListener; |
42 |
|
|
|
43 |
|
|
import org.linuxsampler.lscp.Effect; |
44 |
|
|
import org.linuxsampler.lscp.EffectInstance; |
45 |
|
|
|
46 |
|
|
import static org.jsampler.JSI18n.i18n; |
47 |
|
|
|
48 |
|
|
/** |
49 |
|
|
* |
50 |
|
|
* @author Grigor Iliev |
51 |
|
|
*/ |
52 |
|
|
public class SamplerTreeModel implements TreeModel { |
53 |
|
|
private final SamplerTreeNode root = new SamplerTreeNode(this); |
54 |
|
|
private ArrayList<TreeModelListener> listeners = new ArrayList<TreeModelListener>(); |
55 |
|
|
|
56 |
|
|
private final AudioDevicesTreeNode audioDevices; |
57 |
|
|
private final InternalEffectsTreeNode internalEffects; |
58 |
|
|
|
59 |
|
|
public |
60 |
|
|
SamplerTreeModel() { |
61 |
|
|
audioDevices = new AudioDevicesTreeNode(this, root); |
62 |
|
|
root.addChild(audioDevices); |
63 |
|
|
|
64 |
|
|
for(AudioDeviceModel a : CC.getSamplerModel().getAudioDevices()) { |
65 |
|
|
audioDevices.addChild(new AudioDeviceTreeNode(this, audioDevices, a)); |
66 |
|
|
a.addAudioDeviceListener(getHandler()); |
67 |
|
|
} |
68 |
|
|
|
69 |
|
|
CC.getSamplerModel().addAudioDeviceListListener(getHandler()); |
70 |
|
|
|
71 |
|
|
internalEffects = new InternalEffectsTreeNode(this, root); |
72 |
|
|
root.addChild(internalEffects); |
73 |
|
|
for(int i = 0; i < CC.getSamplerModel().getEffects().getEffectCount(); i++) { |
74 |
|
|
Effect fx = CC.getSamplerModel().getEffects().getEffect(i); |
75 |
|
|
internalEffects.addChild(new InternalEffectTreeNode(internalEffects, fx)); |
76 |
|
|
} |
77 |
|
|
} |
78 |
|
|
|
79 |
|
|
// Tree model methods |
80 |
|
|
@Override |
81 |
|
|
public void |
82 |
|
|
addTreeModelListener(TreeModelListener l) { |
83 |
|
|
listeners.add(l); |
84 |
|
|
} |
85 |
|
|
|
86 |
|
|
@Override |
87 |
|
|
public void |
88 |
|
|
removeTreeModelListener(TreeModelListener l) { |
89 |
|
|
listeners.remove(l); |
90 |
|
|
} |
91 |
|
|
|
92 |
|
|
@Override |
93 |
|
|
public Object |
94 |
|
|
getChild(Object parent, int index) { |
95 |
|
|
return ((TreeNode)parent).getChildAt(index); |
96 |
|
|
} |
97 |
|
|
|
98 |
|
|
@Override |
99 |
|
|
public int |
100 |
|
|
getChildCount(Object parent) { |
101 |
|
|
return ((TreeNode)parent).getChildCount(); |
102 |
|
|
} |
103 |
|
|
|
104 |
|
|
@Override |
105 |
|
|
public Object |
106 |
|
|
getRoot() { return root; } |
107 |
|
|
|
108 |
|
|
@Override |
109 |
|
|
public int |
110 |
|
|
getIndexOfChild(Object parent, Object child) { |
111 |
|
|
if(parent == null || child == null) return -1; |
112 |
|
|
return ((TreeNode)parent).getIndex((TreeNode)child); |
113 |
|
|
} |
114 |
|
|
|
115 |
|
|
@Override |
116 |
|
|
public boolean |
117 |
|
|
isLeaf(Object node) { return ((TreeNode)node).isLeaf(); } |
118 |
|
|
|
119 |
|
|
@Override |
120 |
|
|
public void |
121 |
|
|
valueForPathChanged(TreePath path, Object newValue) { |
122 |
|
|
|
123 |
|
|
} |
124 |
|
|
/////// |
125 |
|
|
|
126 |
|
|
protected Object[] |
127 |
|
|
getPathToRoot(TreeNode node) { |
128 |
|
|
Vector v = new Vector(); |
129 |
|
|
|
130 |
|
|
while(node != null) { |
131 |
|
|
v.insertElementAt(node, 0); |
132 |
|
|
if(node == getRoot()) break; |
133 |
|
|
node = node.getParent(); |
134 |
|
|
} |
135 |
|
|
|
136 |
|
|
return v.toArray(new Object[v.size()]); |
137 |
|
|
} |
138 |
|
|
|
139 |
|
|
private void |
140 |
|
|
fireNodeInserted(TreeNode node, int index) { |
141 |
|
|
Object[] path = getPathToRoot(node.getParent()); |
142 |
|
|
|
143 |
|
|
int[] idxs = { index }; |
144 |
|
|
Object[] objs = { node }; |
145 |
|
|
TreeModelEvent e = new TreeModelEvent(this, path, idxs, objs); |
146 |
|
|
for(TreeModelListener l : listeners) { |
147 |
|
|
l.treeNodesInserted(e); |
148 |
|
|
} |
149 |
|
|
} |
150 |
|
|
|
151 |
|
|
private void |
152 |
|
|
fireNodeChanged(TreeNode node, int index) { |
153 |
|
|
Object[] path = getPathToRoot(node.getParent()); |
154 |
|
|
int[] idxs = { index }; |
155 |
|
|
Object[] objs = { node }; |
156 |
|
|
TreeModelEvent e = new TreeModelEvent(this, path, idxs, objs); |
157 |
|
|
for(TreeModelListener l : listeners) { |
158 |
|
|
l.treeNodesChanged(e); |
159 |
|
|
} |
160 |
|
|
} |
161 |
|
|
|
162 |
|
|
private void |
163 |
|
|
fireNodeRemoved(TreeNode parent, TreeNode node, int index) { |
164 |
|
|
Object[] path = getPathToRoot(parent); |
165 |
|
|
int[] idxs = { index }; |
166 |
|
|
Object[] objs = { node }; |
167 |
|
|
TreeModelEvent e = new TreeModelEvent(this, path, idxs, objs); |
168 |
|
|
for(int i = listeners.size() - 1; i >=0; i--) { |
169 |
|
|
listeners.get(i).treeNodesRemoved(e); |
170 |
|
|
} |
171 |
|
|
} |
172 |
|
|
|
173 |
|
|
private void |
174 |
|
|
fireNodeStructureChanged(TreeNode node) { |
175 |
|
|
Object[] path = getPathToRoot(node); |
176 |
|
|
Object[] objs = { node }; |
177 |
|
|
TreeModelEvent e = new TreeModelEvent(this, path); |
178 |
|
|
for(TreeModelListener l : listeners) { |
179 |
|
|
l.treeStructureChanged(e); |
180 |
|
|
} |
181 |
|
|
} |
182 |
|
|
|
183 |
|
|
|
184 |
|
|
private final EventHandler eventHandler = new EventHandler(); |
185 |
|
|
|
186 |
|
|
private EventHandler |
187 |
|
|
getHandler() { return eventHandler; } |
188 |
|
|
|
189 |
|
|
private class EventHandler extends AudioDeviceAdapter implements ListListener<AudioDeviceModel> { |
190 |
|
|
|
191 |
|
|
|
192 |
|
|
/** Invoked when a new entry is added to a list. */ |
193 |
|
|
@Override |
194 |
|
|
public void |
195 |
|
|
entryAdded(final ListEvent<AudioDeviceModel> e) { |
196 |
|
|
e.getEntry().addAudioDeviceListener(getHandler()); |
197 |
|
|
AudioDeviceTreeNode node = |
198 |
|
|
new AudioDeviceTreeNode(SamplerTreeModel.this, audioDevices, e.getEntry()); |
199 |
|
|
audioDevices.addChild(node); |
200 |
|
|
fireNodeInserted(node, audioDevices.getIndex(node)); |
201 |
|
|
} |
202 |
|
|
|
203 |
|
|
/** Invoked when an entry is removed from a list. */ |
204 |
|
|
@Override |
205 |
|
|
public void |
206 |
|
|
entryRemoved(ListEvent<AudioDeviceModel> e) { |
207 |
|
|
TreeNode node = audioDevices.getChildById(e.getEntry().getDeviceId()); |
208 |
|
|
int i = audioDevices.getIndex(node); |
209 |
|
|
if(i == -1) return; |
210 |
|
|
|
211 |
|
|
audioDevices.removeChildAt(i); |
212 |
|
|
fireNodeRemoved(audioDevices, node, i); |
213 |
|
|
|
214 |
|
|
e.getEntry().removeAudioDeviceListener(getHandler()); |
215 |
|
|
} |
216 |
|
|
|
217 |
|
|
/** Invoked when a new send effect chain is added to the audio device. */ |
218 |
|
|
@Override |
219 |
|
|
public void |
220 |
|
|
sendEffectChainAdded(AudioDeviceEvent e) { |
221 |
|
|
AudioDeviceTreeNode node = |
222 |
|
|
audioDevices.getChildById(e.getAudioDeviceModel().getDeviceId()); |
223 |
|
|
|
224 |
|
|
if(node == null) { |
225 |
|
|
CC.getLogger().warning("Missing audio device node. This is a bug!"); |
226 |
|
|
return; |
227 |
|
|
} |
228 |
|
|
|
229 |
|
|
SendEffectChainTreeNode child = |
230 |
|
|
new SendEffectChainTreeNode(SamplerTreeModel.this, node, e.getEffectChain()); |
231 |
|
|
node.addChild(child); |
232 |
|
|
fireNodeInserted(child, node.getIndex(child)); |
233 |
|
|
} |
234 |
|
|
|
235 |
|
|
/** Invoked when when a send effect chain is removed from the audio device. */ |
236 |
|
|
@Override |
237 |
|
|
public void |
238 |
|
|
sendEffectChainRemoved(AudioDeviceEvent e) { |
239 |
|
|
AudioDeviceTreeNode node = |
240 |
|
|
audioDevices.getChildById(e.getAudioDeviceModel().getDeviceId()); |
241 |
|
|
|
242 |
|
|
if(node == null) { |
243 |
|
|
CC.getLogger().warning("Missing audio device node. This is a bug!"); |
244 |
|
|
return; |
245 |
|
|
} |
246 |
|
|
|
247 |
|
|
SendEffectChainTreeNode child = node.getChildById(e.getEffectChain().getChainId()); |
248 |
|
|
if(child == null) { |
249 |
|
|
CC.getLogger().warning("Missing send effect chain node. This is a bug!"); |
250 |
|
|
return; |
251 |
|
|
} |
252 |
|
|
|
253 |
|
|
int idx = node.getIndex(child); |
254 |
|
|
node.removeChildAt(idx); |
255 |
|
|
fireNodeRemoved(node, child, idx); |
256 |
|
|
} |
257 |
|
|
} |
258 |
|
|
|
259 |
|
|
public static class AbstractTreeNode<T extends TreeNode> implements TreeNode { |
260 |
|
|
protected final SamplerTreeModel treeModel; |
261 |
|
|
private final TreeNode parent; |
262 |
|
|
private final Vector<T> children = new Vector<T>(); |
263 |
|
|
|
264 |
|
|
public |
265 |
|
|
AbstractTreeNode(SamplerTreeModel treeModel) { this(treeModel, null); } |
266 |
|
|
|
267 |
|
|
public |
268 |
|
|
AbstractTreeNode(SamplerTreeModel treeModel, TreeNode parent) { |
269 |
|
|
this.parent = parent; |
270 |
|
|
this.treeModel = treeModel; |
271 |
|
|
} |
272 |
|
|
|
273 |
|
|
// Tree node model methods |
274 |
|
|
@Override |
275 |
|
|
public T |
276 |
|
|
getChildAt(int index) { return children.get(index); } |
277 |
|
|
|
278 |
|
|
@Override |
279 |
|
|
public int |
280 |
|
|
getChildCount() { return children.size(); } |
281 |
|
|
|
282 |
|
|
@Override |
283 |
|
|
public TreeNode |
284 |
|
|
getParent() { return parent; } |
285 |
|
|
|
286 |
|
|
@Override |
287 |
|
|
public int |
288 |
|
|
getIndex(TreeNode node) { return children.indexOf(node); } |
289 |
|
|
|
290 |
|
|
@Override |
291 |
|
|
public boolean |
292 |
|
|
getAllowsChildren() { return true; } |
293 |
|
|
|
294 |
|
|
@Override |
295 |
|
|
public boolean |
296 |
|
|
isLeaf() { return false; } |
297 |
|
|
|
298 |
|
|
@Override |
299 |
|
|
public Enumeration |
300 |
|
|
children() { return children.elements(); } |
301 |
|
|
/////// |
302 |
|
|
|
303 |
|
|
public void |
304 |
|
|
addChild(T child) { children.add(child); } |
305 |
|
|
|
306 |
|
|
public T |
307 |
|
|
removeChildAt(int index) { return children.remove(index); } |
308 |
|
|
|
309 |
|
|
public void |
310 |
|
|
removeAllChildren() { children.removeAllElements(); } |
311 |
|
|
} |
312 |
|
|
|
313 |
|
|
public static class AbstractTreeLeaf<T extends TreeNode> implements TreeNode { |
314 |
|
|
private final T parent; |
315 |
|
|
|
316 |
|
|
public |
317 |
|
|
AbstractTreeLeaf() { this(null); } |
318 |
|
|
|
319 |
|
|
public |
320 |
|
|
AbstractTreeLeaf(T parent) {this.parent = parent; } |
321 |
|
|
|
322 |
|
|
// Tree node model methods |
323 |
|
|
@Override |
324 |
|
|
public TreeNode |
325 |
|
|
getChildAt(int index) { return null; } |
326 |
|
|
|
327 |
|
|
@Override |
328 |
|
|
public int |
329 |
|
|
getChildCount() { return 0; } |
330 |
|
|
|
331 |
|
|
@Override |
332 |
|
|
public T |
333 |
|
|
getParent() { return parent; } |
334 |
|
|
|
335 |
|
|
@Override |
336 |
|
|
public int |
337 |
|
|
getIndex(TreeNode node) { return -1; } |
338 |
|
|
|
339 |
|
|
@Override |
340 |
|
|
public boolean |
341 |
|
|
getAllowsChildren() { return false; } |
342 |
|
|
|
343 |
|
|
@Override |
344 |
|
|
public boolean |
345 |
|
|
isLeaf() { return true; } |
346 |
|
|
|
347 |
|
|
@Override |
348 |
|
|
public Enumeration |
349 |
|
|
children() { return null; } |
350 |
|
|
/////// |
351 |
|
|
} |
352 |
|
|
|
353 |
|
|
public static class SamplerTreeNode extends AbstractTreeNode { |
354 |
|
|
public |
355 |
|
|
SamplerTreeNode(SamplerTreeModel treeModel) { |
356 |
|
|
super(treeModel); |
357 |
|
|
} |
358 |
|
|
|
359 |
|
|
@Override |
360 |
|
|
public String |
361 |
|
|
toString() { return i18n.getLabel("SamplerTreeNode.toString"); } |
362 |
|
|
} |
363 |
|
|
|
364 |
|
|
public static class AudioDevicesTreeNode extends AbstractTreeNode<AudioDeviceTreeNode> { |
365 |
|
|
public |
366 |
|
|
AudioDevicesTreeNode(SamplerTreeModel treeModel, TreeNode parent) { |
367 |
|
|
super(treeModel, parent); |
368 |
|
|
} |
369 |
|
|
|
370 |
|
|
public AudioDeviceTreeNode |
371 |
|
|
getChildById(int audioDeviceId) { |
372 |
|
|
for(int i = 0; i < getChildCount(); i++) { |
373 |
|
|
if(getChildAt(i).getAudioDeviceId() == audioDeviceId) return getChildAt(i); |
374 |
|
|
} |
375 |
|
|
|
376 |
|
|
return null; |
377 |
|
|
} |
378 |
|
|
|
379 |
|
|
@Override |
380 |
|
|
public String |
381 |
|
|
toString() { return i18n.getLabel("AudioDevicesTreeNode.toString"); } |
382 |
|
|
} |
383 |
|
|
|
384 |
|
|
public static class AudioDeviceTreeNode extends AbstractTreeNode<SendEffectChainTreeNode> { |
385 |
|
|
private final AudioDeviceModel audioDevice; |
386 |
|
|
|
387 |
|
|
public |
388 |
|
|
AudioDeviceTreeNode ( |
389 |
|
|
SamplerTreeModel treeModel, TreeNode parent, AudioDeviceModel audioDevice |
390 |
|
|
) { |
391 |
|
|
super(treeModel, parent); |
392 |
|
|
this.audioDevice = audioDevice; |
393 |
|
|
|
394 |
|
|
for(int i = 0; i < audioDevice.getSendEffectChainCount(); i++) { |
395 |
|
|
EffectChain chain = audioDevice.getSendEffectChain(i); |
396 |
|
|
addChild(new SendEffectChainTreeNode(treeModel, this, chain)); |
397 |
|
|
} |
398 |
|
|
} |
399 |
|
|
|
400 |
|
|
public AudioDeviceModel |
401 |
|
|
getAudioDevice() { return audioDevice; } |
402 |
|
|
|
403 |
|
|
public int |
404 |
|
|
getAudioDeviceId() { return audioDevice.getDeviceId(); } |
405 |
|
|
|
406 |
|
|
public SendEffectChainTreeNode |
407 |
|
|
getChildById(int chainId) { |
408 |
|
|
for(int i = 0; i < getChildCount(); i++) { |
409 |
|
|
if(getChildAt(i).getChainId() == chainId) return getChildAt(i); |
410 |
|
|
} |
411 |
|
|
|
412 |
|
|
return null; |
413 |
|
|
} |
414 |
|
|
|
415 |
|
|
@Override |
416 |
|
|
public String |
417 |
|
|
toString() { return i18n.getLabel("AudioDeviceTreeNode.toString", getAudioDeviceId()); } |
418 |
|
|
} |
419 |
|
|
|
420 |
|
|
public static class SendEffectChainTreeNode extends AbstractTreeNode<EffectInstanceTreeNode> |
421 |
|
|
implements EffectChainListener { |
422 |
|
|
private final EffectChain chain; |
423 |
|
|
|
424 |
|
|
public |
425 |
|
|
SendEffectChainTreeNode ( |
426 |
|
|
SamplerTreeModel treeModel, AudioDeviceTreeNode parent, EffectChain chain |
427 |
|
|
) { |
428 |
|
|
super(treeModel, parent); |
429 |
|
|
this.chain = chain; |
430 |
|
|
chain.addAudioDeviceListener(this); |
431 |
|
|
updateEffectInstanceList(); |
432 |
|
|
} |
433 |
|
|
/////// |
434 |
|
|
|
435 |
|
|
public EffectChain |
436 |
|
|
getEffectChain() { return chain; } |
437 |
|
|
|
438 |
|
|
public int |
439 |
|
|
getChainId() { return chain.getChainId(); } |
440 |
|
|
|
441 |
|
|
/** |
442 |
|
|
* Gets the audio device to which the |
443 |
|
|
* send effect chain represented by this tree node belongs. |
444 |
|
|
*/ |
445 |
|
|
public AudioDeviceModel |
446 |
|
|
getAudioDevice() { return ((AudioDeviceTreeNode)getParent()).getAudioDevice(); } |
447 |
|
|
|
448 |
|
|
@Override |
449 |
|
|
public String |
450 |
|
|
toString() { return i18n.getLabel("SendEffectChainTreeNode.toString", chain.getChainId()); } |
451 |
|
|
|
452 |
|
|
@Override |
453 |
|
|
public void |
454 |
|
|
effectInstanceListChanged(EffectChainEvent e) { |
455 |
|
|
updateEffectInstanceList(); |
456 |
|
|
treeModel.fireNodeStructureChanged(this); |
457 |
|
|
} |
458 |
|
|
|
459 |
|
|
private void |
460 |
|
|
updateEffectInstanceList() { |
461 |
|
|
removeAllChildren(); |
462 |
|
|
for(int i = 0; i < chain.getEffectInstanceCount(); i++) { |
463 |
|
|
EffectInstance ei = chain.getEffectInstance(i); |
464 |
|
|
addChild(new EffectInstanceTreeNode(this, ei)); |
465 |
|
|
} |
466 |
|
|
|
467 |
|
|
|
468 |
|
|
} |
469 |
|
|
} |
470 |
|
|
|
471 |
|
|
public static class EffectInstanceTreeNode extends AbstractTreeLeaf<SendEffectChainTreeNode> { |
472 |
|
|
private final EffectInstance effectInstance; |
473 |
|
|
|
474 |
|
|
public |
475 |
|
|
EffectInstanceTreeNode(SendEffectChainTreeNode parent, EffectInstance ei) { |
476 |
|
|
super(parent); |
477 |
|
|
effectInstance = ei; |
478 |
|
|
} |
479 |
|
|
|
480 |
|
|
public int |
481 |
|
|
getInstanceId() { return effectInstance.getInstanceId(); } |
482 |
|
|
|
483 |
|
|
@Override |
484 |
|
|
public String |
485 |
|
|
toString() { return effectInstance.getDescription(); } |
486 |
|
|
} |
487 |
|
|
|
488 |
|
|
public static class InternalEffectsTreeNode extends AbstractTreeNode<InternalEffectTreeNode> { |
489 |
|
|
public |
490 |
|
|
InternalEffectsTreeNode(SamplerTreeModel treeModel, TreeNode parent) { |
491 |
|
|
super(treeModel, parent); |
492 |
|
|
} |
493 |
|
|
|
494 |
|
|
@Override |
495 |
|
|
public String |
496 |
|
|
toString() { return i18n.getLabel("InternalEffectsTreeNode.toString"); } |
497 |
|
|
} |
498 |
|
|
|
499 |
|
|
public static class InternalEffectTreeNode extends AbstractTreeLeaf { |
500 |
|
|
private final Effect effect; |
501 |
|
|
|
502 |
|
|
public |
503 |
|
|
InternalEffectTreeNode(TreeNode parent, Effect effect) { |
504 |
|
|
super(parent); |
505 |
|
|
this.effect = effect; |
506 |
|
|
} |
507 |
|
|
|
508 |
|
|
@Override |
509 |
|
|
public String |
510 |
|
|
toString() { return effect.getDescription(); } |
511 |
|
|
} |
512 |
|
|
|
513 |
|
|
} |