-
Notifications
You must be signed in to change notification settings - Fork 21
/
Copy pathchartmap.lua
429 lines (418 loc) · 16.4 KB
/
chartmap.lua
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
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
-- Copyright (C) 2017 Deyan Dobromirov
-- A chart mapping functionalities library
if not debug.getinfo(3) then
print("This is a module to load with `local chartmap = require('chartmap')`.")
os.exit(1)
end
local common = require("common")
local type = type
local tonumber = tonumber
local tostring = tostring
local setmetatable = setmetatable
local math = math
local isNil = common.isNil
local toBool = common.toBool
local getPick = common.getPick
local isTable = common.isTable
local getClamp = common.getClamp
local getRound = common.getRound
local logStatus = common.logStatus
local chartmap = {}
--[[
* newInterval: Class that maps one interval onto another
* sName > A proper name to be identified as
* nL1 > Lower value first border
* nH1 > Higher value first border
* nL2 > Lower value second border
* nH2 > Higher value second border
]]--
local metaInterval = {}
metaInterval.__index = metaInterval
metaInterval.__type = "chartmap.interval"
metaInterval.__tostring = function(oInterval) return oInterval:getString() end
local function newInterval(sName, nL1, nH1, nL2, nH2)
local self, mVal, mNm = {}, 0, tostring(sName or "")
local mL1, mH1 = (tonumber(nL1) or 0), (tonumber(nH1) or 0)
local mL2, mH2 = (tonumber(nL2) or 0), (tonumber(nH2) or 0)
if(mL1 == mH1) then
return logStatus(metaInterval.__type.."("..mNm.."): Bad input bounds", self) end
if(mL2 == mH2) then
return logStatus(metaInterval.__type.."("..mNm.."): Bad output bounds", self) end
setmetatable(self, metaInterval)
function self:getName() return mNm end
function self:setName(sName) mNm = tostring(sName or "N/A") end
function self:getValue() return mVal end
function self:getBorderIn() return mL1, mH1 end
function self:setBorderIn(nL1, nH1) mL1, mH1 = (tonumber(nL1) or 0), (tonumber(nH1) or 0) end
function self:getBorderOut() return mL2, mH2 end
function self:setBorderOut(nL2, nH2) mL2, mH2 = (tonumber(nL2) or 0), (tonumber(nH2) or 0) end
function self:getString() return "["..metaInterval.__type.."] "..mNm.." {"..mL1..","..mH1.."} >> {"..mL2..","..mH2.."}" end
function self:Convert(nVal, bRev)
local val = tonumber(nVal); if(not val) then
return logStatus(metaInterval.__type..".Convert("..mNm.."): Source <"..tostring(nVal).."> NaN", self) end
mVal = common.getRemap(val, mH1, mL1, mH2, mL2, bRev); return self
end
return self
end
--[[
* newTracer: Class that plots a process variable
* sName > A proper name to be identified as
]]--
local metaTracer = {}
metaTracer.__index = metaTracer
metaTracer.__type = "chartmap.tracer"
metaTracer.__tostring = function(oTracer) return oTracer:getString() end
local function newTracer(sName)
local self = {}; setmetatable(self, metaTracer)
local mName = tostring(sName or "")
local mValO, mValN, mnPs = 0, 0, 2
local mTimO, mTimN = 0, 0
local mPntN = {x=0,y=0}
local mPntO = {x=0,y=0}
local mCach = {Size = 0, Draw = 0}
local mMatX, mMatY
local enDraw = false
function self:setSizeVtx(vS)
mnPs = math.floor(tonumber(vS) or 0)
if(mnPs <= 0) then mnPs = 0 end; return self
end
function self:Cache(nN)
mCach.Size = (tonumber(nN) or 0)
if(mCach.Size > 0) then
for ID = 1, mCach.Size do
if(not mCach[ID]) then mCach[ID] = {x=0, y=0} end
end
else mCach.Size = 0 end; return self
end
function self:Move(dX, dY)
if(mCach.Size > 0) then
local dX = (tonumber(dX) or 0)
local dY = (tonumber(dY) or 0)
for ID = 1, mCach.Size do
if(mCach[ID]) then
mCach[ID].x = mCach[ID].x + dX
mCach[ID].y = mCach[ID].y + dY
end
end
end; return self
end
function self:getCache() return mCach end
function self:getCacheSZ() return mCach.Size, mCach.Draw end
function self:getString() return "["..metaTracer.__type.."] "..mName end
function self:getValue() return mTimN, mValN end
function self:getChart() return mPntN.x, mPntN.y end
function self:getInterval() return mMatX, mMatY end
function self:setInterval(oIntX, oIntY)
mMatX = oIntX; if(getmetatable(mMatX) ~= metaInterval) then mMatX = nil
logStatus(metaTracer.__type..".setInterval: X object invalid", nil) end
mMatY = oIntY; if(getmetatable(mMatY) ~= metaInterval) then mMatY = nil
logStatus(metaTracer.__type..".setInterval: Y object invalid", nil) end
return self
end
function self:Reset()
mCach.Size, mCach.Draw = 0, 0
enDraw, mValO, mValN = false,0,0
mPntN.x, mPntN.y, mPntO.x, mPntO.y = 0,0,0,0
return self
end
function self:Write(nTime, nVal)
mValO, mValN = mValN, nVal
mTimO, mTimN = mTimN, nTime
mPntO.x, mPntO.y = mPntN.x, mPntN.y
if(mMatX) then mPntN.x = mMatX:Convert(nTime):getValue()
else mPntN.x = nTime end
if(mMatY) then mPntN.y = mMatY:Convert(mValN):getValue()
else mPntN.y = mValN end
if(mCach.Size > 0) then
for ID = mCach.Size, 2, -1 do
mCach[ID].x, mCach[ID].y = mCach[ID-1].x, mCach[ID-1].y
end
mCach[1].x, mCach[1].y = mPntN.x, mPntN.y
if(mCach.Draw < mCach.Size) then
mCach.Draw = mCach.Draw + 1 end
end
return self
end
function self:Draw(cCol, vSz)
if(enDraw) then pncl(cCol)
local nSz = math.floor(tonumber(vSz) or mnPs)
local nsE = ((nSz > 0) and ((2 * nSz) + 1) or nil)
if(mCach.Size == 0) then
line(mPntO.x,mPntO.y,mPntN.x,mPntN.y)
if(nsE) then rect(mPntO.x-nSz,mPntO.y-nSz,nsE,nsE) end
else local xyE = mCach[mCach.Draw]
for ID = 2, mCach.Draw do
local vS, vE = mCach[ID-1], mCach[ID]
line(vS.x,vS.y,vE.x,vE.y)
if(nsE) then rect(vS.x-nSz,vS.y-nSz,nsE,nsE) end
end
if(nsE) then rect(xyE.x-nSz,xyE.y-nSz,nsE,nsE) end
end
else enDraw = true end; return self
end
return self
end
--[[
* CoordSys: Class that plots coordinate axises by scale
* sName > A proper name to be identified as
]]--
local metaScope = {}
metaScope.__index = metaScope
metaScope.__type = "chartmap.scope"
metaScope.__tostring = function(oCoordSys) return CoordSys:getString() end
local function newScope(sName)
local mName, mnPs = tostring(sName or ""), 2
local self = {}; setmetatable(self, metaScope)
local mdX, mdY, mnW, mnH, minX, maxX, minY, maxY, midX, midY, moiX, moiY
local mclMid, mcldXY, pxX, pxY = colr(0,0,0), colr(200,200,200)
local mclPos, mclOrg, mclDir = colr(255,0,0), colr(0,255,0), colr(0,0,255)
function self:setSizeVtx(vS)
mnPs = math.floor(tonumber(vS) or 0)
if(mnPs <= 0) then mnPs = 0 end; return self
end
function self:setDelta(nX, nY)
mdX, mdY = (tonumber(nX) or 0), (tonumber(nY) or 0)
if(isNil(minX) and isNil(maxX) and isNil(minY) and isNil(maxY)) then
return logStatus(metaScope.__type..".setDelta: Call /setBorder/ before /setDelta/", nil)
else
if(isNil(mnW) or isNil(mnH)) then
return logStatus(metaScope.__type..".setDelta: Call /setSize/ before /setDelta/", nil)
else
pxX = mnW / (math.abs(maxX - minX) / mdX)
pxY = mnH / (math.abs(maxY - minY) / mdY)
end
end
if(mdX == 0 or mdY == 0) then
return logStatus(metaScope.__type..".setDelta: Delta invalid", nil) end
return self
end
function self:setBorder(nX, xX, nY, xY)
minX, maxX = (tonumber(nX) or 0), (tonumber(xX) or 0)
minY, maxY = (tonumber(nY) or 0), (tonumber(xY) or 0)
if(isNil(nX) and isNil(xX) and isNil(nY) and isNil(xY)) then
logStatus(metaScope.__type..".setBorder: Using intervals")
if(isNil(moiX) or isNil(moiY)) then
return logStatus(metaScope.__type..".setBorder: Call /setInterval/ before /setBorder/", nil)
else
minX, maxX = moiX:getBorderIn()
minY, maxY = moiY:getBorderIn()
end
end
if(minX == maxX or minY == maxY) then
return logStatus(metaScope.__type..".setBorder: Border invalid", nil) end
midX, midY = (minX + ((maxX - minX) / 2)), (minY + ((maxY - minY) / 2))
return self
end
function self:setSize(nW, nH)
mnW, mnH = (tonumber(nW) or 0), (tonumber(nH) or 0)
if(isNil(nW) and isNil(nH)) then
logStatus(metaScope.__type..".setSize: Using intervals")
if(isNil(moiX) or isNil(moiY)) then
return logStatus(metaScope.__type..".setSize: Call /setInterval/ before /setSize/", nil)
else
mnW = math.max(moiX:getBorderOut())
mnH = math.max(moiY:getBorderOut())
end
end
if(mnW <= 0 or mnH <= 0) then
return logStatus(metaScope.__type..".setSize: Size invalid", nil) end; return self
end
function self:Update()
if(isNil(moiX) or isNil(moiY)) then
return logStatus(metaScope.__type..".Update: Skip", nil) end
return self:setBorder():setSize()
end
function self:setInterval(intX, intY)
moiX = intX; if(getmetatable(moiX) ~= metaInterval) then
return logStatus(metaScope.__type..".setInterval: X object invalid", nil) end
moiY = intY; if(getmetatable(moiY) ~= metaInterval) then
return logStatus(metaScope.__type..".setInterval: Y object invalid", nil) end
return self
end
function self:getInterval() return moiX, moiY end
function self:setColorAxis(clMid) mclMid = (clMid or colr(0,0,0)); return self end
function self:setColorDXY(clDXY) mcldXY = (clDXY or colr(200,200,200)); return self end
function self:setColorPos(clPos) mclPos = (clPos or colr(255,0,0)); return self end
function self:setColorOrg(clOrg) mclOrg = (clOrg or colr(0,255,0)); return self end
function self:setColorDir(clDir) mclDir = (clDir or colr(0,0,255)); return self end
function self:setColor(clMid, clDXY, clPos, clOrg, clDir)
mclMid = (clMid or colr(0,0,0))
mclPos = (clPos or colr(255,0,0))
mclOrg = (clOrg or colr(0,255,0))
mclDir = (clDir or colr(0,0,255))
mcldXY = (clDXY or colr(200,200,200))
return self
end
function self:Draw(bMx, bMy, bGrd, bZer)
local xe = moiX:Convert(midX):getValue()
local ye = moiY:Convert(midY):getValue()
if(bGrd) then pncl(mcldXY); local nK
nK = 0; for x = midX, maxX, mdX do
local xp = moiX:Convert(midX + nK * mdX):getValue()
local xm = moiX:Convert(midX - nK * mdX):getValue()
nK = nK + 1; if(not bMy or x ~= midX) then
line(xp, 0, xp, mnH); line(xm, 0, xm, mnH) end
end
nK = 0; for y = midY, maxY, mdY do
local yp = moiY:Convert(midY + nK * mdY):getValue()
local ym = moiY:Convert(midY - nK * mdY):getValue()
nK = nK + 1; if(not bMx or y ~= midY) then
pncl(mcldXY); line(0, yp, mnW, yp); line(0, ym, mnW, ym) end
end
end
if(xe and bMx) then pncl(mclMid); line(0, ye, mnW, ye) end
if(ye and bMy) then pncl(mclMid); line(xe, 0, xe, mnH) end
if(bZer) then
local zx = moiX:Convert(0):getValue()
local zy = moiY:Convert(0):getValue()
if(zx and zy) then
pncl(mclOrg)
line(0, zy, mnW, zy)
line(zx, 0, zx, mnH)
end
end
return self
end
function self:drawComplex(xyP, xyO, bTx, clP, clO)
local ox, oy, px, py = 0, 0, 0, 0
if(xyO) then ox, oy = xyO:getParts() end
px, py = xyP:getParts()
ox = moiX:Convert(ox):getValue()
oy = moiY:Convert(oy):getValue()
px = moiX:Convert(px):getValue()
py = moiY:Convert(py):getValue()
if(mnPs > 0) then local sz = 2*mnPs+1
pncl(clO or mclOrg); rect(ox-mnPs,oy-mnPs,sz,sz)
pncl(clP or mclPos); rect(px-mnPs,py-mnPs,sz,sz)
else
pncl(clO or mclOrg); pixl(ox, oy)
pncl(clP or mclPos); pixl(px, py)
end
pncl(mclDir); line(px, py, ox, oy)
if(bTx) then pncl(mclDir)
local nA = xyP:getSub(xyO):getAngDeg()+90
text(tostring(xyP:getRound(0.001)),nA,px,py)
end return self
end
function self:drawComplexLine(xyS, xyE, bTx)
return self:drawComplex(xyS, xyE, bTx, mclOrg, mclOrg)
end
function self:drawComplexText(xyP, sTx, vA, bSp)
local sMs, cP = tostring(sTx), xyP:getRound(0.001)
local px, py = cP:getParts()
local nA = xyP:getAngDeg()
nA = (vA and (tonumber(vA) or 0) or nA)
px = moiX:Convert(px):getValue()
py = moiY:Convert(py):getValue()
sMs = sMs..tostring(bSp and cP or "")
text(sMs,nA,px,py); return self
end
function self:drawComplexPoint(xyP, clNew, bTx, nA)
local px, py = xyP:getParts()
px = moiX:Convert(px):getValue()
py = moiY:Convert(py):getValue()
if(mnPs > 0) then local sz = 2*mnPs+1
pncl(clNew or mclPos); rect(px-mnPs,py-mnPs,sz,sz)
else pncl(clNew or mclPos); pixl(px, py) end
if(bTx) then pncl(mclDir)
local nA = (tonumber(nA) or (xyP:getAngDeg()+90))
text(tostring(xyP:getRound(0.001)),nA,px,py)
end; return self
end
function self:drawComplexPolygon(tV, bTx, clP, clO, nN, bO)
if(not isTable(tV)) then
return logStatus(metaScope.__type..".drawComplexPolygon: Skip", self) end
local nL = #tV -- Store the length to avoid counting again
local nE = getClamp(math.floor(tonumber(nN) or nL), 1, nL)
for iD = 1, nE do local cS, cE = (tV[iD+1] or tV[1]), tV[iD]
self:drawComplex(cS, cE, bTx, clP, clO)
end
if(not bO and nE < nL) then
self:drawComplex(tV[1], tV[nE+1], bTx, clP, clO) end
return self
end
function self:drawComplexPointSet(tV, clNew, bTx, nN)
if(not isTable(tV)) then
return logStatus(metaScope.__type..".drawComplexPointSet: Skip", self) end
local nL = #tV -- Store the length to avoid counting again
local nE = getClamp(math.floor(tonumber(nN) or nL), 1, nL)
for iD = 1, nE do
self:drawComplexPoint(tV[iD], clNew, bTx)
end; return self
end
function self:drawPointXY(nX, nY, clNew)
local px = moiX:Convert(nX):getValue()
local py = moiY:Convert(nY):getValue()
if(mnPs > 0) then local sz = 2*mnPs+1
pncl(clNew or mclPos); rect(px-mnPs,py-mnPs,sz,sz)
else pncl(clNew or mclPos); pixl(px, py) end
return self
end
function self:drawGraph(tY, tX)
if(not isTable(tY)) then
return logStatus(metaScope.__type..".drawGraph: Skip", self) end
local ntY, bX, ntX, toP = #tY, false
if(isTable(tX)) then ntX, bX = #tX, true
if(ntX ~= ntY) then
logStatus(metaScope.__type..".drawGraph: Shorter <" ..ntX..","..ntY..">")
toP = math.min(ntX, ntY) else toP = ntY end
else toP, bX = ntY, false end
local trA, vX = newTracer("drawGraph"):setInterval(moiX, moiY)
for iD = 1, toP do
vX = getPick(bX, tX and tX[iD], iD)
trA:Write(vX, tY[iD]):Draw(mclDir, mnPs)
end; self:drawPointXY(vX, tY[toP], mclDir)
return self
end
function self:drawStem(tY, tX)
if(not isTable(tY)) then
return logStatus(metaScope.__type..".drawStem: Skip", self) end
local ntY, bX, ntX, toP = #tY, false
if(isTable(tX)) then ntX, bX = #tX, true
if(ntX ~= ntY) then
logStatus(metaScope.__type..".drawStem: Shorter <" ..ntX..","..ntY..">")
toP = math.min(ntX, ntY) else toP = ntY end
else toP, bX = ntY, false end; local vX
local zY = moiY:Convert(0):getValue()
for iD = 1, toP do
vX = getPick(bX, tX and tX[iD], iD)
local nX = moiX:Convert(vX):getValue()
local nY = moiY:Convert(tY[iD]):getValue()
pncl(mclDir); line(nX, nY, nX, zY)
if(mnPs > 0) then local sz = 2*mnPs+1
pncl(mclPos); rect(nX-mnPs,nY-mnPs,sz,sz)
else pncl(mclPos); pixl(nX, nY) end
end; self:drawPointXY(vX, tY[toP], mclPos)
return self
end
function self:drawOval(nX, nY, rX, rY)
local px, py = (tonumber(nX) or 0), (tonumber(nY) or 0)
px = moiX:Convert(px):getValue()
py = moiY:Convert(py):getValue()
local rx, ry = (tonumber(rX) or 0), (tonumber(rY) or 0)
rx, ry = (rX * (pxX / mdX)), (rY * (pxY / mdY))
if(mnPs > 0) then local sz = 2*mnPs+1
pncl(mclOrg); rect(px-mnPs,py-mnPs,sz,sz)
else pncl(mclOrg); pixl(px, py) end
pncl(mclDir); oval(px, py, rx, ry); return self
end
function self:drawComplexOval(xyP, rX, rY)
return self:drawOval(xyP:getReal(), xyP:getImag(), rX, rY)
end
function self:drawCircle(nX, nY, rR)
return self:drawOval(nX, nY, rR, rR)
end
function self:drawComplexCircle(xyP, rR)
return self:drawOval(xyP:getReal(), xyP:getImag(), rR, rR)
end
function self:getString() return "["..metaScope.__type.."] "..mName end
return self
end
function chartmap.New(sType, ...)
local sType = "chartmap."..tostring(sType or "")
if(sType == metaInterval.__type) then return newInterval(...) end
if(sType == metaTracer.__type) then return newTracer(...) end
if(sType == metaScope.__type) then return newScope(...) end
return logStatus("chartmap.New: Object invalid <"..sType..">",nil)
end
return chartmap