AbstractPlayer a playable synthesis process
superclass: AbstractFunction
Players are things that play. Anything that you want to play, you can stick your concept into a subclass of AbstractPlayer and you will inherit powerful server management and patching abilities.
A Player can be told to play and to stop, and will allocate and load any resources needed. Many players have subplayers. PlayerMixer mixes several players together. Patch can take other players as inputs. PlayerSeqTrack can sequence successions of players.
play(group,atTime,bus)
boots the server, prepares the player for play (loading samples etc.)
and spawns the player
all of these args are optional
group - a Server (will use the root node)
a Group
nil (default server's root node)
atTime - see atTime (Nil,Float,Integer or Date)
bus - (Bus,Integer,Nil)
a specific Bus to play on, otherwise will default to the main audio outs
spawn(atTime)
assumes the player has been prepared
stop(atTime)
stops playing but does not deallocate any resources (buffers etc.)
free
stops playind and frees all allocated resources (buffers etc.)
release(releaseTime,atTime)
call release on the synth, with the releaseTime
then calls stop on the player, so even if the synthDef does not have a \gate
input, it will function the same as stop after a short delay.
prepareForPlay(group,private,bus)
group - a Server, a Group or Nil
private - if true, allocates a private bus (not the main audio outs)
bus - a specific Bus to play on. this will cause private to be ignored
this loads the synthDef to the server for the player and all of its children,
and allocates any resources such as buffers, loading sound files etc.
Patching
temporarily disabled. players now patch using .ir outputs,
since its more appropriate to modulate the input than to modulate
the output.
if people really want this, let me know.
/*
p = Patch({ SinOsc.ar(mul: 0.3) });
p.play;
// reassign the bus while playing
p.bus = 5;
p.bus = 0;
p.bus = Bus(\audio,3);
if using an Integer, it will create a Bus on the same server.
*/
Common players
Patch
- specifies a function ( Instr ) and the arguments
with which to play that function.
SFP
- plays sound files
StreamKrDur Stream2Trig
- renders a number stream to a .kr signal
InstrSpawner InstrGateSpawner
Gui
AbstractPlayer also comes with a powerful gui class framework. Because of this, many people make the mistake of assuming that AbstractPlayer is primarily a way to get pretty windows. It has nothing to do with that.
But it is very nice to:
hit the play button and have your sound play.
select a format, choose a path by normal dialog and record your
sound to disk as a soundfile.
hit the save button and save your object with all its parameters to disk.
change the tempo when you want to.
Methods
path - if loaded or saved, the player knows its path
name - if loaded or saved, the filename, else the name of the class (eg. "a Patch")
children - for players that hold other players or objects, return those children
player classes should implement where appropriate.
a Patch has its args as children
SFP has its underlying object as child
using this.children.do({arg item; .... }) can save you from having to muck up
the Player class with extra methods.
allChildren - and all your children's children, and their children...
deepDo(function) - do to allChildren
playing in patterns:
Players can be put inside of standard patterns:
/*
(
p = CropPlayer(
Patch.new({ arg freq=400, freq2=500,pmindex=0,phasemod=0.0,amp=1.0;
PMOsc.ar(freq,freq2,pmindex,phasemod,amp)
},[
150.06,
27.7707,
15.1163,
0,
0.616317
]),0,4);
y = CropPlayer(
Patch.new({ arg freq=400, freq2=500,pmindex=0,phasemod=0.0,amp=1.0;
PMOsc.ar(freq,freq2,pmindex,phasemod,amp)
},[
200,
97.7707,
15.1163,
0,
0.616317
]),0,4);
Pseq([ p,y,p,y ],2).play
)
*/
Subclassing AbstractPlayer
see [playerServerSupport] for a basic rundown of the complexities involved
After that you would usually write a gui class for it.
Then implement the storeParamsOn method which enables you
to load and save your object.
Players to come
possible futures:
Stream2Midi - renders a number stream to midi ( controllers etc.)
EventStream2Midi - renders note/freq,amp/velocity, duration to midi note events
TimelineEventPlayer
timecode based events. (rather than delta based event streams where each event
is responsible for determing when the one succeeding it will happen).
Recording of events
from midi, wacom, gui etc.
any Player could be triggered to play from midi or wacom. it can also go through
SchedPlayer to quantise the start time of its play.
All of that can be recorded. those event-recordings can in turn then be further played with.
SFP is the one player type that can be sub-located (asked to start play somewhere in the middle). Timeline based event recording would be another one. the transport system would work very nicely.
AbstractPlayer is a subclass of AbstractFunction
therefore you can do math with them:
//right now this works, but only with simple ugenGraphs, no Samples,Envs etc.
(Patch({ Saw.ar(400) }).wrap2( 0.5) ).gui
// not yet
//(Patch({ Saw.ar(400) }).wrap2( KrNumberEditor(0.5,[0.0,1.0]) ) ).gui
//(somePatch * EnvPlayer.new(Env.newClear(10)) ).topGui