mirror of
https://github.com/opentx/opentx.git
synced 2025-07-23 08:15:17 +03:00
363 lines
13 KiB
Lua
Executable file
363 lines
13 KiB
Lua
Executable file
--- - #########################################################################
|
|
---- # #
|
|
---- # Copyright (C) OpenTX #
|
|
----- # #
|
|
---- # 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. #
|
|
---- # #
|
|
---- #########################################################################
|
|
|
|
local version = "v2.00"
|
|
|
|
local VALUE = 0
|
|
local COMBO = 1
|
|
|
|
local COLUMN_2 = 300
|
|
|
|
local edit = false
|
|
local page = 1
|
|
local current = 1
|
|
local refreshState = 0
|
|
local refreshIndex = 0
|
|
local calibrationState = 0
|
|
local pageOffset = 0
|
|
local calibrationStep = 0
|
|
local pages = {}
|
|
local fields = {}
|
|
local modifications = {}
|
|
local wingBitmaps = {}
|
|
local mountBitmaps = {}
|
|
local margin = 1
|
|
local spacing = 8
|
|
local counter = 0
|
|
|
|
local configFields = {
|
|
{ "Wing type", COMBO, 0x80, nil, { "Normal", "Delta", "VTail" } },
|
|
{ "Mounting type", COMBO, 0x81, nil, { "Horz", "Horz rev.", "Vert", "Vert rev." } },
|
|
}
|
|
|
|
local wingBitmapsFile = { "/SCRIPTS/TOOLS/bmp/plane.bmp", "/SCRIPTS/TOOLS/bmp/delta.bmp", "/SCRIPTS/TOOLS/bmp/vtail.bmp" }
|
|
local mountBitmapsFile = { "/SCRIPTS/TOOLS/bmp/horz.bmp", "/SCRIPTS/TOOLS/bmp/horz-r.bmp", "/SCRIPTS/TOOLS/bmp/vert.bmp", "/SCRIPTS/TOOLS/bmp/vert-r.bmp" }
|
|
|
|
local settingsFields = {
|
|
{"SxR functions", COMBO, 0x9C, nil, { "Disable", "Enable" } },
|
|
{"Quick Mode:", COMBO, 0xAA, nil, { "Disable", "Enable" } },
|
|
{"CH5 mode", COMBO, 0xA8, nil, { "AIL2", "AUX1" } },
|
|
{"CH6 mode", COMBO, 0xA9, nil, { "ELE2", "AUX2" } },
|
|
{"AIL direction", COMBO, 0x82, nil, { "Normal", "Invers" }, { 255, 0 } },
|
|
{"ELE direction", COMBO, 0x83, nil, { "Normal", "Invers" }, { 255, 0 } },
|
|
{"RUD direction", COMBO, 0x84, nil, { "Normal", "Invers" }, { 255, 0 } },
|
|
{"AIL2 direction", COMBO, 0x9A, nil, { "Normal", "Invers" }, { 255, 0 } },
|
|
{"ELE2 direction", COMBO, 0x9B, nil, { "Normal", "Invers" }, { 255, 0 } },
|
|
{"AIL stab gain", VALUE, 0x85, nil, 0, 200, "%"},
|
|
{"ELE stab gain", VALUE, 0x86, nil, 0, 200, "%"},
|
|
{"RUD stab gain", VALUE, 0x87, nil, 0, 200, "%"},
|
|
{"AIL autolvl gain", VALUE, 0x88, nil, 0, 200, "%"},
|
|
{"ELE autolvl gain", VALUE, 0x89, nil, 0, 200, "%"},
|
|
{"ELE hover gain", VALUE, 0x8C, nil, 0, 200, "%"},
|
|
{"RUD hover gain", VALUE, 0x8D, nil, 0, 200, "%"},
|
|
{"AIL knife gain", VALUE, 0x8E, nil, 0, 200, "%"},
|
|
{"RUD knife gain", VALUE, 0x90, nil, 0, 200, "%"},
|
|
{"AIL autolvl offset", VALUE, 0x91, nil, -20, 20, "%", 0x6C},
|
|
{"ELE autolvl offset", VALUE, 0x92, nil, -20, 20, "%", 0x6C},
|
|
{"ELE hover offset", VALUE, 0x95, nil, -20, 20, "%", 0x6C},
|
|
{"RUD hover offset", VALUE, 0x96, nil, -20, 20, "%", 0x6C},
|
|
{"AIL knife offset", VALUE, 0x97, nil, -20, 20, "%", 0x6C},
|
|
{"RUD knife offset", VALUE, 0x99, nil, -20, 20, "%", 0x6C},
|
|
}
|
|
|
|
local calibrationFields = {
|
|
{ "X:", VALUE, 0x9E, 0, -100, 100, "%" },
|
|
{ "Y:", VALUE, 0x9F, 0, -100, 100, "%" },
|
|
{ "Z:", VALUE, 0xA0, 0, -100, 100, "%" }
|
|
}
|
|
|
|
local function drawScreenTitle(title, page, pages)
|
|
if math.fmod(math.floor(getTime() / 100), 10) == 0 then
|
|
title = version
|
|
end
|
|
if LCD_W == 480 then
|
|
lcd.drawFilledRectangle(0, 0, LCD_W, 30, TITLE_BGCOLOR)
|
|
lcd.drawText(1, 5, title, MENU_TITLE_COLOR)
|
|
lcd.drawText(LCD_W - 40, 5, page .. "/" .. pages, MENU_TITLE_COLOR)
|
|
else
|
|
lcd.drawScreenTitle(title, page, pages)
|
|
end
|
|
end
|
|
|
|
-- Change display attribute to current field
|
|
local function addField(step)
|
|
local field = fields[current]
|
|
local min, max
|
|
if field[2] == VALUE then
|
|
min = field[5]
|
|
max = field[6]
|
|
elseif field[2] == COMBO then
|
|
min = 0
|
|
max = #(field[5]) - 1
|
|
end
|
|
if (step < 0 and field[4] > min) or (step > 0 and field[4] < max) then
|
|
field[4] = field[4] + step
|
|
end
|
|
end
|
|
|
|
-- Select the next or previous page
|
|
local function selectPage(step)
|
|
page = 1 + ((page + step - 1 + #pages) % #pages)
|
|
refreshIndex = 0
|
|
calibrationStep = 0
|
|
pageOffset = 0
|
|
end
|
|
|
|
-- Select the next or previous editable field
|
|
local function selectField(step)
|
|
current = 1 + ((current + step - 1 + #fields) % #fields)
|
|
if current > 7 + pageOffset then
|
|
pageOffset = current - 7
|
|
elseif current <= pageOffset then
|
|
pageOffset = current - 1
|
|
end
|
|
end
|
|
|
|
local function drawProgressBar()
|
|
if LCD_W == 480 then
|
|
local width = (300 * refreshIndex) / #fields
|
|
lcd.drawRectangle(100, 10, 300, 6)
|
|
lcd.drawFilledRectangle(102, 13, width, 2);
|
|
else
|
|
local width = (60 * refreshIndex) / #fields
|
|
lcd.drawRectangle(45, 1, 60, 6)
|
|
lcd.drawFilledRectangle(47, 3, width, 2);
|
|
end
|
|
end
|
|
|
|
-- Redraw the current page
|
|
local function redrawFieldsPage(event)
|
|
lcd.clear()
|
|
drawScreenTitle("SxR", page, #pages)
|
|
|
|
if refreshIndex < #fields then
|
|
drawProgressBar()
|
|
end
|
|
|
|
for index = 1, 10, 1 do
|
|
local field = fields[pageOffset + index]
|
|
if field == nil then
|
|
break
|
|
end
|
|
|
|
local attr = current == (pageOffset + index) and ((edit == true and BLINK or 0) + INVERS) or 0
|
|
|
|
lcd.drawText(1, margin + spacing * index, field[1], attr)
|
|
|
|
if field[4] == nil then
|
|
lcd.drawText(LCD_W, margin + spacing * index, "---", RIGHT + attr)
|
|
else
|
|
if field[2] == VALUE then
|
|
lcd.drawNumber(LCD_W, margin + spacing * index, field[4], RIGHT + attr)
|
|
elseif field[2] == COMBO then
|
|
if field[4] >= 0 and field[4] < #(field[5]) then
|
|
lcd.drawText(LCD_W, margin + spacing * index, field[5][1 + field[4]], RIGHT + attr)
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
local function telemetryRead(field)
|
|
return sportTelemetryPush(0x17, 0x30, 0x0C30, field)
|
|
end
|
|
|
|
local function telemetryWrite(field, value)
|
|
return sportTelemetryPush(0x17, 0x31, 0x0C30, field + value * 256)
|
|
end
|
|
|
|
local telemetryPopTimeout = 0
|
|
local function refreshNext()
|
|
if refreshState == 0 then
|
|
if calibrationState == 1 then
|
|
if telemetryWrite(0x9D, calibrationStep) == true then
|
|
refreshState = 1
|
|
calibrationState = 2
|
|
telemetryPopTimeout = getTime() + 120 -- normal delay is 500ms
|
|
end
|
|
elseif #modifications > 0 then
|
|
telemetryWrite(modifications[1][1], modifications[1][2])
|
|
modifications[1] = nil
|
|
elseif refreshIndex < #fields then
|
|
local field = fields[refreshIndex + 1]
|
|
if telemetryRead(field[3]) == true then
|
|
refreshState = 1
|
|
telemetryPopTimeout = getTime() + 80 -- normal delay is 500ms
|
|
end
|
|
end
|
|
elseif refreshState == 1 then
|
|
local physicalId, primId, dataId, value = sportTelemetryPop()
|
|
if physicalId == 0x1A and primId == 0x32 and dataId == 0x0C30 then
|
|
local fieldId = value % 256
|
|
if calibrationState == 2 then
|
|
if fieldId == 0x9D then
|
|
refreshState = 0
|
|
calibrationState = 0
|
|
calibrationStep = (calibrationStep + 1) % 7
|
|
end
|
|
else
|
|
local field = fields[refreshIndex + 1]
|
|
if fieldId == field[3] then
|
|
local value = math.floor(value / 256)
|
|
if field[3] == 0xAA then
|
|
value = bit32.band(value, 0x0001)
|
|
end
|
|
if field[3] >= 0x9E and field[3] <= 0xA0 then
|
|
local b1 = value % 256
|
|
local b2 = math.floor(value / 256)
|
|
value = b1 * 256 + b2
|
|
value = value - bit32.band(value, 0x8000) * 2
|
|
end
|
|
if field[2] == COMBO and #field == 6 then
|
|
for index = 1, #(field[6]), 1 do
|
|
if value == field[6][index] then
|
|
value = index - 1
|
|
break
|
|
end
|
|
end
|
|
elseif field[2] == VALUE and #field == 8 then
|
|
value = value - field[8] + field[5]
|
|
end
|
|
fields[refreshIndex + 1][4] = value
|
|
refreshIndex = refreshIndex + 1
|
|
refreshState = 0
|
|
end
|
|
end
|
|
elseif getTime() > telemetryPopTimeout then
|
|
fields[refreshIndex + 1][4] = nil
|
|
refreshIndex = refreshIndex + 1
|
|
refreshState = 0
|
|
calibrationState = 0
|
|
end
|
|
end
|
|
end
|
|
|
|
local function updateField(field)
|
|
local value = field[4]
|
|
if field[2] == COMBO and #field == 6 then
|
|
value = field[6][1 + value]
|
|
elseif field[2] == VALUE and #field == 8 then
|
|
value = value + field[8] - field[5]
|
|
end
|
|
modifications[#modifications + 1] = { field[3], value }
|
|
end
|
|
|
|
-- Main
|
|
local function runFieldsPage(event)
|
|
if event == EVT_VIRTUAL_EXIT then -- exit script
|
|
return 2
|
|
elseif event == EVT_VIRTUAL_ENTER then -- toggle editing/selecting current field
|
|
if fields[current][4] ~= nil then
|
|
edit = not edit
|
|
if edit == false then
|
|
updateField(fields[current])
|
|
end
|
|
end
|
|
elseif edit then
|
|
if event == EVT_VIRTUAL_INC or event == EVT_VIRTUAL_INC_REPT then
|
|
addField(1)
|
|
elseif event == EVT_VIRTUAL_DEC or event == EVT_VIRTUAL_DEC_REPT then
|
|
addField(-1)
|
|
end
|
|
else
|
|
if event == EVT_VIRTUAL_NEXT then
|
|
selectField(1)
|
|
elseif event == EVT_VIRTUAL_PREV then
|
|
selectField(-1)
|
|
end
|
|
end
|
|
redrawFieldsPage(event)
|
|
return 0
|
|
end
|
|
|
|
local function runConfigPage(event)
|
|
fields = configFields
|
|
local result = runFieldsPage(event)
|
|
if LCD_W == 128 then
|
|
local mountText = { "Label is facing the sky", "Label is facing ground", "Label is left when", "Label is right when" }
|
|
if fields[2][4] ~= nil then
|
|
lcd.drawText(1, 30, "Pins toward tail")
|
|
lcd.drawText(1, 40, mountText[1 + fields[2][4]])
|
|
if fields[2][4] > 1 then
|
|
lcd.drawText(1, 50, "looking from the tail")
|
|
end
|
|
end
|
|
else
|
|
if fields[1][4] ~= nil then
|
|
if LCD_W == 480 then
|
|
if wingBitmaps[1 + fields[1][4]] == nil then
|
|
wingBitmaps[1 + fields[1][4]] = Bitmap.open(wingBitmapsFile[1 + fields[1][4]])
|
|
end
|
|
lcd.drawBitmap(wingBitmaps[1 + fields[1][4]], 10, 90)
|
|
else
|
|
lcd.drawPixmap(20, 28, wingBitmapsFile[1 + fields[1][4]])
|
|
end
|
|
end
|
|
if fields[2][4] ~= nil then
|
|
if LCD_W == 480 then
|
|
if mountBitmaps[1 + fields[2][4]] == nil then
|
|
mountBitmaps[1 + fields[2][4]] = Bitmap.open(mountBitmapsFile[1 + fields[2][4]])
|
|
end
|
|
lcd.drawBitmap(mountBitmaps[1 + fields[2][4]], 190, 110)
|
|
else
|
|
lcd.drawPixmap(128, 28, mountBitmapsFile[1 + fields[2][4]])
|
|
end
|
|
end
|
|
end
|
|
return result
|
|
end
|
|
|
|
local function runSettingsPage(event)
|
|
fields = settingsFields
|
|
return runFieldsPage(event)
|
|
end
|
|
|
|
-- Init
|
|
local function init()
|
|
current, edit, refreshState, refreshIndex = 1, false, 0, 0
|
|
if LCD_W == 480 then
|
|
margin = 10
|
|
spacing = 20
|
|
wingBitmapsFile = { "/SCRIPTS/TOOLS/img/plane_b.png", "/SCRIPTS/TOOLS/img/delta_b.png", "/SCRIPTS/TOOLS/img/planev_b.png" }
|
|
mountBitmapsFile = { "/SCRIPTS/TOOLS/img/up.png", "/SCRIPTS/TOOLS/img/down.png", "/SCRIPTS/TOOLS/img/vert.png", "/SCRIPTS/TOOLS/img/vert-r.png" }
|
|
end
|
|
pages = {
|
|
runConfigPage,
|
|
runSettingsPage,
|
|
}
|
|
end
|
|
|
|
-- Main
|
|
local function run(event)
|
|
if event == nil then
|
|
error("Cannot be run as a model script!")
|
|
return 2
|
|
elseif event == EVT_VIRTUAL_NEXT_PAGE then
|
|
selectPage(1)
|
|
elseif event == EVT_VIRTUAL_PREV_PAGE then
|
|
killEvents(event);
|
|
selectPage(-1)
|
|
end
|
|
|
|
local result = pages[page](event)
|
|
refreshNext()
|
|
|
|
return result
|
|
end
|
|
|
|
return { init = init, run = run }
|
|
|