Module:Ups and downs notation: Difference between revisions
CompactStar (talk | contribs) No edit summary |
ArrowHead294 (talk | contribs) mNo edit summary |
||
| (42 intermediate revisions by 5 users not shown) | |||
| Line 1: | Line 1: | ||
local p = {} | local p = {} | ||
local ET = require("Module:ET") | |||
local | local utils = require("Module:Utils") | ||
local yesno = require("Module:Yesno") | |||
-- Returns a table of note names | -- Returns a table of note names | ||
-- e.g. for 12edo, intended result would be something like: | -- e.g. for 12edo, intended result would be something like: | ||
-- { | -- { | ||
-- [0] = {" | -- [0] = {"D"}, | ||
-- [1] = {" | -- [1] = {"D#", "Eb"}, | ||
-- [2] = {" | -- [2] = {"E"}, | ||
-- [3] = {" | -- [3] = {"F"}, | ||
-- [4] = {" | -- [4] = {"F#", "Gb"}, | ||
-- [5] = {" | -- [5] = {"G"}, | ||
-- [6] = {" | -- [6] = {"G#", "Ab"}, | ||
-- [7] = {" | -- [7] = {"A"}, | ||
-- [8] = {" | -- [8] = {"A#", Bb"}, | ||
-- [9] = {" | -- [9] = {"B"}, | ||
-- [10] = {" | -- [10] = {"C"}, | ||
-- [11] = {" | -- [11] = {"C#", "Db"}, | ||
-- | -- [12] = {"D"} | ||
-- } | -- } | ||
local function round(x) | |||
-- rounds half-integers towards 0. | |||
return x >= 0 and math.ceil(x - 0.5) or math.floor(x + 0.5) | |||
end | |||
function p.get_note_names_table(et, fifth) | function p.get_note_names_table(et, fifth) | ||
local note_names = {} | local note_names = {} | ||
for i = 0,et.size do | for i = 0, et.size do | ||
note_names[i] = {} | note_names[i] = {} | ||
end | end | ||
local | fifth = fifth or math.floor(math.log(3 / 2) / math.log(2) * et.size + 0.5) | ||
0, | local fourth = et.size - fifth -- 4/3 = [2 -1> | ||
local chroma = (fifth * 7) % et.size -- 2187/2048 = [-11 7> | |||
local is_mavila = fifth / et.size < 4 / 7 | |||
if is_mavila then | |||
chroma = et.size - chroma | |||
end | |||
local diatonic_note_idx = { | |||
0, | |||
(fifth * 2) % et.size, | (fifth * 2) % et.size, | ||
(fifth * | (fifth * -3) % et.size, | ||
fourth, | fourth, | ||
fifth, | fifth, | ||
(fifth * 3) % et.size, | (fifth * 3) % et.size, | ||
(fifth * | (fifth * -2) % et.size, | ||
et.size | et.size, | ||
} | } | ||
local | -- Add D dorian scale notes | ||
table.insert(note_names[diatonic_note_idx[1]], "D") | |||
table.insert(note_names[diatonic_note_idx[2]], "E") | |||
table.insert(note_names[diatonic_note_idx[3]], "F") | |||
table.insert(note_names[diatonic_note_idx[4]], "G") | |||
table.insert(note_names[diatonic_note_idx[5]], "A") | |||
table.insert(note_names[diatonic_note_idx[6]], "B") | |||
table.insert(note_names[diatonic_note_idx[7]], "C") | |||
table.insert(note_names[diatonic_note_idx[8]], "D") | |||
local last_diatonic_note = 0 | |||
-- Add sharp/up notes | -- Add sharp/up notes | ||
for i = 0,et.size-1 do | for i = 0, et.size - 1 do | ||
if table_contains( | if utils.table_contains(diatonic_note_idx, i) then | ||
last_diatonic_note = i | |||
else | else | ||
local num_sharps = | local num_double_sharps = 0 | ||
local num_ups = (i - | local num_sharps = round((i - last_diatonic_note) / chroma) | ||
if num_sharps > 1 then | |||
num_double_sharps = math.floor(num_sharps / 2) | |||
num_sharps = num_sharps % 2 | |||
end | |||
local num_ups = (i - last_diatonic_note) % chroma | |||
local num_downs = 0 | |||
if chroma == 0 then | if chroma == 0 then | ||
num_double_sharps = 0 | |||
num_sharps = 0 | num_sharps = 0 | ||
num_ups = (i - | num_ups = (i - last_diatonic_note) | ||
end | |||
if num_ups > chroma / 2 then | |||
num_downs = chroma - num_ups | |||
num_ups = 0 | |||
end | end | ||
local | local last_diatonic_names = note_names[last_diatonic_note] | ||
for j = | for j = 1, #last_diatonic_names do | ||
table.insert(note_names[i], string.rep("^", num_ups) .. | if num_downs >= 3 then | ||
table.insert( | |||
note_names[i], | |||
"v<sup>" | |||
.. num_downs | |||
.. "</sup>" | |||
.. last_diatonic_names[j] | |||
.. string.rep("♯", num_sharps) | |||
.. string.rep("𝄪", num_double_sharps) | |||
) | |||
elseif num_downs > 0 then | |||
table.insert( | |||
note_names[i], | |||
string.rep("v", num_downs) | |||
.. last_diatonic_names[j] | |||
.. string.rep("♯", num_sharps) | |||
.. string.rep("𝄪", num_double_sharps) | |||
) | |||
elseif num_ups >= 3 then | |||
table.insert( | |||
note_names[i], | |||
"^<sup>" | |||
.. num_ups | |||
.. "</sup>" | |||
.. last_diatonic_names[j] | |||
.. string.rep("♯", num_sharps) | |||
.. string.rep("𝄪", num_double_sharps) | |||
) | |||
else | |||
table.insert( | |||
note_names[i], | |||
string.rep("^", num_ups) | |||
.. last_diatonic_names[j] | |||
.. string.rep("♯", num_sharps) | |||
.. string.rep("𝄪", num_double_sharps) | |||
) | |||
end | |||
end | end | ||
end | end | ||
end | end | ||
last_diatonic_note = et.size | |||
-- Add flat/down notes | -- Add flat/down notes | ||
for i = et.size-1,0,-1 do | for i = et.size - 1, 0, -1 do | ||
if table_contains( | if utils.table_contains(diatonic_note_idx, i) then | ||
last_diatonic_note = i | |||
else | else | ||
local num_flats = | local num_flats = round((last_diatonic_note - i) / chroma) | ||
local num_downs = ( | local num_downs = (last_diatonic_note - i) % chroma | ||
local num_ups = 0 | |||
if chroma == 0 then | if chroma == 0 then | ||
num_flats = 0 | num_flats = 0 | ||
num_downs = ( | num_downs = (last_diatonic_note - i) | ||
end | end | ||
if num_downs > chroma / 2 then | |||
for j = | num_ups = chroma - num_downs | ||
num_downs = 0 | |||
table.insert(note_names[i], | end | ||
local last_diatonic_names = note_names[last_diatonic_note] | |||
for j = 1, #last_diatonic_names do | |||
if num_ups >= 3 then | |||
table.insert( | |||
note_names[i], | |||
"^<sup>" .. num_ups .. "</sup>" .. last_diatonic_names[j] .. string.rep("♭", num_flats) | |||
) | |||
elseif num_ups > 0 then | |||
table.insert( | |||
note_names[i], | |||
string.rep("^", num_ups) .. last_diatonic_names[j] .. string.rep("♭", num_flats) | |||
) | |||
elseif num_downs >= 3 then | |||
table.insert( | |||
note_names[i], | |||
"v<sup>" .. num_downs .. "</sup>" .. last_diatonic_names[j] .. string.rep("♭", num_flats) | |||
) | |||
else | |||
table.insert( | |||
note_names[i], | |||
string.rep("v", num_downs) .. last_diatonic_names[j] .. string.rep("♭", num_flats) | |||
) | |||
end | |||
end | end | ||
end | end | ||
end | end | ||
return note_names | return note_names | ||
end | end | ||
| Line 112: | Line 180: | ||
function p.ups_and_downs_note_name(frame) | function p.ups_and_downs_note_name(frame) | ||
local et = ET.parse(frame.args["tuning"]) or ET.parse("12edo") | local et = ET.parse(frame.args["tuning"]) or ET.parse("12edo") | ||
local debugg = yesno(frame.args["debug"]) | |||
local fifth = tonumber(frame.args["fifth"]) | |||
local step = tonumber(frame.args["step"]) | local step = tonumber(frame.args["step"]) | ||
local result = table.concat(p.get_note_names_table(et, fifth)[step], ", "):sub(0, -1) | |||
if debugg == true then | |||
result = "<syntaxhighlight lang=\"wikitext\">" .. result .. "</syntaxhighlight>" | |||
end | |||
return frame:preprocess(result) | |||
end | end | ||
return p | return p | ||