niki 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- data/.gitignore +3 -0
- data/Gemfile +7 -0
- data/Gemfile.lock +22 -0
- data/Rakefile +2 -0
- data/Readme.md +55 -0
- data/examples/my_song.rb +267 -0
- data/examples/my_song.rns +0 -0
- data/lib/core_ext/array.rb +13 -0
- data/lib/core_ext/fixnum.rb +8 -0
- data/lib/niki.rb +9 -0
- data/lib/niki/chords.rb +44 -0
- data/lib/niki/part.rb +69 -0
- data/lib/niki/song.rb +81 -0
- data/lib/niki/version.rb +3 -0
- data/niki.gemspec +23 -0
- data/vendor/archaeopteryx/archaeopteryx.rb +28 -0
- data/vendor/archaeopteryx/clip.rb +37 -0
- data/vendor/archaeopteryx/core_ext/array.rb +11 -0
- data/vendor/archaeopteryx/core_ext/struct.rb +7 -0
- data/vendor/archaeopteryx/drum.rb +24 -0
- data/vendor/archaeopteryx/live_hacks.rb +8 -0
- data/vendor/archaeopteryx/loop.rb +32 -0
- data/vendor/archaeopteryx/midi/clock.rb +23 -0
- data/vendor/archaeopteryx/midi/file_output/file_midi.rb +44 -0
- data/vendor/archaeopteryx/midi/note.rb +9 -0
- data/vendor/archaeopteryx/midi/practical_ruby_projects/core_foundation.rb +7 -0
- data/vendor/archaeopteryx/midi/practical_ruby_projects/core_midi.rb +16 -0
- data/vendor/archaeopteryx/midi/practical_ruby_projects/live_midi.rb +117 -0
- data/vendor/archaeopteryx/midi/practical_ruby_projects/no_midi_destinations.rb +8 -0
- data/vendor/archaeopteryx/midi/practical_ruby_projects/practical_ruby_projects.rb +30 -0
- data/vendor/archaeopteryx/midi/practical_ruby_projects/timer.rb +35 -0
- data/vendor/archaeopteryx/mix.rb +27 -0
- data/vendor/archaeopteryx/rhythm.rb +29 -0
- data/vendor/archaeopteryx/track.rb +16 -0
- metadata +113 -0
data/.gitignore
ADDED
data/Gemfile
ADDED
data/Gemfile.lock
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
PATH
|
2
|
+
remote: .
|
3
|
+
specs:
|
4
|
+
niki (0.0.1)
|
5
|
+
midilib
|
6
|
+
|
7
|
+
GEM
|
8
|
+
remote: http://rubygems.org/
|
9
|
+
specs:
|
10
|
+
ffi (1.0.9)
|
11
|
+
ffi-coremidi (0.0.8)
|
12
|
+
ffi (>= 1.0)
|
13
|
+
midilib (2.0.0)
|
14
|
+
unimidi (0.2.1)
|
15
|
+
ffi-coremidi
|
16
|
+
|
17
|
+
PLATFORMS
|
18
|
+
ruby
|
19
|
+
|
20
|
+
DEPENDENCIES
|
21
|
+
niki!
|
22
|
+
unimidi
|
data/Rakefile
ADDED
data/Readme.md
ADDED
@@ -0,0 +1,55 @@
|
|
1
|
+
# Niki
|
2
|
+
|
3
|
+
Niki is a Ruby DSL to describe and play musical pieces.
|
4
|
+
|
5
|
+
It leverages Giles Bowkett's [Archaeopteryx](
|
6
|
+
https://github.com/gilesbowkett/archaeopteryx) to send MIDI output to other
|
7
|
+
programs accepting MIDI inputs (Ableton Live, Reason, Garage Band...), which
|
8
|
+
will actually play the song.
|
9
|
+
|
10
|
+
To see what the DSL looks like, take a look at the [example song](https://github.com/txus/niki/blob/master/examples/my_song.rb)
|
11
|
+
|
12
|
+
https://github.com/gilesbowkett/archaeopteryx/blob/master/db_drum_definition.rb
|
13
|
+
|
14
|
+
## Caveats
|
15
|
+
|
16
|
+
* It runs only in Ruby 1.8.7 (due to the dependency on Archaeopteryx) :(
|
17
|
+
* I've tested this only in Snow Leopard, but it should run in more platforms.
|
18
|
+
If your platform is not supported it's because Archaeopteryx
|
19
|
+
doesn't support it yet.
|
20
|
+
|
21
|
+
## Run the example song
|
22
|
+
|
23
|
+
First of all, get the example song files (both the Reason file and the Ruby file)
|
24
|
+
|
25
|
+
$ curl -o my_song.rns https://raw.github.com/txus/niki/master/examples/my_song.rns
|
26
|
+
$ curl -o my_song.rb https://raw.github.com/txus/niki/master/examples/my_song.rb
|
27
|
+
|
28
|
+
Now [download a demo version](
|
29
|
+
http://www.propellerheads.se/download/index.cfm?fuseaction=get_article&article=reason) of Propellerhead's
|
30
|
+
Reason 5 and open up `my_song.rns`.
|
31
|
+
|
32
|
+
### Configuring the midi interface
|
33
|
+
|
34
|
+
This is how you configure the midi interface in Mac OSX (it should not be
|
35
|
+
difficult in other operating systems):
|
36
|
+
|
37
|
+
* Make sure the IAC driver is activated in Mac OSX's Audio/MIDI Setup (inside the
|
38
|
+
MIDI section) (activate the "Device is online" checkbox if it is not)
|
39
|
+
* In this section, make sure at least one port is created (under the Ports
|
40
|
+
section). If not, add one clicking the + button.
|
41
|
+
* Go to the Reason options panel, go to Advanced, and set your first
|
42
|
+
MIDI bus (Bus A) to "IAC Driver IAC Bus 1" (the port you just created).
|
43
|
+
|
44
|
+
### Install the gem and run!
|
45
|
+
|
46
|
+
Now you should install the gem and run the example!
|
47
|
+
|
48
|
+
$ gem install niki
|
49
|
+
$ ruby my_song.rb
|
50
|
+
|
51
|
+
# TODO
|
52
|
+
|
53
|
+
* Tests + refactor
|
54
|
+
* Documentation
|
55
|
+
* Expand this README
|
data/examples/my_song.rb
ADDED
@@ -0,0 +1,267 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'niki'
|
3
|
+
|
4
|
+
song = Niki::Song.new :tempo => 127 do
|
5
|
+
|
6
|
+
# Configure midi channels
|
7
|
+
channel :basses, 1
|
8
|
+
channel :drums, 10
|
9
|
+
channel :chords, 2
|
10
|
+
channel :melodies, 3
|
11
|
+
|
12
|
+
configure :drums do |drums|
|
13
|
+
drums[:kick] = c1
|
14
|
+
drums[:snare] = c1.sharp
|
15
|
+
drums[:hh] = g1
|
16
|
+
drums[:ohh] = g1.sharp
|
17
|
+
end
|
18
|
+
|
19
|
+
part :intro do
|
20
|
+
# Drums
|
21
|
+
2.times do
|
22
|
+
drum [:kick, :hh], 8
|
23
|
+
drum :hh, 8
|
24
|
+
drum [:snare, :hh], 8
|
25
|
+
drum :hh, 8
|
26
|
+
|
27
|
+
drum [:kick, :hh], 8
|
28
|
+
drum :hh, 8
|
29
|
+
drum [:snare, :hh], 8
|
30
|
+
drum [:kick, :hh], 8
|
31
|
+
|
32
|
+
drum :hh, 16
|
33
|
+
drum :hh, 16
|
34
|
+
drum :hh, 16
|
35
|
+
drum :hh, 16
|
36
|
+
drum [:snare, :hh], 8
|
37
|
+
drum :hh, 8
|
38
|
+
|
39
|
+
drum [:kick, :hh], 8
|
40
|
+
drum [:kick, :hh], 8
|
41
|
+
drum [:snare, :hh], 8
|
42
|
+
drum :ohh, 8
|
43
|
+
end
|
44
|
+
|
45
|
+
# Bass
|
46
|
+
bass f2, 4
|
47
|
+
|
48
|
+
bass f2, 8
|
49
|
+
bass e2, 8
|
50
|
+
bass f2, 8
|
51
|
+
bass e2, 8
|
52
|
+
bass f2, 8
|
53
|
+
|
54
|
+
bass e3, 4
|
55
|
+
bass c3, 4
|
56
|
+
bass b2, 8
|
57
|
+
bass c2, 8
|
58
|
+
bass c2, 8
|
59
|
+
bass g2, 4
|
60
|
+
|
61
|
+
bass a2, 4
|
62
|
+
|
63
|
+
bass a2, 8
|
64
|
+
bass g2, 8
|
65
|
+
bass a2, 8
|
66
|
+
bass g2, 8
|
67
|
+
bass a2, 8
|
68
|
+
|
69
|
+
bass e3, 4
|
70
|
+
bass c3, 4
|
71
|
+
bass b2, 8
|
72
|
+
bass c2, 8
|
73
|
+
bass c2, 8
|
74
|
+
bass g2, 4
|
75
|
+
end
|
76
|
+
|
77
|
+
part :intro_ending do
|
78
|
+
# Drums
|
79
|
+
drum [:kick, :hh], 8
|
80
|
+
drum :hh, 8
|
81
|
+
drum [:snare, :hh], 8
|
82
|
+
drum :hh, 8
|
83
|
+
|
84
|
+
drum [:kick, :hh], 8
|
85
|
+
drum :hh, 8
|
86
|
+
drum [:snare, :hh], 8
|
87
|
+
drum [:kick, :hh], 8
|
88
|
+
|
89
|
+
drum :hh, 16
|
90
|
+
drum :hh, 16
|
91
|
+
drum :hh, 16
|
92
|
+
drum :hh, 16
|
93
|
+
drum [:snare, :hh], 8
|
94
|
+
drum :hh, 8
|
95
|
+
|
96
|
+
drum [:kick, :hh], 8
|
97
|
+
drum [:kick, :hh], 8
|
98
|
+
drum [:snare, :hh], 8
|
99
|
+
drum :ohh, 8
|
100
|
+
|
101
|
+
# Drum break
|
102
|
+
drum [:kick, :hh], 8
|
103
|
+
drum :hh, 8
|
104
|
+
drum [:kick, :hh], 8
|
105
|
+
drum :hh, 8
|
106
|
+
drum [:kick, :hh], 8
|
107
|
+
drum :hh, 8
|
108
|
+
drum [:kick, :hh], 8
|
109
|
+
drum :hh, 8
|
110
|
+
|
111
|
+
drum [:snare, :ooh], 4
|
112
|
+
drum [:snare, :ooh], 4
|
113
|
+
drum :snare, 16
|
114
|
+
drum :snare, 16
|
115
|
+
drum :kick, 16
|
116
|
+
drum :kick, 16
|
117
|
+
drum :snare, 16
|
118
|
+
drum :snare, 16
|
119
|
+
drum [:snare, :ooh], 16
|
120
|
+
drum :snare, 16
|
121
|
+
|
122
|
+
# Bass
|
123
|
+
bass f2, 4
|
124
|
+
|
125
|
+
bass f2, 8
|
126
|
+
bass e2, 8
|
127
|
+
bass f2, 8
|
128
|
+
bass e2, 8
|
129
|
+
bass f2, 8
|
130
|
+
|
131
|
+
bass e3, 4
|
132
|
+
bass c3, 4
|
133
|
+
bass b2, 8
|
134
|
+
bass c2, 8
|
135
|
+
bass c2, 8
|
136
|
+
bass g2, 4
|
137
|
+
|
138
|
+
bass a2, 4
|
139
|
+
bass a2, 4
|
140
|
+
bass a2, 4
|
141
|
+
bass a2, 4
|
142
|
+
|
143
|
+
bass b2, 8
|
144
|
+
bass b2, 8
|
145
|
+
bass b2, 8
|
146
|
+
bass c2, 8
|
147
|
+
bass c2, 8
|
148
|
+
bass c2, 8
|
149
|
+
bass d3, 8
|
150
|
+
bass e3, 8
|
151
|
+
end
|
152
|
+
|
153
|
+
part :pre_chorus do
|
154
|
+
bass :from => :intro
|
155
|
+
drum :from => :intro
|
156
|
+
|
157
|
+
melody a2, 4
|
158
|
+
melody a2, 4
|
159
|
+
melody f3, 8
|
160
|
+
melody g3, 8
|
161
|
+
melody a3, 8
|
162
|
+
melody g3, 4
|
163
|
+
melody f3, 8
|
164
|
+
melody e3, 8
|
165
|
+
melody d3, 4
|
166
|
+
melody c3, 8
|
167
|
+
melody e3, 4
|
168
|
+
|
169
|
+
melody a2, 2
|
170
|
+
melody a2, 8
|
171
|
+
melody g2, 4
|
172
|
+
melody c2, 4
|
173
|
+
|
174
|
+
melody c3, 8
|
175
|
+
melody d3, 8
|
176
|
+
melody e3, 8
|
177
|
+
melody f3, 8
|
178
|
+
melody e3, 8
|
179
|
+
melody d3, 8
|
180
|
+
melody c3, 8
|
181
|
+
end
|
182
|
+
|
183
|
+
repeat :pre_chorus
|
184
|
+
|
185
|
+
part :chorus do
|
186
|
+
melody :from => :pre_chorus
|
187
|
+
|
188
|
+
# Bass
|
189
|
+
2.times do
|
190
|
+
bass f2, 8
|
191
|
+
bass f3, 8
|
192
|
+
bass f2, 8
|
193
|
+
bass f3, 8
|
194
|
+
end
|
195
|
+
bass c2, 8
|
196
|
+
bass c3, 8
|
197
|
+
bass c2, 8
|
198
|
+
bass c3, 8
|
199
|
+
|
200
|
+
bass g2, 8
|
201
|
+
bass g3, 8
|
202
|
+
bass f2, 8
|
203
|
+
bass d2, 8
|
204
|
+
|
205
|
+
2.times do
|
206
|
+
bass a1, 8
|
207
|
+
bass a2, 8
|
208
|
+
bass a1, 8
|
209
|
+
bass a2, 8
|
210
|
+
end
|
211
|
+
bass c1, 8
|
212
|
+
bass c2, 8
|
213
|
+
bass c1, 8
|
214
|
+
bass c2, 8
|
215
|
+
|
216
|
+
bass g1, 8
|
217
|
+
bass g2, 8
|
218
|
+
bass f1, 8
|
219
|
+
bass d2, 8
|
220
|
+
|
221
|
+
# Drums
|
222
|
+
7.times do
|
223
|
+
drum [:kick, :hh], 8
|
224
|
+
drum :hh, 8
|
225
|
+
drum [:snare, :kick, :hh], 8
|
226
|
+
drum :hh, 8
|
227
|
+
end
|
228
|
+
drum :kick, 8
|
229
|
+
drum :hh, 16
|
230
|
+
drum :hh, 16
|
231
|
+
drum [:snare, :kick], 16
|
232
|
+
drum :snare, 16
|
233
|
+
drum [:kick, :ohh], 8
|
234
|
+
|
235
|
+
# Chords
|
236
|
+
chord a3MIN(2), 2, :base => f3
|
237
|
+
chord a3MIN(2), 8, :base => f3
|
238
|
+
chord a3MIN(2), 8, :base => f3
|
239
|
+
chord a3MIN(2), 8, :base => f3
|
240
|
+
chord c3MAJ, 4
|
241
|
+
chord c3MAJ, 4
|
242
|
+
chord c3MAJ, 8
|
243
|
+
chord g3MAJ, 2
|
244
|
+
chord [a3, b3, e4]
|
245
|
+
chord c3MAJ, 4
|
246
|
+
chord c3MAJ, 4
|
247
|
+
chord g3MAJ, 8
|
248
|
+
chord g3MAJ, 8
|
249
|
+
chord g3MAJ, 8, :base => f4
|
250
|
+
chord g3MAJ, 8, :base => e4
|
251
|
+
end
|
252
|
+
|
253
|
+
repeat :chorus
|
254
|
+
|
255
|
+
part :outro do
|
256
|
+
drum :from => :intro
|
257
|
+
bass :from => :intro
|
258
|
+
chord :from => :chorus
|
259
|
+
end
|
260
|
+
|
261
|
+
part :outro_ending do
|
262
|
+
bass :from => :intro
|
263
|
+
chord :from => :chorus
|
264
|
+
end
|
265
|
+
end
|
266
|
+
|
267
|
+
song.play
|
Binary file
|
data/lib/niki.rb
ADDED
data/lib/niki/chords.rb
ADDED
@@ -0,0 +1,44 @@
|
|
1
|
+
module Niki
|
2
|
+
module Chords
|
3
|
+
|
4
|
+
(1..5).to_a.each do |octave|
|
5
|
+
|
6
|
+
%w(c d e f g a b).each_with_index do |c, idx|
|
7
|
+
rest = 0
|
8
|
+
if idx > 2
|
9
|
+
rest = 1
|
10
|
+
end
|
11
|
+
base = 24 + (12 * octave) + (idx * 2) - rest
|
12
|
+
|
13
|
+
define_method("#{c}#{octave}") do
|
14
|
+
base
|
15
|
+
end
|
16
|
+
|
17
|
+
define_method("#{c}#{octave}MAJ") do |*args|
|
18
|
+
inversion = args.first || 1
|
19
|
+
case inversion
|
20
|
+
when 1
|
21
|
+
[base, base + 4, base + 7]
|
22
|
+
when 2
|
23
|
+
[base + 4, base + 7, base + 12]
|
24
|
+
when 3
|
25
|
+
[base + 7, base + 12, base + 16]
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
define_method("#{c}#{octave}MIN") do |*args|
|
30
|
+
inversion = args.first || 1
|
31
|
+
case inversion
|
32
|
+
when 1
|
33
|
+
[base, base + 3, base + 7]
|
34
|
+
when 2
|
35
|
+
[base + 3, base + 7, base + 12]
|
36
|
+
when 3
|
37
|
+
[base + 7, base + 12, base + 15]
|
38
|
+
end
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
data/lib/niki/part.rb
ADDED
@@ -0,0 +1,69 @@
|
|
1
|
+
module Niki
|
2
|
+
class Part
|
3
|
+
attr_reader :name, :chords, :drums, :basses, :melodies
|
4
|
+
|
5
|
+
include Niki::Chords
|
6
|
+
|
7
|
+
def initialize(name, song, &block)
|
8
|
+
@name = name
|
9
|
+
@song = song
|
10
|
+
@drum_notes = @song.drum_notes
|
11
|
+
@chords = []
|
12
|
+
@drums = []
|
13
|
+
@basses = []
|
14
|
+
@melodies = []
|
15
|
+
self.instance_eval(&block)
|
16
|
+
end
|
17
|
+
|
18
|
+
def chord note, duration = 1, options = {}
|
19
|
+
if note.is_a?(Hash) && note[:from]
|
20
|
+
copy_from_part(note[:from], :chords)
|
21
|
+
return
|
22
|
+
end
|
23
|
+
|
24
|
+
note.push options[:base]
|
25
|
+
@chords << [note, calculate_duration(duration)]
|
26
|
+
end
|
27
|
+
|
28
|
+
def drum note, duration = 1, options = {}
|
29
|
+
if note.is_a?(Hash) && note[:from]
|
30
|
+
copy_from_part(note[:from], :drums)
|
31
|
+
return
|
32
|
+
end
|
33
|
+
|
34
|
+
drum_note = [note].flatten.map do |n|
|
35
|
+
@drum_notes[n]
|
36
|
+
end
|
37
|
+
@drums << [drum_note, calculate_duration(duration)]
|
38
|
+
end
|
39
|
+
|
40
|
+
def bass note = nil, duration = 1, options = {}
|
41
|
+
if note.is_a?(Hash) && note[:from]
|
42
|
+
copy_from_part(note[:from], :basses)
|
43
|
+
return
|
44
|
+
end
|
45
|
+
|
46
|
+
@basses << [[note], calculate_duration(duration)]
|
47
|
+
end
|
48
|
+
|
49
|
+
def melody note = nil, duration = 1, options = {}
|
50
|
+
if note.is_a?(Hash) && note[:from]
|
51
|
+
copy_from_part(note[:from], :melodies)
|
52
|
+
return
|
53
|
+
end
|
54
|
+
|
55
|
+
@melodies << [[note], calculate_duration(duration)]
|
56
|
+
end
|
57
|
+
|
58
|
+
private
|
59
|
+
|
60
|
+
def calculate_duration(duration)
|
61
|
+
1.to_f / ((@song.tempo.to_f/120) * duration/2)
|
62
|
+
end
|
63
|
+
|
64
|
+
def copy_from_part(name, type)
|
65
|
+
part = @song.get_part(name)
|
66
|
+
instance_variable_set(:"@#{type}", part.send(type))
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
data/lib/niki/song.rb
ADDED
@@ -0,0 +1,81 @@
|
|
1
|
+
module Niki
|
2
|
+
class Song
|
3
|
+
attr_reader :tempo, :drum_notes
|
4
|
+
|
5
|
+
include Niki::Chords
|
6
|
+
|
7
|
+
def initialize(options, &block)
|
8
|
+
@clock = Archaeopteryx::Midi::Clock.new(options[:tempo] || 127)
|
9
|
+
|
10
|
+
@midi = Archaeopteryx::Midi::PracticalRubyProjects::LiveMIDI.new(:clock => @clock )
|
11
|
+
|
12
|
+
@parts = []
|
13
|
+
@channel = {}
|
14
|
+
@drum_notes = {}
|
15
|
+
@tempo = options[:tempo]
|
16
|
+
self.instance_eval &block
|
17
|
+
end
|
18
|
+
|
19
|
+
def channel(instrument, number)
|
20
|
+
@channel[instrument] = number - 1
|
21
|
+
end
|
22
|
+
|
23
|
+
def part(name, &block)
|
24
|
+
raise "#{name} is already defined!" if has_part?(name)
|
25
|
+
@parts << Part.new(name, self, &block)
|
26
|
+
end
|
27
|
+
|
28
|
+
def configure(instrument, &block)
|
29
|
+
case instrument
|
30
|
+
when :drums
|
31
|
+
block.call @drum_notes
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def repeat(name, options = {})
|
36
|
+
(options[:times] || 1).times do
|
37
|
+
@parts << get_part(name)
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def play
|
42
|
+
@parts.each do |part|
|
43
|
+
[:basses, :chords, :melodies, :drums].map do |instrument_name|
|
44
|
+
Thread.start do
|
45
|
+
play_part(part, instrument_name)
|
46
|
+
end
|
47
|
+
end.map { |thread| thread.join }
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
def play_part(part, instrument_name)
|
52
|
+
part.send(instrument_name).each do |instrument|
|
53
|
+
notes = [instrument.first].flatten.compact.map do |note|
|
54
|
+
Archaeopteryx::Midi::Note.create(
|
55
|
+
:channel => @channel[instrument_name],
|
56
|
+
:number => note,
|
57
|
+
:duration => instrument.last,
|
58
|
+
:velocity => 127)
|
59
|
+
end
|
60
|
+
notes.each do |note|
|
61
|
+
@midi.note_on(note)
|
62
|
+
end
|
63
|
+
sleep(instrument.last)
|
64
|
+
notes.each do |note|
|
65
|
+
@midi.note_off(note)
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def get_part(name)
|
71
|
+
@parts.detect {|p| p.name == name }
|
72
|
+
end
|
73
|
+
|
74
|
+
private
|
75
|
+
|
76
|
+
def has_part?(name)
|
77
|
+
!!@parts.map(&:name).detect {|part_name| part_name == name }
|
78
|
+
end
|
79
|
+
|
80
|
+
end
|
81
|
+
end
|
data/lib/niki/version.rb
ADDED
data/niki.gemspec
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
$:.push File.expand_path("../lib", __FILE__)
|
3
|
+
require "niki/version"
|
4
|
+
|
5
|
+
Gem::Specification.new do |s|
|
6
|
+
s.name = "niki"
|
7
|
+
s.version = Niki::VERSION
|
8
|
+
s.platform = Gem::Platform::RUBY
|
9
|
+
s.authors = ["Josep M. Bach"]
|
10
|
+
s.email = ["josep.m.bach@gmail.com"]
|
11
|
+
s.homepage = "http://github.com/txus/niki"
|
12
|
+
s.summary = %q{A DSL to describe and play structured musical pieces, i.e. songs}
|
13
|
+
s.description = %q{A DSL to describe and play structured musical pieces, i.e. songs}
|
14
|
+
|
15
|
+
s.rubyforge_project = "niki"
|
16
|
+
|
17
|
+
s.add_runtime_dependency 'midilib'
|
18
|
+
|
19
|
+
s.files = `git ls-files`.split("\n")
|
20
|
+
s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
|
21
|
+
s.executables = `git ls-files -- bin/*`.split("\n").map{ |f| File.basename(f) }
|
22
|
+
s.require_paths = ["lib", "vendor"]
|
23
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
alias :L :lambda
|
2
|
+
|
3
|
+
%w{rubygems midilib/sequence midilib/consts}.each {|lib| require lib}
|
4
|
+
|
5
|
+
%w{archaeopteryx/core_ext/struct
|
6
|
+
archaeopteryx/core_ext/array
|
7
|
+
|
8
|
+
archaeopteryx/loop
|
9
|
+
archaeopteryx/drum
|
10
|
+
archaeopteryx/rhythm
|
11
|
+
archaeopteryx/mix
|
12
|
+
|
13
|
+
archaeopteryx/live_hacks
|
14
|
+
archaeopteryx/track
|
15
|
+
archaeopteryx/clip
|
16
|
+
|
17
|
+
archaeopteryx/midi/note
|
18
|
+
archaeopteryx/midi/clock
|
19
|
+
|
20
|
+
archaeopteryx/midi/file_output/file_midi
|
21
|
+
|
22
|
+
archaeopteryx/midi/practical_ruby_projects/no_midi_destinations
|
23
|
+
archaeopteryx/midi/practical_ruby_projects/core_midi
|
24
|
+
archaeopteryx/midi/practical_ruby_projects/core_foundation
|
25
|
+
archaeopteryx/midi/practical_ruby_projects/live_midi
|
26
|
+
archaeopteryx/midi/practical_ruby_projects/timer}.each do |archaeopteryx|
|
27
|
+
require File.expand_path(File.dirname(__FILE__)) + "/../#{archaeopteryx}"
|
28
|
+
end
|
@@ -0,0 +1,37 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
class Clip
|
3
|
+
attr_accessor :measures
|
4
|
+
def initialize(attributes)
|
5
|
+
@message = Message.create(attributes.merge(:midi_channel => 0,
|
6
|
+
:value => 127))
|
7
|
+
@measures = attributes[:measures]
|
8
|
+
end
|
9
|
+
def notes(measure)
|
10
|
+
[]
|
11
|
+
end
|
12
|
+
def messages(measure)
|
13
|
+
0 == measure % @measures ? [@message] : []
|
14
|
+
# if 0 == @relative_measure % @measures
|
15
|
+
# [@message]
|
16
|
+
# else
|
17
|
+
# []
|
18
|
+
# @relative_measure += 1
|
19
|
+
# end
|
20
|
+
end
|
21
|
+
def mutate(measure)
|
22
|
+
end
|
23
|
+
# def choose
|
24
|
+
# relative_measure = 1
|
25
|
+
# etc.
|
26
|
+
# end
|
27
|
+
def complete?
|
28
|
+
[true, false][rand(2)]
|
29
|
+
# current_measure == final_measure
|
30
|
+
# BUT! these need to be relative measures in this Clip, not absolute measures in the overall Loop
|
31
|
+
# possible that Loop should be called Engine or some crazy shit
|
32
|
+
# I guess relative_measure is really @current_measure
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
# probably rename this to make it drum-specific
|
@@ -0,0 +1,11 @@
|
|
1
|
+
class Array
|
2
|
+
def random
|
3
|
+
self[rand(self.size)]
|
4
|
+
end
|
5
|
+
end
|
6
|
+
|
7
|
+
# the weird thing is, sometimes this method works and sometimes it doesn't. it happens with other
|
8
|
+
# libs as well as this one. wtf is going on there? gotta investigate that shit.
|
9
|
+
|
10
|
+
# oh fuck I think I know why. I require ActiveSupport in my IRB so I'm assuming Array#rand from
|
11
|
+
# AS is part of Ruby itself. ooops.
|
@@ -0,0 +1,7 @@
|
|
1
|
+
# this allows you to create Structs and Struct subclasses with pseudo-keyword options hashes
|
2
|
+
# rather than with long sequences of unnamed args. written by Michael Fellinger.
|
3
|
+
class Struct
|
4
|
+
def self.create(hash)
|
5
|
+
new(*hash.values_at(*members.map{|member| member.to_sym}))
|
6
|
+
end
|
7
|
+
end
|
@@ -0,0 +1,24 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
class Drum
|
3
|
+
attr_accessor :note, :probabilities, :when, :next, :number_generator
|
4
|
+
def initialize(attributes)
|
5
|
+
%w{note probabilities when next number_generator}.each do |attribute|
|
6
|
+
eval("@#{attribute} = attributes[:#{attribute}]")
|
7
|
+
end
|
8
|
+
@queue = [attributes[:when]]
|
9
|
+
generate
|
10
|
+
end
|
11
|
+
def play?(beat)
|
12
|
+
@when[beat]
|
13
|
+
end
|
14
|
+
def generate
|
15
|
+
beats_on_which_to_play = []
|
16
|
+
@probabilities.each_with_index do |probability, index|
|
17
|
+
beats_on_which_to_play << index if @number_generator[] <= probability
|
18
|
+
end
|
19
|
+
@queue << L{|beat| beats_on_which_to_play.include? beat}
|
20
|
+
@when = @next[@queue]
|
21
|
+
end
|
22
|
+
alias :mutate :generate
|
23
|
+
end
|
24
|
+
end
|
@@ -0,0 +1,32 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
class Loop
|
3
|
+
attr_reader :midi
|
4
|
+
def initialize(attributes)
|
5
|
+
%w{generator
|
6
|
+
measures
|
7
|
+
beats
|
8
|
+
clock
|
9
|
+
midi
|
10
|
+
evil_timer_offset_wtf
|
11
|
+
infinite}.each do |option|
|
12
|
+
eval("@#{option} = attributes[:#{option}]")
|
13
|
+
end
|
14
|
+
end
|
15
|
+
def generate_beats
|
16
|
+
(1..@measures).each do |measure|
|
17
|
+
@generator.mutate(measure)
|
18
|
+
(0..(@beats - 1)).each do |beat|
|
19
|
+
play @generator.notes(beat)
|
20
|
+
@clock.tick
|
21
|
+
end
|
22
|
+
end
|
23
|
+
if @midi.infinite?
|
24
|
+
@midi.timer.at((@clock.start + @clock.time) - @evil_timer_offset_wtf,
|
25
|
+
&(L{generate_beats}))
|
26
|
+
end
|
27
|
+
end
|
28
|
+
def play(music)
|
29
|
+
music.each {|note| @midi.play(note)}
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# get singleton from std lib
|
2
|
+
module Archaeopteryx
|
3
|
+
module Midi
|
4
|
+
class Clock
|
5
|
+
attr_reader :time, :interval, :start
|
6
|
+
def initialize(bpm)
|
7
|
+
# assumes 16-step step sequencer, 4/4 beat, etc.
|
8
|
+
self.bpm = bpm
|
9
|
+
@start = Time.now.to_f
|
10
|
+
@time = 0
|
11
|
+
end
|
12
|
+
def bpm=(bpm)
|
13
|
+
seconds_in_a_minute = 60.0
|
14
|
+
beats_in_a_measure = 4.0
|
15
|
+
@interval = seconds_in_a_minute / bpm.to_f / beats_in_a_measure
|
16
|
+
end
|
17
|
+
def tick
|
18
|
+
@time += @interval
|
19
|
+
@time
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
@@ -0,0 +1,44 @@
|
|
1
|
+
# a quick note about FAIL. FileMIDI is actually of the same unspecified superclass as LiveMIDI. both
|
2
|
+
# need new, clearer names, but the nonexistent abstract superclass is just a class which takes options
|
3
|
+
# for args and has a method called play() which takes a note argument.
|
4
|
+
|
5
|
+
include MIDI
|
6
|
+
|
7
|
+
# christ this should probably live in some kind of module or something
|
8
|
+
|
9
|
+
class FileMIDI
|
10
|
+
attr_accessor :filename
|
11
|
+
def initialize(options)
|
12
|
+
raise :hell unless options.is_a? Hash
|
13
|
+
@filename = options[:filename]
|
14
|
+
@clock = options[:clock]
|
15
|
+
@events = []
|
16
|
+
|
17
|
+
@sequence = MIDI::Sequence.new
|
18
|
+
@sequence.tracks << (@track = MIDI::Track.new(@sequence))
|
19
|
+
@track.events << Tempo.new(Tempo.bpm_to_mpq(options[:tempo])) if options[:tempo]
|
20
|
+
@track.events << MetaEvent.new(META_SEQ_NAME, options[:name]) if options[:name]
|
21
|
+
|
22
|
+
# I'm not sure if this is actually necessary (?)
|
23
|
+
@track.events << Controller.new(0, CC_VOLUME, 127)
|
24
|
+
@track.events << ProgramChange.new(0, 1, 0)
|
25
|
+
end
|
26
|
+
def infinite?
|
27
|
+
false
|
28
|
+
end
|
29
|
+
def midilib_delta
|
30
|
+
# figuring this shit out was an epic fucking nightmare and to be honest I still have no idea why it works
|
31
|
+
((@sequence.note_to_delta("16th") / @clock.interval) * @clock.time).to_i
|
32
|
+
end
|
33
|
+
def play(note)
|
34
|
+
@track.merge [NoteOnEvent.new(note.channel,
|
35
|
+
note.number,
|
36
|
+
note.velocity,
|
37
|
+
midilib_delta)]
|
38
|
+
end
|
39
|
+
def write
|
40
|
+
File.open(@filename, 'wb') do |file|
|
41
|
+
@sequence.write(file)
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module CoreMIDI
|
2
|
+
require 'dl/import'
|
3
|
+
extend DL::Importable
|
4
|
+
|
5
|
+
dlload '/System/Library/Frameworks/CoreMIDI.framework/Versions/Current/CoreMIDI'
|
6
|
+
|
7
|
+
extern "int MIDIClientCreate(void *, void *, void *, void *)"
|
8
|
+
extern "int MIDIClientDispose(void *)"
|
9
|
+
extern "int MIDIGetNumberOfDestinations()"
|
10
|
+
extern "void * MIDIGetDestination(int)"
|
11
|
+
extern "int MIDIOutputPortCreate(void *, void *, void *)"
|
12
|
+
extern "void * MIDIPacketListInit(void *)"
|
13
|
+
# http://groups.google.com/group/ruby-midi/browse_thread/thread/85de6ea9373c57a4
|
14
|
+
extern "void * MIDIPacketListAdd(void *, int, void *, int, int, void*)"
|
15
|
+
extern "int MIDISend(void *, void *, void *)"
|
16
|
+
end
|
@@ -0,0 +1,117 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
module Midi
|
3
|
+
module PracticalRubyProjects
|
4
|
+
class LiveMIDI
|
5
|
+
|
6
|
+
# this object is a gateway between the Objective-C extern-ed functions from the CoreMIDI API
|
7
|
+
# and good old-fashioned Ruby. As such some of the code gets weird. This is nearly all from
|
8
|
+
# Topher Cyll's wicked book referenced in the MIT license file (practical_ruby_projects.rb),
|
9
|
+
# but with some refactoring and modification.
|
10
|
+
|
11
|
+
include CoreMIDI
|
12
|
+
include CoreFoundation
|
13
|
+
|
14
|
+
attr_reader :interval # this is a totally misleading variable name! real interval lives on Clock
|
15
|
+
attr_reader :timer
|
16
|
+
ON = 0x90
|
17
|
+
OFF = 0x80
|
18
|
+
PC = 0xC0 # program change, I think; not actually useable in Propellerhead Reason v3
|
19
|
+
CONTROLLER = 0xB0 # arbitrary controller message
|
20
|
+
|
21
|
+
def infinite?
|
22
|
+
true
|
23
|
+
end
|
24
|
+
|
25
|
+
def to_code
|
26
|
+
"LiveMIDI.new(:clock => @clock = attributes[:clock], :logging => false)"
|
27
|
+
end
|
28
|
+
|
29
|
+
def initialize(options)
|
30
|
+
@clock = options[:clock]
|
31
|
+
@logging = options[:logging]
|
32
|
+
@midi_destination = options[:midi_destination] || 0
|
33
|
+
if @logging
|
34
|
+
puts <<LOG_PLAYBACK
|
35
|
+
require 'lib/archaeopteryx'
|
36
|
+
@midi = #{self.to_code}
|
37
|
+
LOG_PLAYBACK
|
38
|
+
end
|
39
|
+
@interval = 60.0/120 # this is just a polling interval for the Thread - not a musical one
|
40
|
+
@timer = Timer.new(@interval/1000)
|
41
|
+
open
|
42
|
+
end
|
43
|
+
|
44
|
+
def play(midi_note, on_time = @clock.time)
|
45
|
+
if @logging
|
46
|
+
puts "@midi.play(#{midi_note.to_code}, #{on_time})"
|
47
|
+
end
|
48
|
+
on_time += @clock.start
|
49
|
+
@timer.at(on_time) {note_on(midi_note)}
|
50
|
+
@timer.at(on_time + midi_note.duration) {note_off(midi_note)}
|
51
|
+
end
|
52
|
+
|
53
|
+
def send(message)
|
54
|
+
send_controller_message(message.midi_channel, message.controller_number, message.value)
|
55
|
+
end
|
56
|
+
def send_controller_message(midi_channel, controller_number, value, on_time = @clock.time)
|
57
|
+
on_time += @clock.start
|
58
|
+
puts "scheduling #{controller_number} for #{on_time}" if @logging
|
59
|
+
@timer.at(on_time) do
|
60
|
+
control(midi_channel, controller_number, value)
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
def open
|
65
|
+
client_name = CoreFoundation.cFStringCreateWithCString(nil, "RubyMIDI", 0)
|
66
|
+
@client = DL::PtrData.new(nil)
|
67
|
+
CoreMIDI.mIDIClientCreate(client_name, nil, nil, @client.ref)
|
68
|
+
|
69
|
+
port_name = CoreFoundation.cFStringCreateWithCString(nil, "Output", 0)
|
70
|
+
@outport = DL::PtrData.new(nil)
|
71
|
+
CoreMIDI.mIDIOutputPortCreate(@client, port_name, @outport.ref)
|
72
|
+
|
73
|
+
number_of_destinations = CoreMIDI.mIDIGetNumberOfDestinations()
|
74
|
+
raise NoMIDIDestinations if number_of_destinations < 1
|
75
|
+
@destination = CoreMIDI.mIDIGetDestination(@midi_destination)
|
76
|
+
end
|
77
|
+
|
78
|
+
def close
|
79
|
+
CoreMIDI.mIDIClientDispose(@client)
|
80
|
+
end
|
81
|
+
|
82
|
+
def clear
|
83
|
+
@timer.flush
|
84
|
+
end
|
85
|
+
|
86
|
+
def message(*args)
|
87
|
+
format = "C" * args.size
|
88
|
+
bytes = args.pack(format).to_ptr
|
89
|
+
packet_list = DL.malloc(256)
|
90
|
+
packet_ptr = CoreMIDI.mIDIPacketListInit(packet_list)
|
91
|
+
# http://groups.google.com/group/ruby-midi/browse_thread/thread/85de6ea9373c57a4
|
92
|
+
packet_ptr = CoreMIDI.mIDIPacketListAdd(packet_list, 256, packet_ptr, 0, args.size, bytes)
|
93
|
+
CoreMIDI.mIDISend(@outport, @destination, packet_list)
|
94
|
+
end
|
95
|
+
|
96
|
+
def note_on(midi_note)
|
97
|
+
message(ON | midi_note.channel, midi_note.number, midi_note.velocity)
|
98
|
+
end
|
99
|
+
|
100
|
+
def note_off(midi_note)
|
101
|
+
message(OFF | midi_note.channel, midi_note.number, midi_note.velocity)
|
102
|
+
end
|
103
|
+
|
104
|
+
def program_change(channel, preset)
|
105
|
+
message(PC | channel, preset)
|
106
|
+
end
|
107
|
+
|
108
|
+
def pulse(channel, controller_id, value)
|
109
|
+
# puts "sending now: #{Time.now.to_f}" if @logging
|
110
|
+
# puts "#{[channel, controller_id, value].inspect}" if @logging
|
111
|
+
message(CONTROLLER | channel, controller_id, value)
|
112
|
+
end
|
113
|
+
alias :control :pulse
|
114
|
+
end
|
115
|
+
end
|
116
|
+
end
|
117
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
module Midi
|
3
|
+
# most of this code comes from "Practical Ruby Projects" by Topher Cyll, published by
|
4
|
+
# APress, which is a pretty bad-ass book. here's the license:
|
5
|
+
#
|
6
|
+
# The MIT License
|
7
|
+
#
|
8
|
+
# Copyright (c) 2006, 2007 Topher Cyll
|
9
|
+
#
|
10
|
+
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
11
|
+
# of this software and associated documentation files (the "Software"), to deal
|
12
|
+
# in the Software without restriction, including without limitation the rights
|
13
|
+
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
14
|
+
# copies of the Software, and to permit persons to whom the Software is
|
15
|
+
# furnished to do so, subject to the following conditions:
|
16
|
+
#
|
17
|
+
# The above copyright notice and this permission notice shall be included in
|
18
|
+
# all copies or substantial portions of the Software.
|
19
|
+
#
|
20
|
+
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
21
|
+
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
22
|
+
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
23
|
+
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
24
|
+
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
25
|
+
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
26
|
+
# THE SOFTWARE.
|
27
|
+
module PracticalRubyProjects
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,35 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
module Midi
|
3
|
+
module PracticalRubyProjects
|
4
|
+
class Timer
|
5
|
+
def initialize(resolution)
|
6
|
+
@resolution = resolution
|
7
|
+
@queue = []
|
8
|
+
|
9
|
+
Thread.new do
|
10
|
+
while true
|
11
|
+
dispatch
|
12
|
+
sleep(@resolution)
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
def flush
|
18
|
+
@queue = []
|
19
|
+
end
|
20
|
+
|
21
|
+
def at(time, &block)
|
22
|
+
time = time.to_f if time.kind_of?(Time)
|
23
|
+
@queue.push [time, block]
|
24
|
+
end
|
25
|
+
|
26
|
+
private
|
27
|
+
def dispatch
|
28
|
+
now = Time.now.to_f
|
29
|
+
ready, @queue = @queue.partition {|time, proc| time <= now}
|
30
|
+
ready.each {|time, proc| proc.call(time)}
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
class Mix
|
3
|
+
attr_accessor :rhythms
|
4
|
+
def initialize(attributes)
|
5
|
+
@rhythms = attributes[:rhythms]
|
6
|
+
end
|
7
|
+
def notes(beat)
|
8
|
+
notes = []
|
9
|
+
@rhythms.each do |rhythm|
|
10
|
+
notes << rhythm.notes(beat)
|
11
|
+
end
|
12
|
+
notes.flatten
|
13
|
+
end
|
14
|
+
def messages(measure) # obviously these should be refactored to inject()s
|
15
|
+
messages = []
|
16
|
+
@rhythms.each do |rhythm|
|
17
|
+
messages << rhythm.messages(measure)
|
18
|
+
end
|
19
|
+
messages.flatten
|
20
|
+
end
|
21
|
+
def mutate(measure)
|
22
|
+
@rhythms.each {|rhythm| rhythm.mutate(measure)}
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
|
27
|
+
# Mix nearly identical to Rhythm
|
@@ -0,0 +1,29 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
class Rhythm
|
3
|
+
def initialize(attributes)
|
4
|
+
@mutation = attributes[:mutation]
|
5
|
+
@drumfile = attributes[:drumfile]
|
6
|
+
reload
|
7
|
+
end
|
8
|
+
def reload
|
9
|
+
puts "\a" # flash the screen ; only valid on my box and similarly configured machines!
|
10
|
+
@drums = eval(File.read(@drumfile))
|
11
|
+
end
|
12
|
+
def notes(beat)
|
13
|
+
(@drums.collect do |drum|
|
14
|
+
drum.note if drum.play? beat
|
15
|
+
end).compact
|
16
|
+
end
|
17
|
+
def messages(beat)
|
18
|
+
[]
|
19
|
+
end
|
20
|
+
def mutate(measure)
|
21
|
+
if @mutation[measure]
|
22
|
+
reload # reloading can kill mutations!
|
23
|
+
@drums.each {|drum| drum.mutate}
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
# probably rename this to make it drum-specific
|
@@ -0,0 +1,16 @@
|
|
1
|
+
module Archaeopteryx
|
2
|
+
class Track < Mix
|
3
|
+
attr_accessor :current
|
4
|
+
def initialize(attributes)
|
5
|
+
@rhythms = attributes[:rhythms]
|
6
|
+
@measure = 1
|
7
|
+
new_clip
|
8
|
+
end
|
9
|
+
def new_clip
|
10
|
+
@current = @rhythms.random
|
11
|
+
end
|
12
|
+
def messages(measure)
|
13
|
+
@current.messages(measure)
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
metadata
ADDED
@@ -0,0 +1,113 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: niki
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
hash: 29
|
5
|
+
prerelease:
|
6
|
+
segments:
|
7
|
+
- 0
|
8
|
+
- 0
|
9
|
+
- 1
|
10
|
+
version: 0.0.1
|
11
|
+
platform: ruby
|
12
|
+
authors:
|
13
|
+
- Josep M. Bach
|
14
|
+
autorequire:
|
15
|
+
bindir: bin
|
16
|
+
cert_chain: []
|
17
|
+
|
18
|
+
date: 2011-08-15 00:00:00 Z
|
19
|
+
dependencies:
|
20
|
+
- !ruby/object:Gem::Dependency
|
21
|
+
name: midilib
|
22
|
+
prerelease: false
|
23
|
+
requirement: &id001 !ruby/object:Gem::Requirement
|
24
|
+
none: false
|
25
|
+
requirements:
|
26
|
+
- - ">="
|
27
|
+
- !ruby/object:Gem::Version
|
28
|
+
hash: 3
|
29
|
+
segments:
|
30
|
+
- 0
|
31
|
+
version: "0"
|
32
|
+
type: :runtime
|
33
|
+
version_requirements: *id001
|
34
|
+
description: A DSL to describe and play structured musical pieces, i.e. songs
|
35
|
+
email:
|
36
|
+
- josep.m.bach@gmail.com
|
37
|
+
executables: []
|
38
|
+
|
39
|
+
extensions: []
|
40
|
+
|
41
|
+
extra_rdoc_files: []
|
42
|
+
|
43
|
+
files:
|
44
|
+
- .gitignore
|
45
|
+
- Gemfile
|
46
|
+
- Gemfile.lock
|
47
|
+
- Rakefile
|
48
|
+
- Readme.md
|
49
|
+
- examples/my_song.rb
|
50
|
+
- examples/my_song.rns
|
51
|
+
- lib/core_ext/array.rb
|
52
|
+
- lib/core_ext/fixnum.rb
|
53
|
+
- lib/niki.rb
|
54
|
+
- lib/niki/chords.rb
|
55
|
+
- lib/niki/part.rb
|
56
|
+
- lib/niki/song.rb
|
57
|
+
- lib/niki/version.rb
|
58
|
+
- niki.gemspec
|
59
|
+
- vendor/archaeopteryx/archaeopteryx.rb
|
60
|
+
- vendor/archaeopteryx/clip.rb
|
61
|
+
- vendor/archaeopteryx/core_ext/array.rb
|
62
|
+
- vendor/archaeopteryx/core_ext/struct.rb
|
63
|
+
- vendor/archaeopteryx/drum.rb
|
64
|
+
- vendor/archaeopteryx/live_hacks.rb
|
65
|
+
- vendor/archaeopteryx/loop.rb
|
66
|
+
- vendor/archaeopteryx/midi/clock.rb
|
67
|
+
- vendor/archaeopteryx/midi/file_output/file_midi.rb
|
68
|
+
- vendor/archaeopteryx/midi/note.rb
|
69
|
+
- vendor/archaeopteryx/midi/practical_ruby_projects/core_foundation.rb
|
70
|
+
- vendor/archaeopteryx/midi/practical_ruby_projects/core_midi.rb
|
71
|
+
- vendor/archaeopteryx/midi/practical_ruby_projects/live_midi.rb
|
72
|
+
- vendor/archaeopteryx/midi/practical_ruby_projects/no_midi_destinations.rb
|
73
|
+
- vendor/archaeopteryx/midi/practical_ruby_projects/practical_ruby_projects.rb
|
74
|
+
- vendor/archaeopteryx/midi/practical_ruby_projects/timer.rb
|
75
|
+
- vendor/archaeopteryx/mix.rb
|
76
|
+
- vendor/archaeopteryx/rhythm.rb
|
77
|
+
- vendor/archaeopteryx/track.rb
|
78
|
+
homepage: http://github.com/txus/niki
|
79
|
+
licenses: []
|
80
|
+
|
81
|
+
post_install_message:
|
82
|
+
rdoc_options: []
|
83
|
+
|
84
|
+
require_paths:
|
85
|
+
- lib
|
86
|
+
- vendor
|
87
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
88
|
+
none: false
|
89
|
+
requirements:
|
90
|
+
- - ">="
|
91
|
+
- !ruby/object:Gem::Version
|
92
|
+
hash: 3
|
93
|
+
segments:
|
94
|
+
- 0
|
95
|
+
version: "0"
|
96
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
97
|
+
none: false
|
98
|
+
requirements:
|
99
|
+
- - ">="
|
100
|
+
- !ruby/object:Gem::Version
|
101
|
+
hash: 3
|
102
|
+
segments:
|
103
|
+
- 0
|
104
|
+
version: "0"
|
105
|
+
requirements: []
|
106
|
+
|
107
|
+
rubyforge_project: niki
|
108
|
+
rubygems_version: 1.8.6
|
109
|
+
signing_key:
|
110
|
+
specification_version: 3
|
111
|
+
summary: A DSL to describe and play structured musical pieces, i.e. songs
|
112
|
+
test_files: []
|
113
|
+
|