Module:MOS: Difference between revisions
added is_root_mos, under the reasoning that this came up enough times to justify adding it |
mNo edit summary |
||
| (8 intermediate revisions by 2 users not shown) | |||
| Line 1: | Line 1: | ||
-- Module | -- This module follows [[User:Ganaram inukshuk/Provisional style guide for Lua]] | ||
local et = require("Module:ET") | |||
local rat = require("Module:Rational") | |||
local utils = require("Module:Utils") | |||
local p = {} | local p = {} | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
------------------------------ | ----------------------------- MOS-CREATING FUNCTIONS --------------------------- | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
-- Create a new mos as a table containing the counts for large and small steps, | |||
-- plus the equave. | |||
-- Create a new mos | |||
function p.new(nL, ns, equave) | function p.new(nL, ns, equave) | ||
local nL = nL or 5 | local nL = nL or 5 | ||
| Line 64: | Line 20: | ||
end | end | ||
-- Parse a mos from its scalesig. | -- Parse a mos from its scalesig "xL ys<p/q>" or "xL ys (p/q-equivalent)". | ||
-- If no equave "p/q" is provided, it's assumed to be 2/1-equivalent. | |||
function p.parse(unparsed) | function p.parse(unparsed) | ||
local nL, ns, equave = unparsed:match("^(%d+)[Ll] | local nL, ns, equave = unparsed:match("^(%d+)[Ll].-(%d+)[Ss]%s*(.*)$") | ||
nL = tonumber(nL) | nL = tonumber(nL) | ||
ns = tonumber(ns) | ns = tonumber(ns) | ||
| Line 79: | Line 36: | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
---------------------- | ---------------------- VALIDATION AND CHECKING FUNCTIONS ----------------------- | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
| Line 106: | Line 63: | ||
-- Degenerate mosses (nL 0s or 0L ns) produce a string for its corresponding | -- Degenerate mosses (nL 0s or 0L ns) produce a string for its corresponding | ||
-- et (n-ed-p/q). | -- et (n-ed-p/q). | ||
-- Option to use nbsp is provided using the second param; default is nbsp | -- Option to use nbsp is provided using the second param; default is nbsp. | ||
function p.as_string(mos, use_nbsp) | function p.as_string(mos, use_nbsp) | ||
if p.is_valid(mos) then | if p.is_valid(mos) then | ||
| Line 114: | Line 71: | ||
suffix = "⟨" .. rat.as_ratio(mos.equave):lower() .. "⟩" | suffix = "⟨" .. rat.as_ratio(mos.equave):lower() .. "⟩" | ||
end | end | ||
return | return mos.nL .. "L" .. (use_nbsp and " " or " ") .. mos.ns .. "s" .. suffix | ||
else | else | ||
return | return math.max(mos.nL, mos.ns) .. p.et_suffix(mos) | ||
end | end | ||
end | end | ||
| Line 124: | Line 81: | ||
-- Degenerate mosses (nL 0s or 0L ns) produce a string for its corresponding | -- Degenerate mosses (nL 0s or 0L ns) produce a string for its corresponding | ||
-- et (n-ed-p/q). | -- et (n-ed-p/q). | ||
-- Option to use nbsp is provided using the second param; default is nbsp | -- Option to use nbsp is provided using the second param; default is nbsp. | ||
function p.as_long_string(mos, use_nbsp) | function p.as_long_string(mos, use_nbsp) | ||
if p.is_valid(mos) then | if p.is_valid(mos) then | ||
| Line 132: | Line 89: | ||
suffix = (use_nbsp and " " or " ") .. string.format("(%s-equivalent)", rat.as_ratio(mos.equave):lower()) | suffix = (use_nbsp and " " or " ") .. string.format("(%s-equivalent)", rat.as_ratio(mos.equave):lower()) | ||
end | end | ||
return | return mos.nL .. "L" .. (use_nbsp and " " or " ") .. mos.ns .. "s" .. suffix | ||
else | else | ||
return | return math.max(mos.nL, mos.ns) .. p.et_suffix(mos) | ||
end | end | ||
end | end | ||
| Line 147: | Line 104: | ||
return string.format("[[%s]]", link) | return string.format("[[%s]]", link) | ||
else | else | ||
return string.format("[[%s | %s]]", link, text) | return string.format("[[%s|%s]]", link, text) | ||
end | end | ||
end | end | ||
| Line 207: | Line 164: | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
-- Find the parent mos of a mos. May return invalid mosses (nL 0s), meant | -- Find the parent mos of a mos. May return invalid mosses (nL 0s), meant to | ||
-- | -- represent equal divisions of the octave (or arbitrary equave). | ||
function p.parent(mos) | function p.parent(mos) | ||
return p.new(math.min(mos.nL, mos.ns), math.abs(mos.nL-mos.ns), mos.equave) | return p.new(math.min(mos.nL, mos.ns), math.abs(mos.nL-mos.ns), mos.equave) | ||
| Line 321: | Line 278: | ||
end | end | ||
-- List all unique rotations for a mode, by order of leftward shifts. | -- List all unique rotations for a mode, by order of leftward shifts. Order by | ||
-- rotation will usually give a different order compared to order by brightness, | |||
-- but this is expected if the order isn't by brightness (EG, modmosses). | |||
-- Note: there will always be s/p modes, where s is the number of steps in the | -- Note: there will always be s/p modes, where s is the number of steps in the | ||
-- entered mode, and p is the period of repetition. At most, there will be s | -- entered mode, and p is the period of repetition. At most, there will be s | ||
| Line 331: | Line 290: | ||
local current_mode = mode_string | local current_mode = mode_string | ||
for i = 1, #mode_string do | for i = 1, #mode_string do | ||
if not | if not utils.table_contains(rotations, current_mode) then | ||
table.insert(rotations, current_mode) | table.insert(rotations, current_mode) | ||
end | end | ||
| Line 421: | Line 380: | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
--------------- | --------------- FUNCTIONS FOR GENERATOR AND PERIOD INTERVALS ------------------- | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
-- Compute the bright gen as a vector of L's and s's. | -- Compute the bright gen as a vector of L's and s's. Since all mosstep | ||
-- | -- intervals (excluding the root and period) have two sizes, this returns the | ||
-- | -- large/perfect size. | ||
function p.bright_gen(mos) | function p.bright_gen(mos) | ||
local nL = mos.nL | local nL = mos.nL | ||
| Line 458: | Line 416: | ||
end | end | ||
-- Compute the dark gen as a vector of L's and s's. | -- Compute the dark gen as a vector of L's and s's. Since all mosstep | ||
-- | -- intervals (excluding the root and period) have two sizes, this returns the | ||
-- | -- small/perfect size. | ||
function p.dark_gen(mos) | function p.dark_gen(mos) | ||
local bright_gen = p.bright_gen(mos) | local bright_gen = p.bright_gen(mos) | ||
| Line 467: | Line 424: | ||
end | end | ||
-- Compute the period as a vector of L's and s's. Period intervals only | -- Compute the period as a vector of L's and s's. | ||
-- Period intervals as mossteps only appear as one size. | |||
function p.period(mos) | function p.period(mos) | ||
local gcd = utils._gcd(mos.nL, mos.ns) | local gcd = utils._gcd(mos.nL, mos.ns) | ||
| Line 477: | Line 435: | ||
-- Compute the equave as a vector of L's and s's. | -- Compute the equave as a vector of L's and s's. | ||
-- | -- Equaves as mossteps only appear as one size. For a single-period mos, this | ||
-- the same | -- is the same as p.period(). | ||
function p.equave(mos) | function p.equave(mos) | ||
return { | return { | ||
| Line 487: | Line 445: | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
------------------ | ------------------- FUNCTIONS FOR SINGLE-STEP INTERVALS ------------------------ | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
-- | -- Return the unison as a vector of L's and s's. | ||
-- The unison is denoted by moving up from the root by zero steps, and thus does | -- The unison is denoted by moving up from the root by zero steps, and thus does | ||
-- not need a mos as input. It's basically a zero vector. | -- not need a mos as input. It's basically a zero vector. | ||
| Line 498: | Line 456: | ||
end | end | ||
-- | -- Return the vector for a single chroma. It's a large step minus a small step. | ||
-- Adding or subtracting any interval by this interval changes its "size". | -- Adding or subtracting any interval by this interval changes its "size". | ||
function p.chroma() | function p.chroma() | ||
| Line 504: | Line 462: | ||
end | end | ||
-- | -- Return the vector for an augmented step. It's a large step plus a chroma. | ||
function p.augmented_step() | function p.augmented_step() | ||
return { ["L"] = 2, ["s"] = -1 } | return { ["L"] = 2, ["s"] = -1 } | ||
end | end | ||
-- | -- Return the vector for a single large step. | ||
function p.large_step() | function p.large_step() | ||
return { ["L"] = 1, ["s"] = 0 } | return { ["L"] = 1, ["s"] = 0 } | ||
end | end | ||
-- | -- Return the vector for a single small step. | ||
function p.small_step() | function p.small_step() | ||
return { ["L"] = 0, ["s"] = 1 } | return { ["L"] = 0, ["s"] = 1 } | ||
end | end | ||
-- | -- Return the vector for a diminished step. It's a small step minus a chroma. | ||
function p.diminished_step() | function p.diminished_step() | ||
return { ["L"] = -1, ["s"] = 2 } | return { ["L"] = -1, ["s"] = 2 } | ||
| Line 533: | Line 491: | ||
end | end | ||
-- Compute an arbitrary mos interval as a vector of L's and s's. | -- Compute an arbitrary mos interval as a vector of L's and s's. Params: | ||
-- | -- - step_count: the number of steps subtended by the mosstep. | ||
-- - size_offset: denotes whether to return the large size (0) or the small | |||
- | -- size (-1) (or if this is a period interval, the diminished size). Values | ||
-- other than 0 or 1 represent alterations by multiple chromas, such as | |||
-- | -- augmented (1) or diminished (-2). | ||
-- | |||
-- | |||
-- | |||
- | |||
function p.interval_from_mos(mos, step_count, size_offset) | function p.interval_from_mos(mos, step_count, size_offset) | ||
local size_offset = size_offset or 0 -- Optional param; defaults to large size | local size_offset = size_offset or 0 -- Optional param; defaults to large size | ||
| Line 591: | Line 543: | ||
------------------------------- COUNT FUNCTIONS -------------------------------- | ------------------------------- COUNT FUNCTIONS -------------------------------- | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
-- Given a mos, return the number of steps. | |||
function p.step_count(mos) | |||
return mos.nL + mos.ns | |||
end | |||
-- Given a mos, compute the number of steps in its bright gen (L's plus s's). | -- Given a mos, compute the number of steps in its bright gen (L's plus s's). | ||
| Line 608: | Line 565: | ||
end | end | ||
-- | -- TODO: deprecate this since "equave_step_count" is redundant and longer than | ||
-- "step count". | |||
function p.equave_step_count(mos) | function p.equave_step_count(mos) | ||
return mos.nL + mos.ns | return mos.nL + mos.ns | ||
| Line 629: | Line 587: | ||
-- perfect size (for period/root/equave intervals). This requires the mos as | -- perfect size (for period/root/equave intervals). This requires the mos as | ||
-- input. | -- input. | ||
-- | -- size_offset denotes whether to count chromas from the large size; changing | ||
-- this to -1 counts chromas from the small size. Like size_offset for | |||
-- | -- interval_from_mos, this can be used to denote altered mossteps (augmented, | ||
-- | -- diminished, etc). | ||
-- | |||
function p.interval_chroma_count(interval, mos, size_offset) | function p.interval_chroma_count(interval, mos, size_offset) | ||
local size_offset = size_offset or 0 -- Default of 0. | local size_offset = size_offset or 0 -- Default of 0. | ||
| Line 644: | Line 600: | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
--------------- INTERVAL ARITHMETIC AND MANIPULATION FUNCTIONS ----------------- | |||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
| Line 663: | Line 619: | ||
end | end | ||
-- | -- Stack an interval, or repeatedly add the same interval to itself. | ||
function p.interval_mul(interval, amt) | function p.interval_mul(interval, amt) | ||
return { | return { | ||
| Line 677: | Line 633: | ||
interval_1["s"] == interval_2["s"] | interval_1["s"] == interval_2["s"] | ||
end | end | ||
-- Given an interval vector and a mos, find its period complement. This is the | -- Given an interval vector and a mos, find its period complement. This is the | ||
| Line 734: | Line 687: | ||
---------------------------- EQUAL-TUNING FUNCTIONS ---------------------------- | ---------------------------- EQUAL-TUNING FUNCTIONS ---------------------------- | ||
-------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- | ||
-- Given a mos and a step ratio, return an equal tuning (or equal division). | -- Given a mos and a step ratio, return an equal tuning (or equal division). | ||
-- The step ratio is entered as a 2-element array to allow non-simplified | -- The step ratio is entered as a 2-element array to allow non-simplified | ||