line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package MIDI::Simple::Drummer::Euclidean; |
2
|
|
|
|
|
|
|
$MIDI::Simple::Drummer::Euclidean::VERSION = '0.0813'; |
3
|
|
|
|
|
|
|
our $AUTHORITY = 'cpan:GENE'; |
4
|
1
|
|
|
1
|
|
773
|
use strict; |
|
1
|
|
|
|
|
4
|
|
|
1
|
|
|
|
|
32
|
|
5
|
1
|
|
|
1
|
|
6
|
use warnings; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
26
|
|
6
|
1
|
|
|
1
|
|
412
|
use parent 'MIDI::Simple::Drummer'; |
|
1
|
|
|
|
|
286
|
|
|
1
|
|
|
|
|
6
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
sub new { |
9
|
1
|
|
|
1
|
1
|
684
|
my $self = shift; |
10
|
1
|
|
|
|
|
9
|
$self->SUPER::new( |
11
|
|
|
|
|
|
|
-onsets => 4, |
12
|
|
|
|
|
|
|
-patch => 25, |
13
|
|
|
|
|
|
|
-rhythm => undef, |
14
|
|
|
|
|
|
|
-pad => 'kick', |
15
|
|
|
|
|
|
|
@_ |
16
|
|
|
|
|
|
|
); |
17
|
|
|
|
|
|
|
} |
18
|
|
|
|
|
|
|
|
19
|
|
|
|
|
|
|
sub _default_patterns { |
20
|
1
|
|
|
1
|
|
2
|
my $self = shift; |
21
|
|
|
|
|
|
|
return { |
22
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
1 => sub { |
24
|
0
|
|
|
0
|
|
0
|
my $self = shift; |
25
|
|
|
|
|
|
|
my $rhythm = $self->{-rhythm} |
26
|
|
|
|
|
|
|
? $self->{-rhythm} |
27
|
0
|
0
|
|
|
|
0
|
: $self->euclid($self->{-onsets}, $self->beats); |
28
|
0
|
|
|
|
|
0
|
for my $i ( @$rhythm ) { |
29
|
0
|
0
|
|
|
|
0
|
if ( $i eq 'x' ) { |
30
|
0
|
|
|
|
|
0
|
my $pad = $self->{-pad}; |
31
|
0
|
|
0
|
|
|
0
|
my $note = $self->$pad || $self->snare; |
32
|
0
|
|
|
|
|
0
|
$self->note($self->EIGHTH, $note ); |
33
|
|
|
|
|
|
|
} |
34
|
|
|
|
|
|
|
else { |
35
|
0
|
|
|
|
|
0
|
$self->rest($self->EIGHTH); |
36
|
|
|
|
|
|
|
} |
37
|
|
|
|
|
|
|
} |
38
|
|
|
|
|
|
|
} |
39
|
|
|
|
|
|
|
|
40
|
1
|
|
|
|
|
18
|
}; |
41
|
|
|
|
|
|
|
} |
42
|
|
|
|
|
|
|
|
43
|
|
|
|
|
|
|
sub euclid { |
44
|
1
|
|
|
1
|
1
|
578
|
my $self = shift; |
45
|
1
|
|
|
|
|
3
|
my ($p, $q) = @_; |
46
|
|
|
|
|
|
|
|
47
|
|
|
|
|
|
|
# Onsets per measure |
48
|
1
|
|
33
|
|
|
7
|
$p ||= $self->{-onsets}; |
49
|
|
|
|
|
|
|
# Beats per measure |
50
|
1
|
|
33
|
|
|
8
|
$q ||= $self->beats; |
51
|
|
|
|
|
|
|
|
52
|
|
|
|
|
|
|
# Line is from x=0, y=1 to x=$BPM, y=$mod+1 |
53
|
|
|
|
|
|
|
# Then from that, for each $y from # 1..$mod |
54
|
|
|
|
|
|
|
# figure out the x value to see where beat would be. |
55
|
|
|
|
|
|
|
|
56
|
1
|
|
|
|
|
2
|
my $intercept = 1; |
57
|
|
|
|
|
|
|
|
58
|
|
|
|
|
|
|
# y = mx + b; b is 1 as we're drawing the intercept through that point, |
59
|
|
|
|
|
|
|
# and then (y2-y1)/(x2-x1) reduces to just: |
60
|
1
|
|
|
|
|
2
|
my $slope = $p / $q; |
61
|
|
|
|
|
|
|
|
62
|
1
|
|
|
|
|
4
|
my @onsets = ('.') x $q; |
63
|
|
|
|
|
|
|
|
64
|
1
|
|
|
|
|
3
|
for my $y ( 1 .. $p ) { |
65
|
|
|
|
|
|
|
# solve x = (y-b)/m rounding nearest and put the beat there |
66
|
4
|
|
|
|
|
11
|
$onsets[ sprintf "%.0f", ( $y - $intercept ) / $slope ] = 'x'; |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
|
69
|
1
|
|
|
|
|
13
|
return \@onsets; |
70
|
|
|
|
|
|
|
}; |
71
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
sub rotate { |
73
|
0
|
|
|
0
|
1
|
|
my $self = shift; |
74
|
0
|
|
|
|
|
|
my $phrase = shift; |
75
|
|
|
|
|
|
|
|
76
|
0
|
|
|
|
|
|
my $done = 0; |
77
|
0
|
|
|
|
|
|
while ( $done == 0 ) { |
78
|
0
|
|
|
|
|
|
my $i = shift @$phrase; |
79
|
0
|
|
|
|
|
|
push @$phrase, $i; |
80
|
0
|
0
|
|
|
|
|
$done++ if $phrase->[0] eq 'x'; |
81
|
|
|
|
|
|
|
} |
82
|
|
|
|
|
|
|
|
83
|
0
|
|
|
|
|
|
return $phrase; |
84
|
|
|
|
|
|
|
} |
85
|
|
|
|
|
|
|
|
86
|
|
|
|
|
|
|
1; |
87
|
|
|
|
|
|
|
|
88
|
|
|
|
|
|
|
__END__ |