NAME

Music::RecRhythm - rhythms within rhythms within rhythms

SYNOPSIS

use Music::RecRhythm;

my $one = Music::RecRhythm->new( set => [qw(2 2 1 2 2 2 1)] );

my $two = $one->rebuild;  # clone the (original) object

$one->is_silent(1);       # silent (but present)

$one->next($two);         # link for recursion

$one->recurse( sub { ... } );

DESCRIPTION

A utility module for recusive rhythm construction, where a set is defined as an array reference of positive integers (beats). Multiple such objects may be linked through the next attribute, which the recurse method follows. Each next rhythm is played out in full for each beat of the parent rhythm, though whether these events are simultaneous or strung out in time depends on the callback code provided to recurse.

CONSTRUCTOR

The new method accepts any of the "ATTRIBUTES". The set attribute must be supplied.

ATTRIBUTES

count

A read-only count of the beats in the set. Updated when set is changed.

is_silent

Boolean as to whether or not the callback function of recurse will be invoked for beats of the set. False by default. Recursion will continue through silent objects as per usual; is_silent merely disables calling the callback, so "silent, but present" may be a more accurate term for this attribute.

next

Optional next object to recurse into. While often a Music::RecRhythm object, any object that supports the necessary method calls could be used. Recursion will stop should this attribute be undef (the default). Probably should not be changed in the middle of a recurse call.

set

An array reference of one or more positive integers (a.k.a. beats). This attribute must be supplied at construction time.

sum

A read-only sum of the beats in the set. Updated when set is changed.

METHODS

beatfactor

Lowest common multiple of the beats and sum of the beats such that the durations at each level of recursion work out to the same overall duration. Hopefully. Uses Math::BigInt though downgrades that via numify, so integers larger than what perl can handle internally may be problematic.

levels

Returns the number of levels recursion will take place over. May be useful prior to a recurse call if an array of MIDI tracks (one for each level) need be created, or similar. Note that the actual level numbers may be discontiguous if any of the objects enable is_silent.

recurse coderef extra

Iterates the beats of the set and recurses through every next for each beat, calling the coderef unless is_silent is true for the object. extra gets passed along to the callback coderef.

The coderef is passed three arguments. First, the Music::RecRhythm object, second, a hash reference containing various parameters (listed below) and finally extra, a scalar that can be whatever you want it to be (reference, object, whatever). The parameters, which are read- write (though probably should not be changed on the fly), include:

beat

The current beat, a member of the set at the given index.

duration

A calculated duration based on the beat and beatfactor such that each next object can be played entirely for each beat of the parent object without getting into fractional durations. Hopefully.

index

Index of the current beat in the set, numbered from 0 on up.

level

Level of recursion, 0 for the first level, 1 for the second, and so forth. The level numbers will have gaps if is_silent is set.

next

The next object, or undef should this be the lowest level of recursion. Probably should not be changed on the fly.

set

Array reference containing the beats of the current set, of which beat is the current at index index.

validate_set set

Class method. Checks whether a set really is a list of positive numbers (the int truncation is done elsewhere). The empty set is not allowed. Used internally by the set attribute.

Music::RecRhythm->validate_set("cat")      # 0
Music::RecRhythm->validate_set([qw/1 2/])  # 1

BUGS

Reporting Bugs

Please report any bugs or feature requests to bug-music-recrhythm at rt.cpan.org, or through the web interface at http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Music-RecRhythm.

Patches might best be applied towards:

https://github.com/thrig/Music-RecRhythm

Known Issues

Loops created with next calls will run forever as beatproduct, levels, and recurse do not check for loops created by next calls. If this is a risk for generated code, wrap these calls with alarm to abort them should they run for too long (or add loop detection somehow).

SEE ALSO

MIDI or MIDI::Simple may assist in the callback code to produce MIDI during the recursion. Consult the eg/ and t/ directories under this module's distribution for example code.

AUTHOR

thrig - Jeremy Mates (cpan:JMATES) <jmates at cpan.org>

COPYRIGHT AND LICENSE

Copyright (C) 2016 by Jeremy Mates

This program is free software; you can redistribute it and/or modify it under the terms of the the Artistic License (2.0). You may obtain a copy of the full license at:

http://www.perlfoundation.org/artistic_license_2_0