1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179
|
local ltabfill = pd.Class:new():register("ltabfill")
-- AG: Lua 5.2 compatibility (cf. https://stackoverflow.com/a/14554565)
if not setfenv then -- Lua 5.2
-- based on http://lua-users.org/lists/lua-l/2010-06/msg00314.html
-- this assumes f is a function
local function findenv(f)
local level = 1
repeat
local name, value = debug.getupvalue(f, level)
if name == '_ENV' then return level, value end
level = level + 1
until name == nil
return nil end
getfenv = function (f) return(select(2, findenv(f)) or _G) end
setfenv = function (f, t)
local level = findenv(f)
if level then debug.setupvalue(f, level, t) end
return f end
end
if not loadstring then -- Lua 5.3
loadstring = load
end
local function sandbox(e, f, x) -- only supports unary f() with one return
local g = getfenv(f)
setfenv(f, e)
local r = f(x)
setfenv(f, g)
return r
end
local ltabfill_globals = {
abs = math.abs,
acos = math.acos,
asin = math.asin,
atan = math.atan,
atan2 = math.atan2,
ceil = math.ceil,
cos = math.cos,
cosh = math.cosh,
deg = math.deg,
exp = math.exp,
floor = math.floor,
fmod = math.fmod,
log = math.log,
log10 = math.log10,
max = math.max,
min = math.min,
int = function (x) i,f = math.modf(x) ; return i end,
wrap = function (x) i,f = math.modf(x) ; return f end,
pi = math.pi,
pow = math.pow,
rad = math.rad,
sin = math.sin,
sinh = math.sinh,
sqrt = math.sqrt,
tan = math.tan,
tanh = math.tanh,
val = function (s) return (pd.getvalue(s) or 0) end
}
function ltabfill:readexpr(atoms)
local vname = { }
local context = { }
local expr
local i
local k
local v
local j = 2
local inlets
local f
local phase = "table"
for k,v in pairs(ltabfill_globals) do
context[k] = v
end
for i,v in ipairs(atoms) do
if phase == "table" then
if type(v) == "string" then
self.tabname = v
phase = "vars"
else
self:error("ltabfill: table name must be a symbol")
return -1
end
else if phase == "vars" then -- create variables
if v == "->" then
inlets = i - 1
phase = "expr"
expr = ""
else
if type(v) == "string" then
vname[j] = v
context[v] = 0
j = j + 1
else
self:error("ltabfill: variable names must be symbols")
return -1
end
end
else if phase == "expr" then -- build string
expr = expr .. " " .. v
else
self:error("ltabfill: internal error parsing expression")
return -1
end end end
end
f = assert(loadstring("return function(x) return " .. expr .. " end"))()
return inlets, vname, context, f, 0
end
function ltabfill:initialize(sel, atoms)
self.tabname = nil
self.vname = { }
self.context = { }
self.hot = { }
self.f = function (x) return 0 end
function self:in_1_bang()
if self.tabname ~= nil then
local t = pd.Table:new():sync(self.tabname)
if t ~= nil then
local i
local l = t:length()
for i = 1,l do
local y = sandbox(self.context, self.f, (i-1)/l)
t:set(i-1, y)
end
t:redraw()
end
end
end
function self:in_1_symbol(s)
self.tabname = s
if self.hot[1] then self:in_1_bang() end
end
function self:in_1_float(f)
self:error("table name expected, got a float")
end
function self:in_n_float(i, f)
self.context[self.vname[i]] = f
if self.hot[i] then self:in_1_bang() end
end
function self:in_n_symbol(i, s)
self.context[self.vname[i]] = s
if self.hot[i] then self:in_1_bang() end
end
function self:in_n_hot(i, atoms)
if type(atoms[1]) == "number" then
self.hot[i] = atoms[1] ~= 0
else
self:error("hot method expects a float")
end
end
function self:in_1_ltabfill(atoms)
local inlets
local vname
local context
local f
local outlets
inlets, vname, context, f, outlets = self:readexpr(atoms)
if (inlets == self.inlets) and (outlets == self.outlets) then
self.vname = vname
self.context = context
self.f = f
else
self:error("new expression has different inlet/outlet count")
end
end
self.inlets, self.vname, self.context, self.f, self.outlets = self:readexpr(atoms)
if self.inlets < 1 then
pd.post("ltabfill: error: would have no inlets")
return false
end
for i = 1,self.inlets,1 do
self.hot[i] = i == 1
end
return true
end
|