ramekin 0.2.2 → 0.2.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: f859d4c5dc75cc34a8053dcf5d9958aa48a80aea10d58bb44ac130e1b37c9af0
4
- data.tar.gz: b1f2678cb27b008646ad250346c5345c7bf173749ba119d1b42ca5b8dbd3f1f4
3
+ metadata.gz: a249d3680d6f96fe57c566445a3add42d0287de30bd86a556131963b58b239b8
4
+ data.tar.gz: df8ab404da547bd42b68fee9ef45d3eeed5c1c46201c997dea2a04cb65a8e0aa
5
5
  SHA512:
6
- metadata.gz: 5ac7a907935cdb923dd3007b739b257c89262cbd862fd7b987189487330c3b39c6ae762741c5cd91ffa0b8e5f07455ab40ad90fef6178e773db8db32fb9ecf23
7
- data.tar.gz: 3ab59a78f14c06923c2f93011567b8fa6a41aa9c5d56ef8121088157b43f7775da6c8c43ebfe497039498348f3583d3c70523e121904922614215c094f410431
6
+ metadata.gz: 3001f9832d107a892137de8e811c842d972ad226de761e2cd033b79845d455988de28e328c55d14c78215c61a6adb5d11b3819ca35682e66a3c3b9bb0593961c
7
+ data.tar.gz: b52c6b301948c3400d7c0c7db3cdc4f3da93e1827a352cd927d5fa8c7bf06d06aee67e70b21392002262fee92ede83cef2cbdd69dbf4c11d84ecb55af5e890f1
data/README.md CHANGED
@@ -42,10 +42,19 @@ ramekin setup
42
42
  # Usage
43
43
 
44
44
  ```
45
- usage: ramekin [command] [flags]
45
+ usage: ramekin [--pack NAME=PATH] [command] [flags]
46
46
  (default command: compile)
47
47
 
48
+ ramekin --pack my_cool_pack=/path/to/my_cool_pack [...]
49
+ Specifies an additional custom sample location.
50
+ Files compiled this way should use #pack "my_cool_pack",
51
+ and sample paths relative to the deepest common path that
52
+ contains .brr files. Use with `package --list my_cool_pack` to
53
+ see details.
54
+
48
55
  ramekin compile -i filename.rmk [flags]
56
+ ramekin compile filename.rmk [flags]
57
+ ramekin filename.rmk [flags]
49
58
  flags:
50
59
  -h --help
51
60
  display this message
@@ -399,6 +408,26 @@ Or drum instrument helpers:
399
408
 
400
409
  Or anything else you might find helpful.
401
410
 
411
+ ## Custom Samples
412
+
413
+ If you have your own BRR-format samples that you'd like to use, Ramekin supports loading these through *custom packs*, specified on the CLI. First, organize your samples into a directory, with an optional `!patterns.txt` or `tunings.txt` file, much as may be found on [SMW Central's BRR repository](https://www.smwcentral.net/?p=section&s=brrsamples). In the simplest case, this is just a directory with .brr files in it.
414
+
415
+ Then, when running ramekin, use:
416
+
417
+ ```console
418
+ $ ramekin --pack 'my cool pack=/path/to/my_cool_pack' [the rest of the ramekin command]
419
+ ```
420
+
421
+ where `/path/to/my_cool_pack` is replaced with a relative path to your BRR files directory. Your ramekin files can then freely use `#pack "my cool pack"`, and conjure up instruments from the pack. If no tuning file is specified, you may need to use `#tuning:XXXX` to specify a tuning, as the default tuning of `0000` is unlikely to be audible.
422
+
423
+ If Ramekin is having trouble finding your BRR files, try:
424
+
425
+ ```console
426
+ $ ramekin --pack 'my cool pack=/path/to/my_cool_pack' package --list 'my cool pack'
427
+ ```
428
+
429
+ which will display the paths it expects you to use for each BRR sample it found.
430
+
402
431
  # Thank you for trying out Ramekin!
403
432
 
404
433
  If you find any bugs, please report them in the issue tracker here on Codeberg. If you have questions, design ideas, or complaints, or just want to chat, please feel free to ask anything in [my discord](https://jneen.ca/discord). Cheers!
@@ -48,6 +48,10 @@ module Ramekin
48
48
  ln("#@amk_path/samples/default", "#@workdir/samples/default")
49
49
  ln("#@amk_path/samples/optimized", "#@workdir/samples/optimized")
50
50
  ln("#@amk_path/samples/EMPTY.brr", "#@workdir/samples/EMPTY.brr")
51
+ if File.exist?("#@amk_path/samples/SPECIALWAVE.brr")
52
+ ln("#@amk_path/samples/SPECIALWAVE.brr", "#@workdir/samples/SPECIALWAVE.brr")
53
+ end
54
+
51
55
  if @meta.instruments.any?
52
56
  sample_dir = "#@workdir/samples/#{@meta.sample_path || basename}"
53
57
  FileUtils.mkdir_p(sample_dir)
data/lib/ramekin/bends.rb CHANGED
@@ -32,11 +32,11 @@ module Ramekin
32
32
  end
33
33
 
34
34
  def tie?
35
- false
35
+ @notes.first.tie?
36
36
  end
37
37
 
38
38
  def ending_octave
39
- @notes.first.octave
39
+ @notes.last.octave
40
40
  end
41
41
 
42
42
  # in the event of a tie-free bend exactly at a note out where we *also*
@@ -74,7 +74,7 @@ module Ramekin
74
74
  out << from.octave_amk(oct)
75
75
  out << from.note_name
76
76
  out << "=#{bend_duration}"
77
- out << dd(bend_duration, to)
77
+ out << dd(bend_duration, to, from.octave_num)
78
78
  out << "$f4$01^=#{leftover}"
79
79
 
80
80
  out.string
@@ -115,22 +115,22 @@ module Ramekin
115
115
  note = @notes[i]
116
116
  to_note = @notes[i+1]
117
117
 
118
+ next unless check_duration!(note, divisor)
119
+
118
120
  if i == 0
119
121
  out << note.octave_amk(octave)
122
+ octave = note.octave_num
120
123
  out << note.note_name
121
124
  else
122
125
  out << '^'
123
126
  end
124
127
 
125
- next unless check_duration!(note, divisor)
126
-
127
- ticks = note.ticks
128
- ticks /= divisor
128
+ ticks = note.ticks / divisor
129
129
 
130
130
  # always use tick count notation here because
131
131
  # amk can insert ties in some cases which will break $dd
132
132
  out << "=#{ticks}"
133
- out << dd(ticks, to_note)
133
+ out << dd(ticks, to_note, octave)
134
134
  out << '$f4$01' if @end_legato && i == 0
135
135
  end
136
136
 
@@ -138,15 +138,9 @@ module Ramekin
138
138
  out.string
139
139
  end
140
140
 
141
- def dd(dur, note)
142
- hex = note.note_hex
143
-
144
- unless 0x80 <= hex && hex <= 0xC5
145
- error! 'note out of range (o1c - o6a)', el: note
146
- return ''
147
- end
148
-
149
- sprintf '$dd$00$%02x$%02x', dur, note.note_hex
141
+ def dd(dur, note, current_octave)
142
+ oct = note.octave_switching_amk(current_octave)
143
+ sprintf '$dd$00$%02x%s%s', dur, oct, note.note_name
150
144
  end
151
145
  end
152
146
 
data/lib/ramekin/cli.rb CHANGED
@@ -19,10 +19,19 @@ module Ramekin
19
19
  def usage
20
20
  $stderr.puts "Ramekin version #{RAMEKIN_VERSION}"
21
21
  $stderr.puts ""
22
- $stderr.puts "usage: ramekin [command] [flags]"
22
+ $stderr.puts "usage: ramekin [--pack NAME=PATH] [command] [flags]"
23
23
  $stderr.puts " (default command: compile)"
24
24
  $stderr.puts ""
25
+ $stderr.puts "ramekin --pack my_cool_pack=/path/to/my_cool_pack [...]"
26
+ $stderr.puts " Specifies an additional custom sample location."
27
+ $stderr.puts " Files compiled this way should use #pack \"my_cool_pack\","
28
+ $stderr.puts " and sample paths relative to the deepest common path that"
29
+ $stderr.puts " contains .brr files. Use with `package --list my_cool_pack` to"
30
+ $stderr.puts " see details."
31
+ $stderr.puts ""
25
32
  $stderr.puts "ramekin compile -i filename.rmk [flags]"
33
+ $stderr.puts "ramekin compile filename.rmk [flags]"
34
+ $stderr.puts "ramekin filename.rmk [flags]"
26
35
  $stderr.puts " flags:"
27
36
  $stderr.puts " -h --help"
28
37
  $stderr.puts " display this message"
@@ -388,15 +397,21 @@ module Ramekin
388
397
  return 0
389
398
  end
390
399
 
391
- case argv[0]
392
- when 'compile'
393
- compile(*argv[1..])
394
- when 'package', 'pack'
395
- package(*argv[1..])
396
- when 'setup'
397
- setup(*argv[1..])
398
- else
399
- compile(*argv)
400
+ while argv.any?
401
+ case (arg = argv.shift)
402
+ when '--pack'
403
+ pack = argv.shift or fail! '--pack missing a pack definition NAME=PATH'
404
+ pack =~ /\A(.*?)=(.*)\z/ or fail! 'invalid --pack, must be of the form NAME=PATH'
405
+ SamplePack.add_custom($1, File.expand_path($2))
406
+ when 'compile'
407
+ return compile(*argv)
408
+ when 'package', 'pack'
409
+ return package(*argv)
410
+ when 'setup'
411
+ return setup(*argv)
412
+ else
413
+ return compile(arg, *argv)
414
+ end
400
415
  end
401
416
  end
402
417
  end
data/lib/ramekin/meta.rb CHANGED
@@ -284,17 +284,16 @@ module Ramekin
284
284
  end
285
285
 
286
286
  def adsr
287
- @adsr ||= ext_adsr || pack_adsr \
288
- or error! "no adsr configured for #{name.value}", el: name
287
+ @adsr ||= ext_adsr || pack_adsr || [0, 7, 7, 0x1f]
289
288
  end
290
289
 
291
290
  def tuning
292
- @tuning ||= ext_tuning || pack_tuning \
293
- or error! "no tuning configured for #{name.value}", el: name
291
+ @tuning ||= ext_tuning || pack_tuning || [4, 0]
294
292
  end
295
293
 
296
294
  def pack_tuning
297
295
  _, _, _, d, e = pack_hexes
296
+ return nil unless d && e
298
297
  [d, e]
299
298
  end
300
299
 
@@ -319,7 +318,7 @@ module Ramekin
319
318
  end
320
319
 
321
320
  def gain
322
- @gain ||= ext_gain || pack_gain || 0xFF
321
+ @gain ||= ext_gain || pack_gain || 0x8F
323
322
  end
324
323
 
325
324
  def hexes
@@ -332,7 +331,7 @@ module Ramekin
332
331
  adsr1 = ((7 - d)*16 | 0x80) + (15 - a)
333
332
  adsr2 = (s*32 + (31-r))
334
333
 
335
- [to_hex(adsr1), to_hex(adsr2), g, t1, t2]
334
+ [adsr1, adsr2, g, t1, t2].map(&method(:to_hex))
336
335
  end
337
336
 
338
337
  def to_hex(x)
@@ -107,6 +107,14 @@ module Ramekin
107
107
  end
108
108
  end
109
109
 
110
+ def octave_switching_amk(current_octave)
111
+ diff = octave_num - current_octave
112
+
113
+ return '' if diff == 0
114
+ return '<' * (-diff) if diff < 0
115
+ return '>' * diff if diff > 0
116
+ end
117
+
110
118
  def to_amk(current_octave=nil, divisor=1)
111
119
  return "#{octave_amk(current_octave)}#{note_name}#{length_amk(divisor)}"
112
120
  end
@@ -58,7 +58,7 @@ module Ramekin
58
58
  yield "; https://codeberg.org/jneen/ramekin\n\n"
59
59
 
60
60
  # TODO
61
- yield "#amk #{@track.meta.amk&.value || 2}\n\n"
61
+ yield "#amk #{@track.meta.amk&.value || 4}\n\n"
62
62
 
63
63
  if m.instruments.any?
64
64
  yield "#path #{(m.sample_path || @filename.chomp('.rmk')).inspect}\n"
@@ -130,7 +130,7 @@ module Ramekin
130
130
 
131
131
  yield el.to_amk(@octave, @track.meta.divisor)
132
132
 
133
- @octave = el.ending_octave
133
+ @octave = el.ending_octave unless el.tie?
134
134
  when NoteEvent
135
135
  old_tick = @tick
136
136
  @tick += el.ticks
@@ -70,6 +70,12 @@ module Ramekin
70
70
  def self.each(&b)
71
71
  return enum_for(:each) unless block_given?
72
72
 
73
+ custom_packs.each do |name, path|
74
+ Dir.chdir(path) do
75
+ yield CustomSamplePack.new(name, path)
76
+ end
77
+ end
78
+
73
79
  Dir.chdir Ramekin.config.packages_dir do
74
80
  Dir.entries('.').sort.each do |subd|
75
81
  next unless /\A\d+\z/ =~ subd
@@ -80,6 +86,14 @@ module Ramekin
80
86
  end
81
87
  end
82
88
 
89
+ def self.custom_packs
90
+ @custom_packs ||= {}
91
+ end
92
+
93
+ def self.add_custom(name, path)
94
+ custom_packs[name] = path
95
+ end
96
+
83
97
  def initialize(data)
84
98
  @meta = data
85
99
  end
@@ -143,7 +157,7 @@ module Ramekin
143
157
 
144
158
  def find_tunings(name_re)
145
159
  raw = []
146
- Dir.glob('**/*.txt').sort.each do |entry|
160
+ Dir.glob('**/*.txt', File::FNM_CASEFOLD).sort.each do |entry|
147
161
  next unless File.basename(entry).downcase =~ name_re
148
162
 
149
163
  contents = File.read(entry, encoding: 'binary')
@@ -169,7 +183,19 @@ module Ramekin
169
183
  out
170
184
  end
171
185
 
172
- def index_packs
186
+ def index_and_save!
187
+ index!
188
+
189
+ File.write('.index.json', JSON.dump(
190
+ prefix: @prefix,
191
+ brrs: @brrs,
192
+ tunings: @tunings,
193
+ ))
194
+
195
+ @cached_index = nil
196
+ end
197
+
198
+ def index!
173
199
  paths = []
174
200
  raw_tunings = []
175
201
 
@@ -177,7 +203,7 @@ module Ramekin
177
203
  other_tunings = find_tunings(/\Atuning/)
178
204
  misc_tunings = find_tunings(//)
179
205
 
180
- @brrs = Dir.glob('**/*.brr')
206
+ @brrs = Dir.glob('**/*.brr', File::FNM_CASEFOLD)
181
207
  @brrs.sort!
182
208
  @brrs.uniq!
183
209
  @prefix = shortest_prefix(@brrs.map(&File.method(:dirname)))
@@ -200,14 +226,6 @@ module Ramekin
200
226
 
201
227
  @tunings[brr] = candidates.map { |p, _, h| [p, h] }
202
228
  end
203
-
204
- File.write('.index.json', JSON.dump(
205
- prefix: @prefix,
206
- brrs: @brrs,
207
- tunings: @tunings,
208
- ))
209
-
210
- @cached_index = nil
211
229
  end
212
230
 
213
231
  def cached_index
@@ -235,12 +253,13 @@ module Ramekin
235
253
  end
236
254
 
237
255
  def prefix_dir
256
+ return dir if prefix == '.'
238
257
  File.join(dir, prefix)
239
258
  end
240
259
 
241
260
  def self.download_all!
242
261
  smwc_each(&:download)
243
- each(&:index_packs)
262
+ each(&:index_and_save!)
244
263
  end
245
264
 
246
265
  def needs_download?
@@ -298,4 +317,32 @@ module Ramekin
298
317
  File.write(meta_file, JSON.dump(@meta))
299
318
  end
300
319
  end
320
+
321
+ class CustomSamplePack < SamplePack
322
+ attr_reader :dir, :name
323
+ def initialize(name, dir)
324
+ @name = name
325
+ @dir = dir
326
+ end
327
+
328
+ def cached_index
329
+ @cached_index ||= begin
330
+ Dir.chdir(dir) { index! }
331
+
332
+ {
333
+ 'prefix' => @prefix,
334
+ 'brrs' => @brrs,
335
+ 'tunings' => @tunings,
336
+ }
337
+ end
338
+ end
339
+
340
+ def needs_download?
341
+ false
342
+ end
343
+
344
+ def download
345
+ # pass
346
+ end
347
+ end
301
348
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: ramekin
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.2.2
4
+ version: 0.2.4
5
5
  platform: ruby
6
6
  authors:
7
7
  - jneen