mirror of
https://github.com/pascallanger/DIY-Multiprotocol-TX-Module.git
synced 2025-02-04 13:48:13 +00:00
b2f8f482bb
* #751 DSM Enhancements #751 DSM Forward Programming Enhancements (New GUI, etc) * Make both work on EdgeTx and OpenTX * #751 Turn OFF simulation by default Distribution code with RX simulation OFF Simulation should be only for Development * #751 Update Readme Documentation Updated the Readme.txt documentation and removed compiled luac file that was check in by mistake * #751 Fix problems With Reset RX 1. Fix problem when trying to Factory Reset. Enter Bind Mode. Save backup, Restore Backup 2. Found a way to advance on the Gyro initial Setup menus.. a bit of a hack, but works. 3. Handle RX resets properly. It needed after initial setup * #751 Cosmetic and Show Orientation Images #751 1. Fix problems when text contradictions between Menu/Line Headers and List Values 2. Show Images of RX orientations 3. Able to Hack getting into Initial Setup and other menus who was failing before 4. Custumize the way Flight Mode reports the Value on Screen * #751 add check for required libraries Add check that the required files in DSMLIB exist * #751 Write documentation about the protocol so that we don't forget later what we know, and enable others to understand the logs and maybe help solve problems. * #766 Change the way to detect that the files exist. now works on both ETX and OTX * #766 Strange Flickering in OTX Strange Flickering happening on OTX. Refreshing the screen on every cycle fixed the problem * #766 Change way of dectecting EdgeTX Change way of detecting OTX in multiple versions: OTX 2.3.14 and 2.3.15 * #766 make editable Gain Values Gains and other settings should be editable even when they are VALUE_NOCHANGING. Flight Mode is an exception that is handled properly. Right align numbers. * #766 More enhacements Added AR630 Make numbers right justified Cleanup some log messages and line types. Updated DSM FWD prog documentation * #751 more cosmetic things 1. Added AR10360T, 2. Simplify way to configured the hack for more receivers. 3. Change some texts on menus to march spektrum 4. Background color in Spektrum theme to match * #751 A few final changes 1. Update channel names to include channel number. i.e: Ch5 (Gear) 2, Fix flight mode display for Heli Receiver 3. i think the unknown lines are to request info about the TX settings * #751 1. Added Warning Screen 2. Correct handling of Unknown lines in Gyro Settings->Initial Setup * #751 New v0.51 version. - Added new menus to configure Model/Wing type. Without it, the initial setup will not work properly. * #751 More fixes on mixers and servo reverse -- Fix problem reversing servos when using vtail/delta mix -- Properly detect ch order of multimodule * #751 Updated channel naming and docs Updated readme documentation Consistent naming of Ch across the code. * #751 Fix message displaying data path * #751 More improvements 1. Much easier to select channels > Ch6 for FMode, Gain and Panic channels 2. B&W version for smaller screens (128x64).. Memory footprint still a problem. 3. Fix a lot typos/misspell/grammar in the documentation * Create DSM_AR636_TextGen.lua Script to show Telemetry TextGen screens for AR636 Receiver. Really useful for BLADE helis using the AR636 Could replace dsmPID.lua Still needs to be ported to smaller screens. * #751 Enhancements for Lua Script tools Enhancements * Delete DSM_AR636_TextGen.lua The TextGen functionality is included in DSM_AR636_Tel.lua. No longer needed * Version 0.54 1. Fix problem with "Attitude Trim" Menu 2. New "MINimalistic" version for radios with very low memory 3. Externalized menu messages shared by all versions. the idea is to allow to translate it into other languages. 4. Correction of TextGen tools to work on black&white radios (some Lua functional differences). TextGen will be working on EdgeTx 2,8.3 --------- Co-authored-by: pascallanger <pascal_langer@yahoo.fr>
604 lines
18 KiB
Lua
604 lines
18 KiB
Lua
local toolName = "TNS|DSM Smart RX Telemetry|TNE"
|
|
---- ######################################################################### #
|
|
---- # License GPLv2: http://www.gnu.org/licenses/gpl-2.0.html #
|
|
---- # #
|
|
---- # This program is free software; you can redistribute it and/or modify #
|
|
---- # it under the terms of the GNU General Public License version 2 as #
|
|
---- # published by the Free Software Foundation. #
|
|
---- # #
|
|
---- # This program is distributed in the hope that it will be useful #
|
|
---- # but WITHOUT ANY WARRANTY; without even the implied warranty of #
|
|
---- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #
|
|
---- # GNU General Public License for more details. #
|
|
---- # #
|
|
---- #########################################################################
|
|
|
|
------------------------------------------------------------------------------
|
|
-- Developer: Francisco Arzu
|
|
|
|
|
|
local DEBUG_ON = false
|
|
--
|
|
|
|
local TEXT_SIZE = 0 -- NORMAL
|
|
local X_COL1_HEADER = 6
|
|
local X_COL1_DATA = 60
|
|
local X_COL2_HEADER = 170
|
|
local X_COL2_DATA = 220
|
|
local Y_LINE_HEIGHT = 20
|
|
local Y_HEADER = 0
|
|
local Y_DATA = Y_HEADER + Y_LINE_HEIGHT*2
|
|
local X_DATA_LEN = 80
|
|
local X_DATA_SPACE = 5
|
|
|
|
|
|
|
|
|
|
local function getPage(iParam)
|
|
-- get page from 0-based index
|
|
-- {0,1,2,3}: cyclic (1), {4,5,6,7}: tail (2)
|
|
local res = (math.floor(iParam/4)==0) and 0 or 1
|
|
return res
|
|
end
|
|
|
|
local function round(v)
|
|
-- round float
|
|
local factor = 100
|
|
return math.floor(v * factor + 0.5) / factor
|
|
end
|
|
|
|
|
|
local function readValue(sensor)
|
|
-- read from sensor, round and return
|
|
local v = getValue(sensor)
|
|
--v = round(v)
|
|
return v
|
|
end
|
|
|
|
local function readValueById(sensor)
|
|
local i = getFieldInfo(sensor)
|
|
if (i==nil) then return nil end
|
|
|
|
local v = getValue(i.id)
|
|
return v
|
|
end
|
|
|
|
|
|
|
|
local function readBatValue(sensor)
|
|
-- read from sensor, round and return
|
|
local v = getValue(sensor)
|
|
if (v==nil) then return v end
|
|
|
|
return string.format("%2.2f",v)
|
|
end
|
|
|
|
local function readActiveParamValue(sensor)
|
|
-- read and return a validated active parameter value
|
|
local v = getValue(sensor)
|
|
if (v<1 or v>8) then
|
|
return -1
|
|
end
|
|
return v
|
|
end
|
|
|
|
|
|
|
|
local function drawFlightLogScreen(event)
|
|
-- draw labels and params on screen
|
|
local h = getValue("Hold")
|
|
local activeParam = h-1 -- H
|
|
|
|
lcd.clear()
|
|
lcd.drawText (X_COL1_HEADER, Y_HEADER, "Flight Log", TEXT_SIZE + INVERS)
|
|
|
|
-- read and return parameters
|
|
local a = getValue("FdeA")
|
|
local b = getValue("FdeB")
|
|
local l = getValue("FdeL")
|
|
local r = getValue("FdeR")
|
|
local f = getValue("FLss")
|
|
|
|
local titles = {[0]="A:", "B:", "L:", "R:", "F:", "H:"}
|
|
local values = {[0]=a,b,l,r,f,h}
|
|
|
|
local y = Y_LINE_HEIGHT+Y_DATA
|
|
|
|
for iParam=0,3 do -- A,B,L,R
|
|
-- highlight selected parameter (rund)
|
|
local attr = ((activeParam%4)==iParam) and INVERS or 0
|
|
-- labels
|
|
local x = X_COL1_HEADER
|
|
local val = titles[iParam]
|
|
lcd.drawText (x, y, val, TEXT_SIZE)
|
|
|
|
-- Values
|
|
val = values[iParam]
|
|
x = X_COL1_DATA + X_DATA_LEN
|
|
if (val~=16384) then -- Active value
|
|
lcd.drawText (x, y, val, attr + TEXT_SIZE + RIGHT)
|
|
end
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
|
|
y = Y_LINE_HEIGHT+Y_DATA
|
|
for iParam=4,5 do -- F, H
|
|
-- labels
|
|
local x = X_COL2_HEADER
|
|
local val = titles[iParam]
|
|
lcd.drawText (x, y, val, TEXT_SIZE + BOLD)
|
|
|
|
-- Values
|
|
val = values[iParam]
|
|
x = X_COL2_DATA + X_DATA_LEN
|
|
lcd.drawText (x, y, val, TEXT_SIZE + RIGHT + BOLD)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
|
|
-- Bat
|
|
y = y + Y_LINE_HEIGHT
|
|
local bat = readBatValue("A2") or "--"
|
|
lcd.drawText (X_COL2_HEADER, y, "Bat:", TEXT_SIZE)
|
|
lcd.drawText (X_COL2_DATA + X_DATA_LEN, y, bat, TEXT_SIZE + RIGHT)
|
|
lcd.drawText (X_COL2_DATA + X_DATA_LEN + X_DATA_SPACE, y, "v", TEXT_SIZE)
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
local function Unsigned_to_SInt16(value)
|
|
if value >= 0x8000 then -- Negative value??
|
|
return value - 0x10000
|
|
end
|
|
return value
|
|
end
|
|
|
|
local function getDegreesValue(sensor)
|
|
local i = getFieldInfo(sensor)
|
|
if (i==nil) then return "-unk-" end
|
|
|
|
local v = getValue(i.id)
|
|
if v==nil then return "---" end
|
|
local vs = Unsigned_to_SInt16(v)
|
|
|
|
return string.format("%0.1f o",vs/10)
|
|
end
|
|
|
|
|
|
local function getDecHexValue(sensor)
|
|
local i = getFieldInfo(sensor)
|
|
if (i==nil) then return "-unk-" end
|
|
|
|
local v = getValue(i.id)
|
|
if v==nil then return "---" end
|
|
local vs = Unsigned_to_SInt16(v)
|
|
|
|
return string.format("%d (0x%04X)",vs,v)
|
|
end
|
|
|
|
local as3xData = {[0]=0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0}
|
|
local function drawAS3XSettings(event, page)
|
|
local s0500 = getDecHexValue("0500")
|
|
local s0502 = getDecHexValue("0502")
|
|
local s0504 = getDecHexValue("0504")
|
|
local s0506 = getDecHexValue("0506")
|
|
local s0508 = getDecHexValue("0508")
|
|
local s050B = getDecHexValue("050B")
|
|
local s050D = getDecHexValue("050D")
|
|
|
|
local d0500 = readValueById("0500") or 0
|
|
local flags = bit32.rshift(d0500,8)
|
|
local state = bit32.band(d0500,0xFF)
|
|
|
|
local flagsMsg=""
|
|
-- flags bits: Safe Envelop, ?, Angle Demand, Stab
|
|
if (bit32.band(flags,0x1)~=0) then flagsMsg=flagsMsg.."AS3X Stab" end
|
|
-- This one, only one should show
|
|
if (bit32.band(flags,0x2)~=0) then flagsMsg=flagsMsg..", Angle Demand"
|
|
elseif (bit32.band(flags,0x8)~=0) then flagsMsg=flagsMsg..", Safe Envelope"
|
|
elseif (bit32.band(flags,0x4)~=0) then flagsMsg=flagsMsg..", AS3X Heading" end
|
|
|
|
local d0502 = readValueById("0502") or 0 -- 0x?F?S
|
|
local fm = bit32.band(bit32.rshift(d0502,8),0xF) -- 0,1,2
|
|
|
|
local axis = bit32.band(d0502,0xF) -- 0=Gains,1=Headings,2=Angle Limits (cointinus iterating to provide all values)
|
|
|
|
local d0504 = readValueById("0504") or 0
|
|
local d0506 = readValueById("0506") or 0
|
|
local d0508 = readValueById("0508") or 0
|
|
|
|
local d0 = bit32.rshift(d0504,8)
|
|
local d1 = bit32.band(d0504,0xFF)
|
|
local d2 = bit32.rshift(d0506,8)
|
|
local d3 = bit32.band(d0506,0xFF)
|
|
local d4 = bit32.rshift(d0508,8)
|
|
local d5 = bit32.band(d0508,0xFF)
|
|
|
|
--axis: 0=Gains+Headings (RG,PG,YG,RH,PH,YH), 1=Safe Gains (R,P,Y),2=Angle Limits(L,R,U,D)
|
|
--Constantly changing from 0..2 to represent different data, thats why we have to store the values
|
|
--in a script/global variable, and not local to the function
|
|
local s = axis*6
|
|
as3xData[s+0] = d0
|
|
as3xData[s+1] = d1
|
|
as3xData[s+2] = d2
|
|
as3xData[s+3] = d3
|
|
as3xData[s+4] = d4
|
|
as3xData[s+5] = d5
|
|
|
|
|
|
lcd.clear()
|
|
lcd.drawText (0,0, "AS3X/SAFE Settings", TEXT_SIZE + INVERS)
|
|
|
|
local y = Y_DATA
|
|
-- Flight Mode
|
|
lcd.drawText (X_COL1_HEADER,y, "FM: "..(fm+1), TEXT_SIZE)
|
|
lcd.drawText (X_COL1_DATA+X_DATA_LEN*0.3,y, "Flags: "..flags, TEXT_SIZE)
|
|
lcd.drawText (X_COL2_HEADER+X_DATA_LEN*0.3,y, "State: "..state, TEXT_SIZE)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, flagsMsg, TEXT_SIZE)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
|
|
if (page==1) then
|
|
lcd.drawText (X_COL1_HEADER+X_DATA_LEN*0.3,y, "AS3X Gains", TEXT_SIZE+BOLD)
|
|
lcd.drawText (X_COL2_HEADER+X_DATA_LEN*0.3,y, "AS3X Headings", TEXT_SIZE+BOLD)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, "Roll:", TEXT_SIZE)
|
|
lcd.drawText (X_COL1_DATA+X_DATA_LEN,y, as3xData[0], TEXT_SIZE + RIGHT) -- Roll G
|
|
lcd.drawText (X_COL2_DATA+X_DATA_LEN,y, as3xData[3], TEXT_SIZE + RIGHT) -- Roll H
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, "Pitch:", TEXT_SIZE)
|
|
lcd.drawText (X_COL1_DATA+X_DATA_LEN,y,as3xData[1], TEXT_SIZE + RIGHT) -- Pitch G
|
|
lcd.drawText (X_COL2_DATA+X_DATA_LEN,y, as3xData[4], TEXT_SIZE + RIGHT) -- Pitch H
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, "Yaw:", TEXT_SIZE)
|
|
lcd.drawText (X_COL1_DATA+X_DATA_LEN,y, as3xData[2], TEXT_SIZE + RIGHT) -- Yaw G
|
|
lcd.drawText (X_COL2_DATA+X_DATA_LEN,y, as3xData[5], TEXT_SIZE + RIGHT) -- Yaw H
|
|
end
|
|
|
|
|
|
if (page==2) then
|
|
local x_data1 = X_COL1_DATA+X_DATA_LEN
|
|
local x_data2 = X_COL2_HEADER+X_DATA_LEN*1.6
|
|
|
|
lcd.drawText (X_COL1_HEADER+X_DATA_LEN*0.3,y, "SAFE Gains", TEXT_SIZE+BOLD)
|
|
lcd.drawText (X_COL2_HEADER+X_DATA_LEN*0.1,y, "Angle Limits", TEXT_SIZE+BOLD)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, "Roll:", TEXT_SIZE)
|
|
lcd.drawText (x_data1,y, as3xData[6], TEXT_SIZE + RIGHT)
|
|
|
|
lcd.drawText (X_COL2_HEADER,y, "Roll R:", TEXT_SIZE)
|
|
lcd.drawText (x_data2,y, as3xData[12], TEXT_SIZE + RIGHT)
|
|
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, "Pitch:", TEXT_SIZE)
|
|
lcd.drawText (x_data1,y,as3xData[7], TEXT_SIZE + RIGHT)
|
|
|
|
lcd.drawText (X_COL2_HEADER,y, "Roll L:", TEXT_SIZE)
|
|
lcd.drawText (x_data2,y,as3xData[13], TEXT_SIZE + RIGHT)
|
|
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL1_HEADER,y, "Yaw:", TEXT_SIZE)
|
|
lcd.drawText (x_data1,y, as3xData[8], TEXT_SIZE + RIGHT)
|
|
|
|
lcd.drawText (X_COL2_HEADER,y, "Pitch U:", TEXT_SIZE)
|
|
lcd.drawText (x_data2,y, as3xData[14], TEXT_SIZE + RIGHT)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
lcd.drawText (X_COL2_HEADER,y, "Pitch D:", TEXT_SIZE)
|
|
lcd.drawText (x_data2,y, as3xData[15], TEXT_SIZE + RIGHT)
|
|
end
|
|
|
|
-- Debug Values
|
|
if (DEBUG_ON) then
|
|
local titles = {[0]=
|
|
"0500","0502","0504","0506","0508","050B","050D"}
|
|
|
|
local values = {[0]=
|
|
s0500,s0502,s0504,s0506,s0508,s050B,s050D }
|
|
|
|
y = Y_LINE_HEIGHT*2 + Y_HEADER
|
|
for iParam=0,#titles do -- ??
|
|
-- labels
|
|
local x = X_COL1_HEADER+250
|
|
local val = titles[iParam]
|
|
lcd.drawText (x, y, val, TEXT_SIZE)
|
|
|
|
val = values[iParam] or "--"
|
|
x = X_COL1_DATA+250
|
|
lcd.drawText (x, y, val, TEXT_SIZE)
|
|
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
end
|
|
end
|
|
|
|
local function drawAS3XSettingsP1(event)
|
|
drawAS3XSettings(event, 1)
|
|
end
|
|
|
|
local function drawAS3XSettingsP2(event)
|
|
drawAS3XSettings(event, 2)
|
|
end
|
|
|
|
|
|
local function doFloat(v)
|
|
if v==nil then return 0.0 end
|
|
|
|
local vs = string.format("%1.2f",v)
|
|
|
|
return vs + 0.0
|
|
end
|
|
|
|
|
|
local ESC_Title={[0]="","RPM:","Volts:","Motor:","Mot Out:","Throttle:","FET Temp:", "BEC V:", "BEC T:", "BEC A:"}
|
|
local ESC_uom={[0]="","","V","A","%","%","C", "V","C","A"}
|
|
local ESC_Status={[0]=0,0,0,0,0,0,0,0,0,0,0}
|
|
local ESC_Min={[0]=0,0,0,0,0,0,0,0,0,0,0}
|
|
local ESC_Max={[0]=0,0,0,0,0,0,0,0,0,0,0}
|
|
|
|
local function drawESCStatus(event)
|
|
lcd.clear()
|
|
ESC_Status[1] = getValue("Erpm") -- RPM
|
|
ESC_Status[2] = doFloat(getValue("EVIN")) -- Volts
|
|
ESC_Status[3] = doFloat(getValue("ECUR")) -- Current
|
|
ESC_Status[4] = doFloat(getValue("EOUT")) -- % Output
|
|
ESC_Status[5] = doFloat(getValue("ETHR")) -- Throttle % (EOUT)
|
|
ESC_Status[6] = getValue("TFET") -- Temp FET
|
|
|
|
ESC_Status[7] = doFloat(getValue("VBEC")) -- Volts BEC
|
|
ESC_Status[8] = getValue("TBEC") -- Temp BEC
|
|
ESC_Status[9] = doFloat(getValue("CBEC")) -- Current BEC
|
|
|
|
for i=1,9 do
|
|
if (ESC_Status~=nil) then
|
|
if (ESC_Min[i]==0) then
|
|
ESC_Min[i]=ESC_Status[i]
|
|
else
|
|
ESC_Min[i] = math.min(ESC_Min[i],ESC_Status[i])
|
|
end
|
|
|
|
ESC_Max[i] = math.max(ESC_Max[i],ESC_Status[i])
|
|
end
|
|
end
|
|
|
|
lcd.drawText (0,0, "ESC", TEXT_SIZE+INVERS)
|
|
|
|
local y = 0
|
|
local x_data = X_COL1_DATA+X_DATA_LEN*1.5
|
|
local x_data2 = X_COL2_DATA+X_DATA_LEN*0.5
|
|
local x_data3 = x_data2 + X_DATA_LEN*0.8
|
|
|
|
|
|
lcd.drawText (x_data,y , "Status", TEXT_SIZE+BOLD+RIGHT)
|
|
lcd.drawText (x_data2,y, "Min", TEXT_SIZE+BOLD+RIGHT)
|
|
lcd.drawText (x_data3,y, "Max", TEXT_SIZE+BOLD+RIGHT)
|
|
|
|
y = Y_DATA
|
|
for i=1,9 do
|
|
lcd.drawText (X_COL1_HEADER,y, ESC_Title[i], TEXT_SIZE + BOLD)
|
|
|
|
lcd.drawText (x_data,y, ESC_Status[i] or "--", TEXT_SIZE + RIGHT)
|
|
lcd.drawText (x_data + X_DATA_SPACE,y, ESC_uom[i], TEXT_SIZE)
|
|
|
|
lcd.drawText (x_data2,y, ESC_Min[i] or "--", TEXT_SIZE + RIGHT)
|
|
lcd.drawText (x_data3,y, ESC_Max[i] or "--", TEXT_SIZE + RIGHT)
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
end
|
|
|
|
|
|
local function drawBATStatus(event)
|
|
local Title={[0]="","Bat:","Temp:","Rem :","Curr:","Used:","Imbal:","Cycles:", "RX:", "BCpT?:"}
|
|
local uom={[0]="","V","C","%","mAh","mAh","mV","", "V",""}
|
|
local Values={[0]=0,0,0,0,0,0,0,0,0,0,0}
|
|
local CellValues={[0]=0,0,0,0,0,0,0,0,0,0,0}
|
|
|
|
lcd.clear()
|
|
|
|
local ESC_Volts = getValue("EVIN") or 0 -- Volts
|
|
local ESC_Current = getValue("ECUR") or 0 -- Current
|
|
|
|
Values[1] = 0 -- compute later
|
|
Values[2] = getValue("BTmp") -- Current (C)
|
|
Values[3] = nil -- Remaining???
|
|
Values[4] = getValue("BCur") -- Current (mAh)
|
|
Values[5] = getValue("BUse") -- Current Used (mAh)
|
|
Values[6] = getValue("CLMa") -- 0.0 (mV) Imbalance
|
|
Values[7] = getValue("Cycl") -- Cycles
|
|
Values[8] = readBatValue("A2") -- v
|
|
Values[9] = getValue("BCpT") -- Current (mAh) ????
|
|
|
|
|
|
--- Total Voltange Calculation
|
|
local VTotal=0
|
|
for i=1,10 do
|
|
CellValues[i] = getValue("Cel"..i)
|
|
VTotal = VTotal + CellValues[i]
|
|
end
|
|
|
|
if (VTotal==0) then -- No Inteligent Battery,use intelligent ESC if any
|
|
VTotal = ESC_Volts
|
|
Values[4] = string.format("%d",ESC_Current * 1000)
|
|
end
|
|
|
|
Values[1] = string.format("%2.2f",VTotal)
|
|
|
|
--- SCREEN
|
|
|
|
lcd.drawText (X_COL1_HEADER,0, "Battery Stats", TEXT_SIZE+INVERS)
|
|
|
|
|
|
|
|
local y = Y_DATA
|
|
local x_data = X_COL1_DATA+X_DATA_LEN+X_DATA_SPACE*3
|
|
for i=2,9 do
|
|
lcd.drawText (X_COL1_HEADER, y, Title[i], TEXT_SIZE + BOLD)
|
|
lcd.drawText (x_data, y, Values[i] or "--", TEXT_SIZE + RIGHT)
|
|
lcd.drawText (x_data+X_DATA_SPACE, y, uom[i], TEXT_SIZE)
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
|
|
y = Y_DATA
|
|
x_data = X_COL2_DATA+X_DATA_LEN+X_DATA_SPACE*5
|
|
for i=1,8 do
|
|
if ((CellValues[i] or 0) > 0) then
|
|
lcd.drawText (X_COL2_HEADER+X_DATA_LEN/2,y, "Cel "..i..":", TEXT_SIZE + BOLD)
|
|
lcd.drawText (x_data,y, string.format("%2.2f",CellValues[i] or 0), TEXT_SIZE + RIGHT)
|
|
lcd.drawText (x_data+X_DATA_SPACE,y, "v", TEXT_SIZE)
|
|
end
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
|
|
lcd.drawText (X_COL2_HEADER+X_DATA_LEN/2,0, Title[1], TEXT_SIZE + INVERS + BOLD)
|
|
lcd.drawText (x_data,0, string.format("%2.2f",Values[1] or 0), TEXT_SIZE + INVERS+ RIGHT)
|
|
lcd.drawText (x_data+X_DATA_SPACE, 0, uom[1], TEXT_SIZE + INVERS)
|
|
end
|
|
|
|
|
|
|
|
|
|
local function openTelemetryRaw(i2cId)
|
|
--Init telemetry (Spectrun Telemetry Raw STR)
|
|
multiBuffer( 0, string.byte('S') )
|
|
multiBuffer( 1, string.byte('T') )
|
|
multiBuffer( 2, string.byte('R') )
|
|
multiBuffer( 3, i2cId ) -- Monitor this teemetry data
|
|
multiBuffer( 4, 0 ) -- Allow to get Data
|
|
end
|
|
|
|
local function closeTelemetryRaw()
|
|
multiBuffer(0, 0) -- Destroy the STR header
|
|
multiBuffer(3, 0) -- Not requesting any Telementry ID
|
|
end
|
|
|
|
local lineText = {nil}
|
|
local I2C_TEXT_GEN = 0x0C
|
|
local function drawTextGen(event)
|
|
if (multiBuffer(0)~=string.byte('S')) then -- First time run???
|
|
openTelemetryRaw(I2C_TEXT_GEN) -- I2C_ID for TEXT_GEN
|
|
lineText = {nil}
|
|
end
|
|
|
|
-- Proces TEXT GEN Telementry message
|
|
if multiBuffer( 4 ) == I2C_TEXT_GEN then -- Specktrum Telemetry ID of data received
|
|
local instanceNo = multiBuffer( 5 )
|
|
local lineNo = multiBuffer( 6 )
|
|
|
|
local line = ""
|
|
for i=0,13 do
|
|
line = line .. string.char(multiBuffer( 7 + i ))
|
|
end
|
|
|
|
multiBuffer( 4, 0 ) -- Clear Semaphore, to notify that we fully process the current message
|
|
lineText[lineNo]=line
|
|
end
|
|
|
|
lcd.clear()
|
|
-- Header
|
|
if (lineText[0]) then
|
|
lcd.drawText (X_COL1_HEADER,0, " "..lineText[0].." ", TEXT_SIZE + BOLD + INVERS)
|
|
else
|
|
lcd.drawText (X_COL1_HEADER,0, "TextGen", TEXT_SIZE+INVERS)
|
|
end
|
|
|
|
-- Menu lines
|
|
local y = Y_DATA
|
|
for i=1,8 do
|
|
if (lineText[i]) then
|
|
lcd.drawText (X_COL1_HEADER,y, lineText[i], TEXT_SIZE)
|
|
end
|
|
y = y + Y_LINE_HEIGHT
|
|
end
|
|
|
|
if event == EVT_VIRTUAL_EXIT then -- Exit?? Clear menu data
|
|
closeTelemetryRaw()
|
|
end
|
|
end
|
|
|
|
|
|
local telPage = 1
|
|
local telPageSelected = 0
|
|
local pageTitle = {[0]="Main", "AS3X Settings", "SAFE Settings", "ESC Status", "Battery Status","TextGen","Flight Log"}
|
|
|
|
local function drawMainScreen(event)
|
|
lcd.clear()
|
|
lcd.drawText (X_COL1_HEADER, Y_HEADER, "Main Telemetry (Smart RXs)", TEXT_SIZE + INVERS)
|
|
|
|
for iParam=1,#pageTitle do
|
|
-- highlight selected parameter
|
|
local attr = (telPage==iParam) and INVERS or 0
|
|
|
|
-- set y draw coord
|
|
local y = (iParam)*Y_LINE_HEIGHT+Y_DATA
|
|
|
|
-- labels
|
|
local x = X_COL1_HEADER
|
|
local val = pageTitle[iParam]
|
|
lcd.drawText (x, y, val, attr + TEXT_SIZE)
|
|
end
|
|
|
|
if event == EVT_VIRTUAL_PREV then
|
|
if (telPage>1) then telPage = telPage - 1 end
|
|
elseif event == EVT_VIRTUAL_NEXT then
|
|
if (telPage<#pageTitle) then telPage = telPage + 1 end
|
|
elseif event == EVT_VIRTUAL_ENTER then
|
|
telPageSelected = telPage
|
|
end
|
|
end
|
|
|
|
|
|
local pageDraw = {[0]=drawMainScreen, drawAS3XSettingsP1, drawAS3XSettingsP2, drawESCStatus, drawBATStatus, drawTextGen, drawFlightLogScreen}
|
|
|
|
local function run_func(event)
|
|
if event == nil then
|
|
error("Cannot be run as a model script!")
|
|
return 2
|
|
end
|
|
|
|
-- draw specific page
|
|
pageDraw[telPageSelected](event)
|
|
|
|
if event == EVT_VIRTUAL_EXIT then
|
|
if (telPageSelected==0) then return 1 end -- on Main?? Exit Script
|
|
telPageSelected = 0 -- any page, return to Main
|
|
end
|
|
|
|
return 0
|
|
end
|
|
|
|
local function init_func()
|
|
|
|
if (LCD_W <= 128 or LCD_H <=64) then -- Smaller Screens
|
|
TEXT_SIZE = SMLSIZE
|
|
X_COL1_HEADER = 0
|
|
X_COL1_DATA = 20
|
|
|
|
X_COL2_HEADER = 60
|
|
X_COL2_DATA = 90
|
|
|
|
X_DATA_LEN = 28
|
|
X_DATA_SPACE = 1
|
|
|
|
|
|
Y_LINE_HEIGHT = 8
|
|
Y_DATA = Y_HEADER + Y_LINE_HEIGHT
|
|
|
|
end
|
|
end
|
|
|
|
return { run=run_func, init=init_func }
|