diff --git a/.github/ISSUE_TEMPLATE/Bug_report.md b/.github/ISSUE_TEMPLATE/Bug_report.md index d55a7885..9e9802c1 100644 --- a/.github/ISSUE_TEMPLATE/Bug_report.md +++ b/.github/ISSUE_TEMPLATE/Bug_report.md @@ -7,20 +7,20 @@ assignees: teckel12 --- ### Describe the bug - + ### To reproduce - + 1. Go to '....' 2. Click on '....' 3. Scroll down to '....' 4. See error ### Expected behavior - + ### Screenshots - + ### Radio and model settings ### Additional context - + diff --git a/.github/ISSUE_TEMPLATE/Feature_request.md b/.github/ISSUE_TEMPLATE/Feature_request.md index d5ee5e13..aa2968a7 100644 --- a/.github/ISSUE_TEMPLATE/Feature_request.md +++ b/.github/ISSUE_TEMPLATE/Feature_request.md @@ -7,13 +7,13 @@ assignees: teckel12 --- ### Is your feature request related to a problem? Please describe - + ### Describe the solution you'd like - + ### Describe alternatives you've considered - + ### Additional context - + diff --git a/.github/ISSUE_TEMPLATE/Question.md b/.github/ISSUE_TEMPLATE/Question.md index f66dcb48..d50ec8c4 100644 --- a/.github/ISSUE_TEMPLATE/Question.md +++ b/.github/ISSUE_TEMPLATE/Question.md @@ -7,20 +7,20 @@ assignees: teckel12 --- ### Describe your question - + ### To reproduce - + 1. Go to '....' 2. Click on '....' 3. Scroll down to '....' 4. Should it be doing this? ### Expected behavior - + ### Screenshots - + ### Additional context - + diff --git a/README.md b/README.md index 0af1a17a..92332f9c 100644 --- a/README.md +++ b/README.md @@ -1,47 +1,49 @@ -## INAV Lua Telemetry Flight Status for Taranis/Horus/Jumper T16 - v1.7.2 +## INAV Lua Telemetry Flight Status for Taranis/Horus/Jumper/Nirvana - v1.7.3 -### FrSky SmartPort(S.Port), D-series, F.Port & TBS Crossfire telemetry on Taranis, Horus & Jumper T16 transmitters +### FrSky SmartPort(S.Port), D-series, F.Port & TBS Crossfire telemetry on Taranis, Horus, Jumper & Nirvana transmitters [![Build Status](https://travis-ci.com/iNavFlight/LuaTelemetry.svg?branch=master)](https://travis-ci.com/iNavFlight/LuaTelemetry) ## Interface -#### [Video of Lua Telemetry](https://youtu.be/YaUgywuT1YM) +#### [Video of Lua Telemetry running on X9D+](https://youtu.be/YaUgywuT1YM) -#### Horus/Jumper T16 view +#### Horus/Jumper T16 and Nirvana view -![sample](assets/iNavHorus.png "View on Horus transmitters") +![sample](assets/iNavHorus.png "View on Horus transmitters")   +![sample](assets/iNavNirvana.jpg "View on Nirvana NV14 transmitter") #### Pilot (glass cockpit) view for fixed wing pilots -![sample](assets/iNavQX7pilot.png "Pilot view on Q X7 and X-Lite")   +![sample](assets/iNavQX7pilot.png "Pilot view on Q X7, X-Lite & Jumper T12")   ![sample](assets/iNavX9Dpilot.png "Pilot view on Taranis X9D, X9D+ and X9E") #### Radar (map) view -![sample](assets/iNavQX7radar.png "Radar view on Q X7 and X-Lite")   +![sample](assets/iNavQX7radar.png "Radar view on Q X7, X-Lite & Jumper T12")   ![sample](assets/iNavX9Dradar.png "Radar view on Taranis X9D, X9D+ and X9E") #### Altitude graph view -![sample](assets/iNavQX7alt.png "Altitude graph view on Q X7 and X-Lite")   +![sample](assets/iNavQX7alt.png "Altitude graph view on Q X7, X-Lite & Jumper T12")   ![sample](assets/iNavX9Dalt.png "Altitude graph view on Taranis X9D, X9D+ and X9E") #### Classic view -![sample](assets/iNavQX7.png "Classic view on Q X7 and X-Lite")   +![sample](assets/iNavQX7.png "Classic view on Q X7, X-Lite & Jumper T12")   ![sample](assets/iNavX9D.png "Classic view on Taranis X9D, X9D+ and X9E") ## Features -* Works with all FrSky telemetry receivers (X, R9 and D series), all TBS Crossfire receivers, all FrSky Taranis and Horus transmitters, and the Jumper T16 transmitter -* Compatible with Betaflight using FrSky X or R9 series receivers (with reduced functionality) and TBS Crossfire support with Betaflight 4.0.0+ +* Supported receivers: All FrSky telemetry receivers (X, R9 and D series) and all TBS Crossfire receivers +* Supported transmitters: All FrSky Taranis and Horus transmitters, Jumper T12 & T16 and the FLYSKY Nirvana NV14 +* Compatible with Betaflight using FrSky X or R9 series receivers (with reduced functionality) and TBS Crossfire support with Betaflight v4.0.0+ * Launch/pilot-based model orientation and location indicators (great for lost orientation/losing sight of your model) * Compass-based direction indicator (with magnetometer sensor on multirotor or fixed-wing with GPS) * Pilot (glass cockpit) view which includes attitude indicator as well as pilot-familiar layout of additional data * Radar (map) view shows model in relationship to home position, can be displayed either as launch/pilot-based or compass-based orientation * Altitude graph view shows altitude for the last 1-6 minutes -* Horus and Jumper T16 transmitters show all views at the same time, and include additional features like roll scale +* Horus, Jumper T16 & FLYSKY Nirvana NV14 transmitters show all views at the same time, and include additional features like roll scale * Bar gauges for Fuel (% battery mAh capacity remaining), Battery voltage, RSSI strength, Transmitter battery, GPS accuracy (HDOP), Variometer (and Altitude for X9D, X9D+ and X9E transmitters) * Display and voice alerts for flight modes and flight mode modifiers (altitude hold, heading hold, home reset, etc.) * Voice notifications for % battery remaining (based on current), voltage low/critical, high altitude, lost GPS, ready to arm, armed, disarmed, etc. @@ -56,12 +58,11 @@ ## Requirements -* [OpenTX v2.2.2+](http://www.open-tx.org/) running on Taranis Q X7/Q X7S, X9D/X9D+, X9E, X9 Lite, X-Lite/X-Lite Pro, Horus X10/X10S or X12S (OpenTX v2.2.3+ is suggested) -* Jumper T16 requires [JumperTX 2.2.3+](https://www.jumper.xyz/jumpertx-t16) (May 23, 2019 or after release) -* The OpenTX/JumperTX firmware on your transmitter needs to [includes the `luac` build option](https://youtu.be/nYeB0IXT-10?t=283) -* FrSky X, R9 or D series telemetry receiver: X4RSB, X8R, XSR, R-XSR, XSR-M, XSR-E, RX4R, RX6R, XM, XM+, R9, R9 Slim, R9 Slim+, R9 Mini, R9 MM, D8R-II plus, D8R-XP, D4R-II, etc. or any TBS Crossfire receiver: Micro, Nano, Diversity, etc. -* [INAV v1.7.3+](https://github.com/iNavFlight/inav/releases) running on your flight controller (INAV v2.1.0+ is suggested for full functionality) - Also compatible with Betaflight (with reduced functionality) -* GPS - If you're looking for a GPS module, I suggest the [Beitian BN-880](https://www.banggood.com/UBLOX-NEO-M8N-BN-880-Flight-Control-GPS-Module-Dual-Module-Compass-p-971082.html) +* [INAV v2.1.0+](https://github.com/iNavFlight/inav/releases) running on your flight controller - Also compatible with Betaflight v4.0.0+ (with reduced functionality) +* [OpenTX v2.2.3+](http://www.open-tx.org/) running on Taranis Q X7/Q X7S, X9D/X9D+, X9E, X9 Lite, X-Lite/X-Lite Pro, Horus X10/X10S or X12S +* Jumper T12/T16 requires [JumperTX 2.2.3+](https://www.jumper.xyz/) (May 23, 2019 or after release) +* FrSky X, R9 or D series telemetry receiver: X4RSB, X8R, XSR, R-XSR, XSR-M, XSR-E, RX4R, RX6R, R9, R9 Slim, R9 Slim+, R9 Mini, R9 MM, D8R-II plus, D8R-XP, D4R-II, etc. or any TBS Crossfire receiver: Micro, Nano, Diversity, etc. +* GPS - If you're looking for a GPS module, I suggest the [Beitian BN-880](https://us.banggood.com/custlink/vvGD6DZWyg) ## Suggested Sensors @@ -71,20 +72,16 @@ ## Notes -* INAV v2.1.0+ is required for TBS Crossfire support (some telemetry missing from Crossfire: HDOP, GPS altitude, variometer and heading hold notifications) -* INAV v2.0.0+ is required for FrSky D-series telemetry and proper GPS accuracy (HDOP) display -* If using pilot or radar view or a Horus transmitter and INAV v2.0+, set `frsky_pitch_roll = ON` in CLI settings for more accurate attitude display -* INAV v1.9.1+ is required for F.Port compatibility -* INAV v1.8.0+ is required for `Home reset` voice notification -* OpenTX v2.2.2 (release version) is required for compatibility with Taranis X-Lite transmitter and Crossfire telemetry -* Betaflight compatibility is mostly complete, except for some GPS and flight mode information missing from Betaflight -* Use the OSD to control VTx band, frequency and power (Betaflight lua script can't be run at the same time as INAV Lua Telemetry due to limited transmitter memory) +* Some telemetry is missing from Crossfire: HDOP, GPS altitude and some secondary flight mode notifications like heading hold +* Betaflight v4.0.0+ mostly works, except for some GPS and flight mode information which is missing from Betaflight +* Use the OSD to control VTx band, frequency and power (except for on 2019 series Taranis transmitters, Betaflight's lua script can't run at the same time as INAV Lua Telemetry due to limited transmitter memory) ## Special Thanks * [Team Black Sheep](https://www.team-blacksheep.com/) - Sponsoring TBS Crossfire telemetry support -* [FrSky](https://www.frsky-rc.com/) - Sponsoring Horus transmitter support +* [FrSky](https://www.frsky-rc.com/) - Sponsoring [FrSky Horus](https://us.banggood.com/custlink/vG3D6Kiprr) transmitter support * [Jumper](https://www.jumper.xyz/) - Sponsoring Jumper T16 transmitter support +* [FLYSKY](https://www.flysky-cn.com/) - Sponsoring [FLYSKY Nirvana NV14](https://us.banggood.com/custlink/GmGm0GZcpt) transmitter support ## Setup diff --git a/assets/iNavNirvana.jpg b/assets/iNavNirvana.jpg new file mode 100644 index 00000000..9bdad5ca Binary files /dev/null and b/assets/iNavNirvana.jpg differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav.lua b/dist/SCRIPTS/TELEMETRY/iNav.lua index e003aa0f..aff1112b 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav.lua and b/dist/SCRIPTS/TELEMETRY/iNav.lua differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav.luac b/dist/SCRIPTS/TELEMETRY/iNav.luac index e003aa0f..aff1112b 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav.luac and b/dist/SCRIPTS/TELEMETRY/iNav.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/alt.luac b/dist/SCRIPTS/TELEMETRY/iNav/alt.luac index 5693b232..2deec7b8 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/alt.luac and b/dist/SCRIPTS/TELEMETRY/iNav/alt.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/crsf.luac b/dist/SCRIPTS/TELEMETRY/iNav/crsf.luac index 4fac00fb..349b5bda 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/crsf.luac and b/dist/SCRIPTS/TELEMETRY/iNav/crsf.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/data.luac b/dist/SCRIPTS/TELEMETRY/iNav/data.luac index 488d70ca..536c1c29 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/data.luac and b/dist/SCRIPTS/TELEMETRY/iNav/data.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/func_h.luac b/dist/SCRIPTS/TELEMETRY/iNav/func_h.luac index 067aa345..13f30ec6 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/func_h.luac and b/dist/SCRIPTS/TELEMETRY/iNav/func_h.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/func_t.luac b/dist/SCRIPTS/TELEMETRY/iNav/func_t.luac index a1f4852c..1461268e 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/func_t.luac and b/dist/SCRIPTS/TELEMETRY/iNav/func_t.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/horus.luac b/dist/SCRIPTS/TELEMETRY/iNav/horus.luac index 1138dc45..6925d08e 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/horus.luac and b/dist/SCRIPTS/TELEMETRY/iNav/horus.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/log.luac b/dist/SCRIPTS/TELEMETRY/iNav/log.luac index 1c0eb849..7ba774e2 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/log.luac and b/dist/SCRIPTS/TELEMETRY/iNav/log.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/menu.luac b/dist/SCRIPTS/TELEMETRY/iNav/menu.luac index 16856435..7af585bf 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/menu.luac and b/dist/SCRIPTS/TELEMETRY/iNav/menu.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/modes.luac b/dist/SCRIPTS/TELEMETRY/iNav/modes.luac index f71305dc..a0e9212b 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/modes.luac and b/dist/SCRIPTS/TELEMETRY/iNav/modes.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/nirvana.luac b/dist/SCRIPTS/TELEMETRY/iNav/nirvana.luac new file mode 100644 index 00000000..df298642 Binary files /dev/null and b/dist/SCRIPTS/TELEMETRY/iNav/nirvana.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/other.luac b/dist/SCRIPTS/TELEMETRY/iNav/other.luac index 75ca4ac3..5bacb8f2 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/other.luac and b/dist/SCRIPTS/TELEMETRY/iNav/other.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/pics/bgnv.png b/dist/SCRIPTS/TELEMETRY/iNav/pics/bgnv.png new file mode 100644 index 00000000..5dd7a70e Binary files /dev/null and b/dist/SCRIPTS/TELEMETRY/iNav/pics/bgnv.png differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/pilot.luac b/dist/SCRIPTS/TELEMETRY/iNav/pilot.luac index aa4ae339..07e135c1 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/pilot.luac and b/dist/SCRIPTS/TELEMETRY/iNav/pilot.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/radar.luac b/dist/SCRIPTS/TELEMETRY/iNav/radar.luac index 2aca7e53..3b9eda86 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/radar.luac and b/dist/SCRIPTS/TELEMETRY/iNav/radar.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/reset.luac b/dist/SCRIPTS/TELEMETRY/iNav/reset.luac index 514080d9..13fab8c6 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/reset.luac and b/dist/SCRIPTS/TELEMETRY/iNav/reset.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/save.luac b/dist/SCRIPTS/TELEMETRY/iNav/save.luac index 32ad15c8..b3f5de34 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/save.luac and b/dist/SCRIPTS/TELEMETRY/iNav/save.luac differ diff --git a/dist/SCRIPTS/TELEMETRY/iNav/view.luac b/dist/SCRIPTS/TELEMETRY/iNav/view.luac index 753e5243..8f86bb2e 100644 Binary files a/dist/SCRIPTS/TELEMETRY/iNav/view.luac and b/dist/SCRIPTS/TELEMETRY/iNav/view.luac differ diff --git a/dist/WIDGETS/iNav/main.lua b/dist/WIDGETS/iNav/main.lua index b5266312..d92df506 100644 --- a/dist/WIDGETS/iNav/main.lua +++ b/dist/WIDGETS/iNav/main.lua @@ -1,7 +1,7 @@ local buildMode = ... local iNav = nil local options = { - { "Restore", BOOL, 0}, + { "Restore", BOOL, 1}, { "Text", COLOR, BLACK}, { "Warning", COLOR, YELLOW} } @@ -13,6 +13,15 @@ if string.sub(r, -4) == "simu" and buildMode ~= true then loadScript(TELE_PATH .. "iNav", "tc")(true) end +-- Nirvana NV14 doesn't have have these global constants, so we set them +if r == "NV14" then + EVT_SYS_FIRST = 1 --1542 + EVT_ROT_LEFT = 2 --57088 + EVT_ROT_RIGHT = 3 --56832 + EVT_ENTER_BREAK = 4 --514 + EVT_EXIT_BREAK = 5 --516 +end + -- Run once at the creation of the widget local function create(zone, options) iNavZone = { zone = zone, options = options } diff --git a/dist/WIDGETS/iNav/main.luac b/dist/WIDGETS/iNav/main.luac index 708d3f7c..123e8cb0 100644 Binary files a/dist/WIDGETS/iNav/main.luac and b/dist/WIDGETS/iNav/main.luac differ diff --git a/src/iNav.lua b/src/iNav.lua index ef5a07c6..1a58a859 100644 --- a/src/iNav.lua +++ b/src/iNav.lua @@ -3,10 +3,10 @@ -- Docs: https://github.com/iNavFlight/LuaTelemetry local buildMode = ... -local VERSION = "1.7.2" +local VERSION = "1.7.3" local FILE_PATH = "/SCRIPTS/TELEMETRY/iNav/" local SMLCD = LCD_W < 212 -local HORUS = LCD_W >= 480 +local HORUS = LCD_W >= 480 or LCD_H >= 480 local FLASH = HORUS and WARNING_COLOR or 3 local tmp, view, lang, playLog local env = "bx" @@ -17,17 +17,17 @@ local v, r, m, i, e = getVersion() if string.sub(r, -4) == "simu" then env = "tx" if buildMode ~= false then - loadScript(FILE_PATH .. "build", "tx")(buildMode) + loadScript(FILE_PATH .. "build", env)(buildMode) end end local config = loadScript(FILE_PATH .. "config" .. ext, env)(SMLCD) collectgarbage() -local modes, units, labels = loadScript(FILE_PATH .. "modes" .. ext, env)() +local modes, units, labels, dir = loadScript(FILE_PATH .. "modes" .. ext, env)(HORUS) collectgarbage() -local data, getTelemetryId, getTelemetryUnit, PREV, NEXT, MENU = loadScript(FILE_PATH .. "data" .. ext, env)(r, m, i, HORUS) +local data, getTelemetryId, getTelemetryUnit, PREV, NEXT, MENU, text, line, rect, fill, frmt = loadScript(FILE_PATH .. "data" .. ext, env)(r, m, i, HORUS) collectgarbage() loadScript(FILE_PATH .. "load" .. ext, env)(config, data, FILE_PATH) @@ -49,7 +49,7 @@ collectgarbage() local crsf, distCalc = loadScript(FILE_PATH .. "other" .. ext, env)(config, data, units, getTelemetryId, getTelemetryUnit, FILE_PATH, env, SMLCD, FLASH) collectgarbage() -local title, gpsDegMin, hdopGraph, icons = loadScript(FILE_PATH .. "func_" .. (HORUS and "h" or "t") .. ext, env)(config, data, FILE_PATH) +local title, gpsDegMin, hdopGraph, icons, rect = loadScript(FILE_PATH .. "func_" .. (HORUS and "h" or "t") .. ext, env)(config, data, modes, dir, SMLCD, FILE_PATH, text, line, rect, fill, frmt) collectgarbage() local function playAudio(f, a) @@ -459,12 +459,12 @@ local function run(event) -- Clear screen if HORUS then -- Display error if Horus widget isn't full screen - if icons.nfs ~= nil then - icons.nfs() + if data.nfs ~= nil then + data.nfs() return 0 end -- On Horus use sticks to control the menu - event = icons.clear(event, data) + event = data.clear(event) else lcd.clear() end @@ -472,7 +472,7 @@ local function run(event) -- Display system error --[[ if data.msg then - lcd.drawText((LCD_W - string.len(data.msg) * (HORUS and 13 or 5.2)) * 0.5, HORUS and 130 or 27, data.msg, HORUS and MIDSIZE or 0) + text((LCD_W - string.len(data.msg) * (HORUS and 13 or 5.2)) * 0.5, HORUS and 130 or 27, data.msg, HORUS and MIDSIZE or 0) return 0 end ]] @@ -486,15 +486,15 @@ local function run(event) data.v = 9 end tmp = config[30].v - view(data, config, units, lang, event, gpsDegMin, getTelemetryId, getTelemetryUnit, FILE_PATH, SMLCD, FLASH, PREV, NEXT, HORUS, env) + view(data, config, units, lang, event, gpsDegMin, getTelemetryId, getTelemetryUnit, SMLCD, FLASH, PREV, NEXT, HORUS, text, rect, fill, frmt, env) if HORUS then - icons.menu(config, data, icons, tmp) + data.menu(tmp) end -- Exit menu or select log for playback, save config settings if data.configSelect == 0 and (event == EVT_EXIT_BREAK or (event == EVT_ENTER_BREAK and data.configStatus == 34 and config[34].x > -1 and not data.armed)) then view = nil collectgarbage() - loadScript(FILE_PATH .. "save" .. ext, env)(config, data, FILE_PATH) + loadScript(FILE_PATH .. "save" .. ext, env)(config, data, frmt, FILE_PATH) end else -- User input @@ -520,15 +520,15 @@ local function run(event) if data.v ~= config[25].v then view = nil collectgarbage() - view = loadScript(FILE_PATH .. (HORUS and "horus" or (config[25].v == 0 and "view" or (config[25].v == 1 and "pilot" or (config[25].v == 2 and "radar" or "alt")))) .. ext, env)() + view = loadScript(FILE_PATH .. (HORUS and (data.nv and "nirvana" or "horus") or (config[25].v == 0 and "view" or (config[25].v == 1 and "pilot" or (config[25].v == 2 and "radar" or "alt")))) .. ext, env)() data.v = config[25].v end - view(data, config, modes, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH) + view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) end collectgarbage() -- Paint title - title(data, config, SMLCD) + title() return 0 end diff --git a/src/iNav/alt.lua b/src/iNav/alt.lua index dc9d04dd..e66e5064 100644 --- a/src/iNav/alt.lua +++ b/src/iNav/alt.lua @@ -1,5 +1,4 @@ -local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH) - +local function view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) local LEFT_DIV = 36 local LEFT_POS = SMLCD and LEFT_DIV or 73 local RIGHT_POS = SMLCD and LCD_W - 31 or LCD_W - 53 @@ -10,16 +9,16 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic -- Startup message if data.startup == 2 then if not SMLCD then - lcd.drawText(LEFT_POS + 8, 28, "Lua Telemetry") + text(LEFT_POS + 8, 28, "Lua Telemetry") end - lcd.drawText(X_CNTR - 10, SMLCD and 29 or 40, "v" .. VERSION) + text(X_CNTR - 10, SMLCD and 29 or 40, "v" .. VERSION) end -- Flight modes tmp = X_CNTR - (SMLCD and 16 or 19) - lcd.drawText(tmp + 1, 9, modes[data.modeId].t, SMLSIZE + modes[data.modeId].f) + text(tmp + 1, 9, modes[data.modeId].t, SMLSIZE + modes[data.modeId].f) if data.headFree then - lcd.drawText(tmp, 9, "HF", SMLSIZE + FLASH + RIGHT) + text(tmp, 9, "HF", SMLSIZE + FLASH + RIGHT) end -- Pitch calculation @@ -34,19 +33,19 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if SMLCD then if data.showDir then -- GPS coords - lcd.drawText(RIGHT_POS, 49, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(RIGHT_POS, 57, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(RIGHT_POS, 49, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(RIGHT_POS, 57, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) else - lcd.drawLine(LEFT_POS + 31, 48, LEFT_POS + 31, 63, SOLID, FORCE) + line(LEFT_POS + 31, 48, LEFT_POS + 31, 63, SOLID, FORCE) -- Distance tmp = data.showMax and data.distanceMax or data.distanceLast - lcd.drawText(LEFT_POS + 29, 57, tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (string.format("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")), SMLSIZE + RIGHT + data.telemFlags) - lcd.drawText(LEFT_POS + 10, 49, "Dist", SMLSIZE) + text(LEFT_POS + 29, 57, tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (frmt("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")), SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS + 10, 49, "Dist", SMLSIZE) icons.home(LEFT_POS + 2, 49) -- Altitude tmp = data.showMax and data.altitudeMax or data.altitude - lcd.drawText(RIGHT_POS, 57, math.floor(tmp + 0.5) .. units[data.alt_unit], SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) - lcd.drawText(RIGHT_POS, 49, "Alt", SMLSIZE + RIGHT) + text(RIGHT_POS, 57, math.floor(tmp + 0.5) .. units[data.alt_unit], SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(RIGHT_POS, 49, "Alt", SMLSIZE + RIGHT) if data.altHold then icons.lock(RIGHT_POS - 20, 49) end @@ -54,37 +53,37 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic end -- Min/Max if not data.showDir and data.showMax then - lcd.drawText(RIGHT_POS, 9, "\192", SMLSIZE + RIGHT) + text(RIGHT_POS, 9, "\192", SMLSIZE + RIGHT) end if data.startup == 0 then -- Altitude graph local BOTTOM = SMLCD and 47 or 63 tmp = (SMLCD and 30 or 40) / (data.altMax - data.altMin) - lcd.drawLine(RIGHT_POS - 60, BOTTOM, RIGHT_POS - 1, BOTTOM, SOLID, SMLCD and FORCE or GREY_DEFAULT + FORCE) + line(RIGHT_POS - 60, BOTTOM, RIGHT_POS - 1, BOTTOM, SOLID, SMLCD and FORCE or GREY_DEFAULT + FORCE) for i = 1, 60 do local cx = RIGHT_POS - 61 + i local cy = math.floor(BOTTOM - (data.alt[((data.altCur - 2 + i) % 60) + 1] - data.altMin) * tmp) if cy < BOTTOM then - lcd.drawLine(cx, cy, cx, BOTTOM - 1, SOLID, SMLCD and FORCE or GREY_DEFAULT + FORCE) + line(cx, cy, cx, BOTTOM - 1, SOLID, SMLCD and FORCE or GREY_DEFAULT + FORCE) end if (i ~= 1 or not SMLCD) and (i - 1) % (60 / config[28].v) == 0 then - lcd.drawLine(cx, BOTTOM - (SMLCD and 30 or 40), cx, BOTTOM, DOTTED, 0) + line(cx, BOTTOM - (SMLCD and 30 or 40), cx, BOTTOM, DOTTED, 0) end end if data.altMin < -1 then local cy = data.altMin * tmp + BOTTOM - lcd.drawLine(RIGHT_POS - 60, cy, RIGHT_POS - 1, cy, DOTTED, 0) + line(RIGHT_POS - 60, cy, RIGHT_POS - 1, cy, DOTTED, 0) end if not SMLCD then - lcd.drawText(RIGHT_POS - 60, 19, math.floor(data.altMax + 0.5) .. units[data.alt_unit], SMLSIZE + RIGHT) + text(RIGHT_POS - 60, 19, math.floor(data.altMax + 0.5) .. units[data.alt_unit], SMLSIZE + RIGHT) end -- Orientation if not SMLCD and data.telem then if data.showDir or data.headingRef == -1 then - lcd.drawText(LEFT_POS + 12, 29, "N", SMLSIZE) - lcd.drawText(LEFT_POS + 25 - (data.heading < 100 and 3 or 0) - (data.heading < 10 and 3 or 0), 57, math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS + 12, 29, dir[0], SMLSIZE) + text(LEFT_POS + 25 - (data.heading < 100 and 3 or 0) - (data.heading < 10 and 3 or 0), 57, math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) tmp = data.heading else tmp = data.heading - data.headingRef @@ -94,104 +93,104 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local r3 = math.rad(tmp - 145) local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, LEFT_POS + 14, 45, 7) if data.headingHold then - lcd.drawFilledRectangle((x2 + x3) * 0.5 - 1, (y2 + y3) * 0.5 - 1, 3, 3, SOLID) + fill((x2 + x3) * 0.5 - 1, (y2 + y3) * 0.5 - 1, 3, 3, SOLID) else - lcd.drawLine(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) + line(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) end - lcd.drawLine(x1, y1, x2, y2, SOLID, FORCE) - lcd.drawLine(x1, y1, x3, y3, SOLID, FORCE) + line(x1, y1, x2, y2, SOLID, FORCE) + line(x1, y1, x3, y3, SOLID, FORCE) end end -- Variometer if config[7].v % 2 == 1 then - lcd.drawLine(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + (SMLCD and 4 or 6), 8, RIGHT_POS + (SMLCD and 4 or 6), 63, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + 1, 35, RIGHT_POS + (SMLCD and 3 or 5), 35, SMLCD and DOTTED or SOLID, SMLCD and 0 or GREY_DEFAULT) + line(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) + line(RIGHT_POS + (SMLCD and 4 or 6), 8, RIGHT_POS + (SMLCD and 4 or 6), 63, SOLID, FORCE) + line(RIGHT_POS + 1, 35, RIGHT_POS + (SMLCD and 3 or 5), 35, SMLCD and DOTTED or SOLID, SMLCD and 0 or GREY_DEFAULT) if data.armed then tmp = math.log(1 + math.min(math.abs(0.6 * (data.vspeed_unit == 6 and data.vspeed * 0.3048 or data.vspeed)), 10)) * (data.vspeed < 0 and -1 or 1) local y1 = 36 - (tmp * 11) local y2 = 36 - (tmp * 9) - lcd.drawLine(RIGHT_POS + 1, y1 - 1, RIGHT_POS + (SMLCD and 3 or 5), y2 - 1, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + 1, y1, RIGHT_POS + (SMLCD and 3 or 5), y2, SOLID, FORCE) + line(RIGHT_POS + 1, y1 - 1, RIGHT_POS + (SMLCD and 3 or 5), y2 - 1, SOLID, FORCE) + line(RIGHT_POS + 1, y1, RIGHT_POS + (SMLCD and 3 or 5), y2, SOLID, FORCE) end else - lcd.drawLine(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) + line(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) end -- Right data - GPS - lcd.drawText(LCD_W, data.crsf and 20 or 8, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) + text(LCD_W, data.crsf and 20 or 8, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) icons.gps(LCD_W - (SMLCD and 23 or 22), data.crsf and 24 or 12) if data.crsf then - lcd.drawText(LCD_W, SMLCD and 9 or 11, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) + text(LCD_W, SMLCD and 9 or 11, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) else - lcd.drawText(LCD_W + 1, SMLCD and 43 or 24, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) + text(LCD_W + 1, SMLCD and 43 or 24, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) end if SMLCD then if data.crsf == false then - lcd.drawText(LCD_W + 1, config[22].v == 0 and 32 or 22, "HDOP", RIGHT + SMLSIZE) + text(LCD_W + 1, config[22].v == 0 and 32 or 22, "HDOP", RIGHT + SMLSIZE) end hdopGraph(LCD_W - 12, config[22].v == 0 and (data.crsf and 37 or 24) or 31, MIDSIZE, SMLCD) else hdopGraph(LCD_W - 39, data.crsf and 24 or 10, MIDSIZE, SMLCD) if data.crsf == false then - lcd.drawText(LCD_W - (config[22].v == 0 and 24 or 25), config[22].v == 0 and 18 or 20, "HDOP", RIGHT + SMLSIZE) + text(LCD_W - (config[22].v == 0 and 24 or 25), config[22].v == 0 and 18 or 20, "HDOP", RIGHT + SMLSIZE) end - lcd.drawText(LCD_W + 1, 33, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(LCD_W + 1, 42, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) - lcd.drawText(RIGHT_POS + 8, 57, data.crsf and "LQ" or "RSSI", SMLSIZE) + text(LCD_W + 1, 33, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(LCD_W + 1, 42, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(RIGHT_POS + 8, 57, data.crsf and "LQ" or "RSSI", SMLSIZE) end - lcd.drawLine(RIGHT_POS + (config[7].v % 2 == 1 and (SMLCD and 5 or 7) or 0), 50, LCD_W, 50, SOLID, FORCE) + line(RIGHT_POS + (config[7].v % 2 == 1 and (SMLCD and 5 or 7) or 0), 50, LCD_W, 50, SOLID, FORCE) local rssiFlags = RIGHT + ((not data.telem or data.rssi < data.rssiLow) and FLASH or 0) - lcd.drawText(LCD_W - (data.crsf and 6 or 10), 52, math.min(data.showMax and data.rssiMin or data.rssiLast, data.crsf and 100 or 99), MIDSIZE + rssiFlags) - lcd.drawText(LCD_W, 57, data.crsf and "%" or "dB", SMLSIZE + rssiFlags) + text(LCD_W - (data.crsf and 6 or 10), 52, math.min(data.showMax and data.rssiMin or data.rssiLast, data.crsf and 100 or 99), MIDSIZE + rssiFlags) + text(LCD_W, 57, data.crsf and "%" or "dB", SMLSIZE + rssiFlags) -- Left data - Battery - lcd.drawLine(LEFT_DIV, 8, LEFT_DIV, 63, SOLID, FORCE) + line(LEFT_DIV, 8, LEFT_DIV, 63, SOLID, FORCE) tmp = (not data.telem or data.cell < config[3].v or (data.showFuel and config[23].v == 0 and data.fuel <= config[17].v)) and FLASH or 0 if data.showFuel then if config[23].v > 0 or (data.crsf and data.showMax) then - lcd.drawText(LEFT_DIV, data.showCurr and 8 or 10, (data.crsf and data.fuelRaw or data.fuel), MIDSIZE + RIGHT + tmp) - lcd.drawText(LEFT_DIV, data.showCurr and 20 or 23, data.fUnit[data.crsf and 1 or config[23].v], SMLSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 8 or 10, (data.crsf and data.fuelRaw or data.fuel), MIDSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 20 or 23, data.fUnit[data.crsf and 1 or config[23].v], SMLSIZE + RIGHT + tmp) else - lcd.drawText(LEFT_DIV - 5, data.showCurr and 8 or 12, data.fuel, DBLSIZE + RIGHT + tmp) - lcd.drawText(LEFT_DIV, data.showCurr and 17 or 21, "%", SMLSIZE + RIGHT + tmp) + text(LEFT_DIV - 5, data.showCurr and 8 or 12, data.fuel, DBLSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 17 or 21, "%", SMLSIZE + RIGHT + tmp) end end - lcd.drawText(LEFT_DIV - 5, data.showCurr and 25 or 32, string.format(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), DBLSIZE + RIGHT + tmp) - lcd.drawText(LEFT_DIV, data.showCurr and 34 or 41, "V", SMLSIZE + RIGHT + tmp) + text(LEFT_DIV - 5, data.showCurr and 25 or 32, frmt(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), DBLSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 34 or 41, "V", SMLSIZE + RIGHT + tmp) if data.showCurr then tmp = data.showMax and data.currentMax or data.current - lcd.drawText(LEFT_DIV - 5, 42, tmp >= 99.5 and math.floor(tmp + 0.5) or string.format("%.1f", tmp), MIDSIZE + RIGHT + data.telemFlags) - lcd.drawText(LEFT_DIV, 47, "A", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV - 5, 42, tmp >= 99.5 and math.floor(tmp + 0.5) or frmt("%.1f", tmp), MIDSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV, 47, "A", SMLSIZE + RIGHT + data.telemFlags) end - lcd.drawLine(0, data.showCurr and 55 or 53, LEFT_DIV, data.showCurr and 55 or 53, SOLID, FORCE) + line(0, data.showCurr and 55 or 53, LEFT_DIV, data.showCurr and 55 or 53, SOLID, FORCE) tmp = data.showMax and data.speedMax or data.speed - lcd.drawText(LEFT_DIV, data.showCurr and 57 or 56, tmp >= 99.5 and math.floor(tmp + 0.5) .. units[data.speed_unit] or string.format("%.1f", tmp) .. units[data.speed_unit], SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV, data.showCurr and 57 or 56, tmp >= 99.5 and math.floor(tmp + 0.5) .. units[data.speed_unit] or frmt("%.1f", tmp) .. units[data.speed_unit], SMLSIZE + RIGHT + data.telemFlags) -- Left data - Wide screen if not SMLCD then - lcd.drawLine(LEFT_POS, 8, LEFT_POS, 63, SOLID, FORCE) + line(LEFT_POS, 8, LEFT_POS, 63, SOLID, FORCE) -- Altitude tmp = data.showMax and data.altitudeMax or data.altitude local tmp2 = data.alt_unit == 9 and 6 or 2 - lcd.drawText(LEFT_DIV + 2, 9, "Alt", SMLSIZE) - lcd.drawText(LEFT_POS - tmp2, data.alt_unit == 9 and 21 or 17, units[data.alt_unit], SMLSIZE + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) - lcd.drawText(LEFT_POS - tmp2, 16, math.floor(tmp + 0.5), MIDSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(LEFT_DIV + 2, 9, "Alt", SMLSIZE) + text(LEFT_POS - tmp2, data.alt_unit == 9 and 21 or 17, units[data.alt_unit], SMLSIZE + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(LEFT_POS - tmp2, 16, math.floor(tmp + 0.5), MIDSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) if data.altHold then icons.lock(LEFT_POS - 6, 9) end -- Distance tmp = data.showMax and data.distanceMax or data.distanceLast tmp2 = data.dist_unit == 9 and (tmp < 1000 and 6 or 11) or (tmp < 1000 and 2 or 10) - lcd.drawText(LEFT_DIV + 2, 30, "Dist", SMLSIZE) - lcd.drawText(LEFT_POS - tmp2, (data.dist_unit == 9 or tmp >= 1000) and 42 or 38, tmp < 1000 and units[data.dist_unit] or (data.dist_unit == 9 and "km" or "mi"), SMLSIZE + data.telemFlags) - lcd.drawText(LEFT_POS - tmp2, 37, tmp < 1000 and math.floor(tmp + 0.5) or string.format("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)), MIDSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV + 2, 30, "Dist", SMLSIZE) + text(LEFT_POS - tmp2, (data.dist_unit == 9 or tmp >= 1000) and 42 or 38, tmp < 1000 and units[data.dist_unit] or (data.dist_unit == 9 and "km" or "mi"), SMLSIZE + data.telemFlags) + text(LEFT_POS - tmp2, 37, tmp < 1000 and math.floor(tmp + 0.5) or frmt("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)), MIDSIZE + RIGHT + data.telemFlags) --Pitch - lcd.drawLine(LEFT_DIV, 50, LEFT_POS, 50, SOLID, FORCE) - lcd.drawText(LEFT_DIV + 5, 54, pitch > 0 and "\194" or (pitch == 0 and "->" or "\195"), SMLSIZE) - lcd.drawText(LEFT_POS, 53, "\64", SMLSIZE + RIGHT + data.telemFlags) - lcd.drawText(LEFT_POS - 4, 52, pitch, MIDSIZE + RIGHT + data.telemFlags) + line(LEFT_DIV, 50, LEFT_POS, 50, SOLID, FORCE) + text(LEFT_DIV + 5, 54, pitch > 0 and "\194" or (pitch == 0 and "->" or "\195"), SMLSIZE) + text(LEFT_POS, 53, "\64", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS - 4, 52, pitch, MIDSIZE + RIGHT + data.telemFlags) end end diff --git a/src/iNav/build.lua b/src/iNav/build.lua index 26776179..ed0be3c5 100644 --- a/src/iNav/build.lua +++ b/src/iNav/build.lua @@ -5,19 +5,24 @@ local SMLCD = LCD_W < 212 local HORUS = LCD_W >= 480 local v, r, m, i, e = getVersion() local env = "tc" +--env = "tcd" -- Uncomment to compile with extra debug info print("") print("--------------------- COMPILE SCRIPTS ---------------------") print("") local config = loadScript(FILE_PATH .. "config", env)(SMLCD) -local modes, units, labels = loadScript(FILE_PATH .. "modes", env)() -local data, getTelemetryId, getTelemetryUnit, PREV, NEXT, MENU = loadScript(FILE_PATH .. "data", env)(r, m, i, HORUS) +local modes, units, labels, dir = loadScript(FILE_PATH .. "modes", env)(HORUS) +local data, getTelemetryId, getTelemetryUnit, PREV, NEXT, MENU, text, line, rect, fill, frmt = loadScript(FILE_PATH .. "data", env)(r, m, i, HORUS) loadScript(FILE_PATH .. "load", env)(config, data, FILE_PATH) if HORUS then - local title, gpsDegMin, hdopGraph, icons = loadScript(FILE_PATH .. "func_h", env)(config, data, FILE_PATH) + local title, gpsDegMin, hdopGraph, icons, rect = loadScript(FILE_PATH .. "func_h", env)(config, data, modes, dir, SMLCD, FILE_PATH, text, line, rect, fill) +else + Bitmap = {} + function Bitmap.open() end + loadScript(FILE_PATH .. "func_h", env)(config, data, modes, dir, SMLCD, FILE_PATH, text, line, rect, fill) end -local title, gpsDegMin, hdopGraph, icons = loadScript(FILE_PATH .. "func_t", env)(config, data, FILE_PATH) +local title, gpsDegMin, hdopGraph, icons, rect = loadScript(FILE_PATH .. "func_t", env)(config, data, modes, dir, SMLCD, FILE_PATH, text, line, rect, fill) data.lang = "en" data.voice = "en" @@ -40,11 +45,12 @@ loadScript(FILE_PATH .. "pilot", env)() loadScript(FILE_PATH .. "radar", env)() loadScript(FILE_PATH .. "alt", env)() loadScript(FILE_PATH .. "horus", env)() +loadScript(FILE_PATH .. "nirvana", env)() loadScript(FILE_PATH .. "menu", env)() loadScript(FILE_PATH .. "log", env)() loadScript(FILE_PATH .. "log_c", env)() loadScript(FILE_PATH .. "log_s", env)() -loadScript(FILE_PATH .. "save", env)(config, data, FILE_PATH) +loadScript(FILE_PATH .. "save", env)(config, data, frmt, FILE_PATH) if buildMode == nil then loadScript("/WIDGETS/iNav/main", env)(true) diff --git a/src/iNav/crsf.lua b/src/iNav/crsf.lua index 51b2de62..f6946c85 100644 --- a/src/iNav/crsf.lua +++ b/src/iNav/crsf.lua @@ -54,7 +54,7 @@ local function crsf(data) data.fm = getValue(data.fm_id) data.modePrev = data.mode --Fake HDOP based on satellite lock count and assume GPS fix when there's at least 6 satellites - data.satellites = data.satellites + (math.floor(math.min(data.satellites + 10, 25) * 0.36 + 0.5) * 100) + (data.satellites >= 6 and 1000 or 0) + data.satellites = math.min(data.satellites, 99) + (math.floor(math.min(data.satellites + 10, 25) * 0.36 + 0.5) * 100) + (data.satellites >= 6 and 1000 or 0) -- In Betaflight 4.0+, flight mode ends with '*' when not armed local bfArmed = true diff --git a/src/iNav/data.lua b/src/iNav/data.lua index 762d8ce6..06306524 100644 --- a/src/iNav/data.lua +++ b/src/iNav/data.lua @@ -10,15 +10,21 @@ local function getTelemetryUnit(n) return (field and field.unit <= 10) and field.unit or 0 end +--[[ Replace with EVT_VIRTUAL_XXX at begining of 2020 and require OpenTX 2.3+ + Currently missing EVT_VIRTUAL_MENU on Jumper T12 + Can remove PREV, NEXT, MENU constants from code + Also, changes in menu.lua, iNav.lua to use the ENV_VIRTUAL_XXX constants +]] local tx = string.sub(r, 0, 2) -if HORUS or string.sub(r, 0, 3) == "x9e" or string.sub(r, 0, 6) == "x9lite" then +if HORUS or string.sub(r, 0, 3) == "x9e" or string.sub(r, 0, 4) == "x9li" or string.sub(r, 0, 6) == "x9d+20" then tx = "x7" end -local tmp = tx == "x9" and EVT_PLUS_FIRST or (tx == "xl" and EVT_UP_FIRST) +local tmp = tx == "x9" and EVT_PLUS_FIRST or EVT_UP_FIRST local PREV = tx == "x7" and EVT_ROT_LEFT or tmp -tmp = tx == "x9" and EVT_MINUS_FIRST or (tx == "xl" and EVT_DOWN_FIRST) +tmp = tx == "x9" and EVT_MINUS_FIRST or EVT_DOWN_FIRST local NEXT = tx == "x7" and EVT_ROT_RIGHT or tmp -local MENU = tx == "xl" and EVT_SHIFT_BREAK or (HORUS and EVT_SYS_FIRST or EVT_MENU_BREAK) +local MENU = tx == "xl" and EVT_SHIFT_BREAK or (HORUS and EVT_SYS_FIRST or (string.sub(r, 0, 3) == "t12" and EVT_VIRTUAL_PREVIOUS or EVT_MENU_BREAK)) + local general = getGeneralSettings() local distSensor = getTelemetryId("Dist") > -1 and "Dist" or (getTelemetryId("0420") > -1 and "0420" or "0007") local data = { @@ -88,9 +94,10 @@ local data = { alt = {}, v = -1, simu = string.sub(r, -4) == "simu", + nv = r == "NV14", --msg = m + i * 0.1 < 2.2 and "OpenTX v2.2+ Required" or false, lastLock = { lat = 0, lon = 0 }, fUnit = {"mAh", "mWh"}, } -return data, getTelemetryId, getTelemetryUnit, PREV, NEXT, MENU \ No newline at end of file +return data, getTelemetryId, getTelemetryUnit, PREV, NEXT, MENU, lcd.drawText, lcd.drawLine, lcd.drawRectangle, lcd.drawFilledRectangle, string.format \ No newline at end of file diff --git a/src/iNav/func_h.lua b/src/iNav/func_h.lua index cb9fad3c..4965facc 100644 --- a/src/iNav/func_h.lua +++ b/src/iNav/func_h.lua @@ -1,49 +1,55 @@ -local config, data, FILE_PATH = ... +local config, data, modes, dir, SMLCD, FILE_PATH, text, line, rect, fill, frmt = ... -local function title(data, config, SMLCD) - local text = lcd.drawText - local fill = lcd.drawFilledRectangle +local function title() local color = lcd.setColor - local fmt = string.format local tmp = 0 if not data.telem then - color(WARNING_COLOR, RED) tmp = WARNING_COLOR end + -- Title color(CUSTOM_COLOR, BLACK) fill(0, 0, LCD_W, 20, CUSTOM_COLOR) + + -- Model text(0, 0, model.getInfo().name) - if config[13].v > 0 then - if data.doLogs and data.time ~= nil then - text(340, 0, data.time, WARNING_COLOR) - else - lcd.drawTimer(340, 0, data.timer) - end - end + + -- TX battery + local bat = data.nv and 135 or 197 if config[19].v > 0 then - fill(197, 3, 43, 14, TEXT_COLOR) - fill(240, 6, 2, 8, TEXT_COLOR) - local tmp = math.max(math.min((data.txBatt - data.txBattMin) / (data.txBattMax - data.txBattMin) * 42, 42), 0) + 197 - for i = 200, tmp, 4 do + fill(bat, 3, 43, 14, TEXT_COLOR) + fill(bat + 43, 6, 2, 8, TEXT_COLOR) + local lev = math.max(math.min((data.txBatt - data.txBattMin) / (data.txBattMax - data.txBattMin) * 42, 42), 0) + bat + for i = bat + 3, lev, 4 do fill(i, 5, 2, 10, CUSTOM_COLOR) end end if config[19].v ~= 1 then - text(290, 0, fmt("%.1fV", data.txBatt), RIGHT) + text(data.nv and 180 or bat + 93, 0, frmt("%.1fV", data.txBatt), RIGHT) + end + + -- Timer + if config[13].v > 0 then + if data.doLogs and data.time ~= nil then + text(data.nv and 184 or 340, 0, data.time, WARNING_COLOR) + else + lcd.drawTimer(data.nv and 202 or 340, 0, data.timer) + end end + -- Receiver voltage or Crossfire speed if data.rxBatt > 0 and config[14].v == 1 then - text(LCD_W, 0, fmt("%.1fV", data.rxBatt), RIGHT + tmp) + text(LCD_W, 0, frmt("%.1fV", data.rxBatt), RIGHT + tmp) elseif data.crsf then text(LCD_W, 0, (data.rfmd == 2 and 150 or (data.telem and 50 or "--")) .. "Hz", RIGHT + tmp) end + -- Data on config menu if data.configStatus > 0 then color(CUSTOM_COLOR, 12678) -- Dark grey fill(0, 30, 75, (22 * (data.crsf and 1 or 2)) + 14, CUSTOM_COLOR) - lcd.drawRectangle(0, 30, 75, (22 * (data.crsf and 1 or 2)) + 14, TEXT_COLOR) + rect(0, 30, 75, (22 * (data.crsf and 1 or 2)) + 14, TEXT_COLOR) text(4, 37, "Sats:", 0) text(72, 37, data.satellites % 100, RIGHT + tmp) if not data.crsf then @@ -52,12 +58,17 @@ local function title(data, config, SMLCD) end end - --[[ Show FPS - data.frames = data.frames + 1 - text(180, 0, fmt("%.1f", data.frames / (getTime() - data.fpsStart) * 100), RIGHT) - --text(130, 0, fmt("%.1f", math.min(100 / (getTime() - data.start), 20)), RIGHT) - ]] + --[[ Show FPS ]] + if data.nv then + data.frames = data.frames + 1 + --text(data.nv and 75 or 130, 0, frmt("%.1f", math.min(100 / (getTime() - data.start), 20)), RIGHT) + text(data.nv and 115 or 180, 0, frmt("%.1f", data.frames / (getTime() - data.fpsStart) * 100), RIGHT) + end + --[[ Show usage + text(data.nv and 75 or 130, 0, getUsage() .. "%", RIGHT) + ]] + -- Reset colors color(WARNING_COLOR, YELLOW) if data.widget then @@ -66,20 +77,20 @@ local function title(data, config, SMLCD) color(WARNING_COLOR, iNavZone.options.Warning) end end + end local function gpsDegMin(c, lat) local gpsD = math.floor(math.abs(c)) local gpsM = math.floor((math.abs(c) - gpsD) * 60) - return string.format("%d\64%d'%05.2f\"", gpsD, gpsM, ((math.abs(c) - gpsD) * 60 - gpsM) * 60) .. (lat and (c >= 0 and "N" or "S") or (c >= 0 and "E" or "W")) + return frmt(data.nv and "%d\64%d'%04.1f\"" or "%d\64%d'%05.2f\"", gpsD, gpsM, ((math.abs(c) - gpsD) * 60 - gpsM) * 60) .. (lat and (c >= 0 and dir[0] or dir[4]) or (c >= 0 and dir[2] or dir[6])) end local function hdopGraph(x, y) - local fill = lcd.drawFilledRectangle lcd.setColor(CUSTOM_COLOR, data.hdop < 11 - config[21].v * 2 and YELLOW or WHITE) for i = 4, 9 do if i > data.hdop then - lcd.setColor(CUSTOM_COLOR, GREY) + lcd.setColor(CUSTOM_COLOR, 33874) end fill(i * 4 + x - 16, y - (i * 3 - 10), 2, i * 3 - 10, CUSTOM_COLOR) end @@ -93,30 +104,56 @@ icons.home = { [2] = Bitmap.open(FILE_PATH .. "pics/homel.png"), } icons.fpv = Bitmap.open(FILE_PATH .. "pics/fpv.png") -icons.bg = Bitmap.open(FILE_PATH .. "pics/bg.png") +icons.bg = Bitmap.open(FILE_PATH .. (data.nv and "pics/bgnv.png" or "pics/bg.png")) icons.roll = Bitmap.open(FILE_PATH .. "pics/roll.png") icons.fg = Bitmap.open(FILE_PATH .. "pics/fg" .. config[30].v .. ".png") data.hcurx_id = getFieldInfo("ail").id data.hcury_id = getFieldInfo("ele").id data.hctrl_id = getFieldInfo("rud").id -data.t6_id = getFieldInfo("trim-t6").id +data.t6_id = getFieldInfo("trim-t6") ~= nil and getFieldInfo("trim-t6").id or nil data.lastevt = 0 data.lastt6 = nil +-- Remove spaces from front of modes to center +for i = 1, #modes do + if modes[i].f == 0 then + while string.sub(modes[i].t, 1, 1) == " " do + modes[i].t = string.sub(modes[i].t, 2) + end + end +end +modes[4].t = "ACRO" + +-- Make sure widget is full screen if type(iNavZone) == "table" and type(iNavZone.zone) ~= "nil" then data.widget = true - if iNavZone.zone.w < 450 or iNavZone.zone.h < 250 then + if iNavZone.zone.w < (data.nv and 280 or 450) or iNavZone.zone.h < (data.nv and 450 or 250) then data.startupTime = math.huge - function icons.nfs() - lcd.drawText(iNavZone.zone.x + 14, iNavZone.zone.y + 16, "Full screen required", SMLSIZE + WARNING_COLOR) + function data.nfs() + text(iNavZone.zone.x + 14, iNavZone.zone.y + 16, "Full screen required", SMLSIZE + WARNING_COLOR) end end end -function icons.clear(event, data) - lcd.setColor(CUSTOM_COLOR, 264) --lcd.RGB(0, 32, 65) +-- Nirvana's drawRectangle function is broken, replace it +if data.nv then + function rect(x, y, w, h, color) + w = w - 1 + h = h - 1 + line(x, y, x + w, y, SOLID, color) + line(x + w, y, x + w, y + h, SOLID, color) + line(x + w, y + h, x, y + h, SOLID, color) + line(x, y + h, x, y, SOLID, color) + end +end + +function data.clear(event) + lcd.setColor(CUSTOM_COLOR, data.nv and (data.configStatus > 0 and 25422 or 12942) or 264) --lcd.RGB(98, 106, 115) / lcd.RGB(50, 82, 115) / lcd.RGB(0, 32, 65) lcd.clear(CUSTOM_COLOR) + lcd.setColor(TEXT_COLOR, WHITE) + lcd.setColor(WARNING_COLOR, data.telem and (data.nv and 65516 or YELLOW) or (data.nv and 64300 or RED)) --lcd.RGB(255, 255, 100) / lcd.RGB(255, 100, 100) + if event == 0 or event == nil then event = 0 if not data.armed then @@ -150,7 +187,7 @@ function icons.clear(event, data) if event == 0 and data.doLogs and getValue(data.hcurx_id) < -940 then event = EVT_EXIT_BREAK -- Left (exit) end - data.lastt6 = getValue(data.t6_id) + data.lastt6 = not data.nv and getValue(data.t6_id) or nil if data.lastt6 == 0 then data.lastt6 = nil end @@ -158,27 +195,28 @@ function icons.clear(event, data) return event end -function icons.menu(config, data, icons, prev) +function data.menu(prev) if config[30].v ~= prev then icons.fg = Bitmap.open(FILE_PATH .. "pics/fg" .. config[30].v .. ".png") end + -- Aircraft symbol preview if data.configStatus == 27 and data.configSelect ~= 0 then - lcd.setColor(CUSTOM_COLOR, 982) -- Sky - lcd.drawFilledRectangle(356, 111, 123, 31, CUSTOM_COLOR) - lcd.setColor(CUSTOM_COLOR, 25121) -- Ground - lcd.drawFilledRectangle(356, 142, 123, 31, CUSTOM_COLOR) - lcd.drawBitmap(icons.fg, 355, 110, 50) - lcd.drawRectangle(355, 110, 125, 64, TEXT_COLOR) + lcd.setColor(CUSTOM_COLOR, data.nv and 13660 or 982) -- Sky + fill(LCD_W - 124, (data.nv and 28 or 111), 123, 31, CUSTOM_COLOR) + lcd.setColor(CUSTOM_COLOR, data.nv and 37799 or 25121) -- Ground + fill(LCD_W - 124, (data.nv and 59 or 142), 123, 31, CUSTOM_COLOR) + lcd.drawBitmap(icons.fg, LCD_W - 125, (data.nv and 27 or 110), 50) + rect(LCD_W - 125, (data.nv and 27 or 110), 125, 64, TEXT_COLOR) end -- Return throttle stick to bottom center if data.stickMsg ~= nil and not data.armed then lcd.setColor(CUSTOM_COLOR, BLACK) - lcd.drawFilledRectangle(20, 128, 439, 30, CUSTOM_COLOR) + fill(data.nv and 6 or 20, data.nv and 270 or 128, data.nv and 308 or 439, 30, CUSTOM_COLOR) lcd.setColor(CUSTOM_COLOR, YELLOW) - lcd.drawRectangle(19, 127, 441, 32, CUSTOM_COLOR) - lcd.drawText(28, 128, data.stickMsg, MIDSIZE + CUSTOM_COLOR) + rect(data.nv and 5 or 19, data.nv and 269 or 127, data.nv and 310 or 441, 32, CUSTOM_COLOR) + text(data.nv and 14 or 28, data.nv and 275 or 128, data.stickMsg, (data.nv and SMLSIZE or MIDSIZE) + CUSTOM_COLOR) end end -return title, gpsDegMin, hdopGraph, icons \ No newline at end of file +return title, gpsDegMin, hdopGraph, icons, rect \ No newline at end of file diff --git a/src/iNav/func_t.lua b/src/iNav/func_t.lua index b22ba09b..73485b06 100644 --- a/src/iNav/func_t.lua +++ b/src/iNav/func_t.lua @@ -1,81 +1,85 @@ -local config, data, FILE_PATH = ... +local config, data, modes, dir, SMLCD, FILE_PATH, text, line, rect, fill, frmt = ... -local function title(data, config, SMLCD) - lcd.drawFilledRectangle(0, 0, LCD_W, 8, FORCE) - lcd.drawText(0, 0, model.getInfo().name, INVERS) +local function title() + fill(0, 0, LCD_W, 8, FORCE) + text(0, 0, model.getInfo().name, INVERS) if data.doLogs and data.time ~= nil then - lcd.drawText(SMLCD and 50 or 145, 1, data.time, SMLSIZE + 3) - lcd.drawLine(SMLCD and 50 or 145, 7, SMLCD and 83 or 178, 7, SOLID, FORCE) + text(SMLCD and 50 or 145, 1, data.time, SMLSIZE + 3) + line(SMLCD and 50 or 145, 7, SMLCD and 83 or 178, 7, SOLID, FORCE) elseif config[13].v > 0 then lcd.drawTimer(SMLCD and 60 or 150, 1, data.timer, SMLSIZE + INVERS) end if config[19].v > 0 then - lcd.drawFilledRectangle(86, 1, 19, 6, ERASE) - lcd.drawLine(105, 2, 105, 5, SOLID, ERASE) + fill(86, 1, 19, 6, ERASE) + line(105, 2, 105, 5, SOLID, ERASE) tmp = math.max(math.min((data.txBatt - data.txBattMin) / (data.txBattMax - data.txBattMin) * 17, 17), 0) + 86 for i = 87, tmp, 2 do - lcd.drawLine(i, 2, i, 5, SOLID, FORCE) + line(i, 2, i, 5, SOLID, FORCE) end end if config[19].v ~= 1 then - lcd.drawText(SMLCD and ((config[14].v == 1 or data.crsf) and 105 or LCD_W) or 128, 1, string.format("%.1fV", data.txBatt), SMLSIZE + RIGHT + INVERS) + text(SMLCD and ((config[14].v == 1 or data.crsf) and 105 or LCD_W) or 128, 1, frmt("%.1fV", data.txBatt), SMLSIZE + RIGHT + INVERS) end if data.rxBatt > 0 and data.telem and config[14].v == 1 then - lcd.drawText(LCD_W, 1, string.format("%.1fV", data.rxBatt), SMLSIZE + RIGHT + INVERS) + text(LCD_W, 1, frmt("%.1fV", data.rxBatt), SMLSIZE + RIGHT + INVERS) elseif data.crsf then - lcd.drawText(LCD_W, 1, (data.rfmd == 2 and 150 or (data.telem and 50 or "--")) .. (SMLCD and "" or "Hz"), SMLSIZE + RIGHT + INVERS) + text(LCD_W, 1, (data.rfmd == 2 and 150 or (data.telem and 50 or "--")) .. (SMLCD and "" or "Hz"), SMLSIZE + RIGHT + INVERS) end - --[[ Show FPS + --[[ Show FPS - Should always be 20fps on Taranis data.frames = data.frames + 1 - lcd.drawText(SMLCD and 57 or 80, 1, string.format("%.1f", data.frames / (getTime() - data.fpsStart) * 100), SMLSIZE + RIGHT + INVERS) + text(SMLCD and 57 or 80, 1, frmt("%.1f", data.frames / (getTime() - data.fpsStart) * 100), SMLSIZE + RIGHT + INVERS) + ]] + + --[[ Show usage + text(SMLCD and 57 or 80, 1, getUsage() .. "%", SMLSIZE + RIGHT + INVERS) ]] end local function gpsDegMin(c, lat) local gpsD = math.floor(math.abs(c)) - return gpsD .. string.format("\64%05.2f", (math.abs(c) - gpsD) * 60) .. (lat and (c >= 0 and "N" or "S") or (c >= 0 and "E" or "W")) + return gpsD .. frmt("\64%05.2f", (math.abs(c) - gpsD) * 60) .. (lat and (c >= 0 and dir[0] or dir[4]) or (c >= 0 and dir[2] or dir[6])) end local function hdopGraph(x, y, s, SMLCD) local tmp = ((data.armed or data.modeId == 6) and data.hdop < 11 - config[21].v * 2) or not data.telem if config[22].v == 0 then if tmp then - lcd.drawText(x, y, " ", SMLSIZE + 3) + text(x, y, " ", SMLSIZE + 3) end for i = 4, 9 do - lcd.drawLine(x - 8 + (i * 2), (data.hdop >= i or not SMLCD) and y + 8 - i or y + 5, x - 8 + (i * 2), y + 5, SOLID, (data.hdop >= i or SMLCD) and 0 or GREY_DEFAULT) + line(x - 8 + (i * 2), (data.hdop >= i or not SMLCD) and y + 8 - i or y + 5, x - 8 + (i * 2), y + 5, SOLID, (data.hdop >= i or SMLCD) and 0 or GREY_DEFAULT) end else - lcd.drawText(x + 12, s == SMLSIZE and y or y - 2, (data.hdop == 0 and not data.gpsFix) and "--" or (9 - data.hdop) * 0.5 + 0.8, s + RIGHT + (tmp and 3 or 0)) + text(x + 12, s == SMLSIZE and y or y - 2, (data.hdop == 0 and not data.gpsFix) and "--" or (9 - data.hdop) * 0.5 + 0.8, s + RIGHT + (tmp and 3 or 0)) end end local icons = {} function icons.gps(x, y) - lcd.drawLine(x + 1, y, x + 5, y + 4, SOLID, 0) - lcd.drawLine(x + 1, y + 1, x + 4, y + 4, SOLID, 0) - lcd.drawLine(x + 1, y + 2, x + 3, y + 4, SOLID, 0) - lcd.drawLine(x, y + 5, x + 2, y + 5, SOLID, 0) + line(x + 1, y, x + 5, y + 4, SOLID, 0) + line(x + 1, y + 1, x + 4, y + 4, SOLID, 0) + line(x + 1, y + 2, x + 3, y + 4, SOLID, 0) + line(x, y + 5, x + 2, y + 5, SOLID, 0) lcd.drawPoint(x + 4, y + 1) lcd.drawPoint(x + 1, y + 4) end function icons.lock(x, y) - lcd.drawRectangle(x, y + 2, 5, 4, 0) - lcd.drawRectangle(x + 1, y, 3, 5, FORCE) + rect(x, y + 2, 5, 4, 0) + rect(x + 1, y, 3, 5, FORCE) end function icons.home(x, y) lcd.drawPoint(x + 3, y - 1) - lcd.drawLine(x + 2, y, x + 4, y, SOLID, 0) - lcd.drawLine(x + 1, y + 1, x + 5, y + 1, SOLID, 0) - lcd.drawLine(x, y + 2, x + 6, y + 2, SOLID, 0) - lcd.drawLine(x + 1, y + 3, x + 1, y + 5, SOLID, 0) - lcd.drawLine(x + 5, y + 3, x + 5, y + 5, SOLID, 0) - lcd.drawLine(x + 2, y + 5, x + 4, y + 5, SOLID, 0) + line(x + 2, y, x + 4, y, SOLID, 0) + line(x + 1, y + 1, x + 5, y + 1, SOLID, 0) + line(x, y + 2, x + 6, y + 2, SOLID, 0) + line(x + 1, y + 3, x + 1, y + 5, SOLID, 0) + line(x + 5, y + 3, x + 5, y + 5, SOLID, 0) + line(x + 2, y + 5, x + 4, y + 5, SOLID, 0) lcd.drawPoint(x + 3, y + 4) end -return title, gpsDegMin, hdopGraph, icons \ No newline at end of file +return title, gpsDegMin, hdopGraph, icons, rect \ No newline at end of file diff --git a/src/iNav/horus.lua b/src/iNav/horus.lua index cc72ce75..718de7a7 100644 --- a/src/iNav/horus.lua +++ b/src/iNav/horus.lua @@ -1,5 +1,6 @@ -local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH) +local function view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) + local rgb = lcd.RGB local SKY = 982 --rgb(0, 121, 180) local GROUND = 25121 --rgb(98, 68, 8) --local SKY2 = 8943 --rgb(32, 92, 122) @@ -17,22 +18,20 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local BOTTOM = 146 local Y_CNTR = 83 --(TOP + BOTTOM) / 2 local DEGV = 160 + local CENTERED = 4 + --local TINSIZE = 256 + --local VERTICAL = 16384 local tmp, tmp2, top2, bot2, pitch, roll, roll1, upsideDown - local text = lcd.drawText - local line = lcd.drawLine - local fill = lcd.drawFilledRectangle local bmap = lcd.drawBitmap - local rgb = lcd.RGB local color = lcd.setColor - local max = math.max - local min = math.min local floor = math.floor + local min = math.min + local max = math.max local abs = math.abs local rad = math.rad local deg = math.deg local sin = math.sin local cos = math.cos - local fmt = string.format function intersect(s1, e1, s2, e2) local d = (s1.x - e1.x) * (s2.y - e2.y) - (s1.y - e1.y) * (s2.x - e2.x) @@ -86,8 +85,6 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic -- Setup bmap(icons.bg, 0, TOP) - color(TEXT_COLOR, WHITE) - color(WARNING_COLOR, data.telem and YELLOW or RED) -- Calculate orientation if data.pitchRoll then @@ -245,7 +242,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic end end if not data.showMax then - text(X_CNTR - 65, Y_CNTR - 9, fmt("%.0f", upsideDown and -tmp or tmp) .. "\64", SMLSIZE + RIGHT) + text(X_CNTR - 65, Y_CNTR - 9, frmt("%.0f", upsideDown and -tmp or tmp) .. "\64", SMLSIZE + RIGHT) end end @@ -265,10 +262,8 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic for i = 0, 348.75, 11.25 do tmp = floor(((i - data.heading + (361 + HEADING_DEG * 0.5)) % 360) * PIXEL_DEG - 2.5) if tmp >= 9 and tmp <= RIGHT_POS - 12 then - if i % 90 == 0 then - text(tmp - (i < 270 and 3 or 5), bot2, i == 0 and "N" or (i == 90 and "E" or (i == 180 and "S" or "W")), SMLSIZE) - elseif i % 45 == 0 then - text(tmp - (i < 225 and 7 or 9), bot2, i == 45 and "NE" or (i == 135 and "SE" or (i == 225 and "SW" or "NW")), SMLSIZE) + if i % 45 == 0 then + text(tmp, bot2, dir[i / 45], CENTERED + SMLSIZE) else line(tmp, BOTTOM - 4, tmp, BOTTOM - 1, SOLID, 0) end @@ -331,7 +326,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic -- Speed & altitude tmp = data.showMax and data.speedMax or data.speed - text(39, Y_CNTR - 9, tmp >= 99.5 and floor(tmp + 0.5) or fmt("%.1f", tmp), SMLSIZE + RIGHT + data.telemFlags) + text(39, Y_CNTR - 9, tmp >= 99.5 and floor(tmp + 0.5) or frmt("%.1f", tmp), SMLSIZE + RIGHT + data.telemFlags) tmp = data.showMax and data.altitudeMax or data.altitude text(RIGHT_POS - 2, Y_CNTR - 9, floor(tmp + 0.5), SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) if data.altHold then @@ -372,7 +367,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic line(RIGHT_POS, y1, RIGHT_POS + 9, y2, SOLID, CUSTOM_COLOR) end if data.startup == 0 then - text(RIGHT_POS + 13, TOP - 1, fmt(abs(data.vspeed) >= 9.95 and "%.0f" or "%.1f", data.vspeed) .. units[data.vspeed_unit], SMLSIZE + data.telemFlags) + text(RIGHT_POS + 13, TOP - 1, frmt(abs(data.vspeed) >= 9.95 and "%.0f" or "%.1f", data.vspeed) .. units[data.vspeed_unit], SMLSIZE + data.telemFlags) end end @@ -392,8 +387,8 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if data.startup == 0 then -- Launch/north-based orientation if data.showDir or data.headingRef == -1 then - text(LEFT_POS + 2, Y_CNTR - 9, "W", SMLSIZE) - text(RIGHT_POS, Y_CNTR - 9, "E", SMLSIZE + RIGHT) + text(LEFT_POS + 2, Y_CNTR - 9, dir[6], SMLSIZE) + text(RIGHT_POS, Y_CNTR - 9, dir[2], SMLSIZE + RIGHT) end local cx, cy, d @@ -459,7 +454,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic line(x2, y2, x3, y3, SOLID, CUSTOM_COLOR) line(x1, y1, x2, y2, SOLID, TEXT_COLOR) line(x1, y1, x3, y3, SOLID, TEXT_COLOR) - tmp = data.distanceLast < 1000 and floor(data.distanceLast + 0.5) .. units[data.dist_unit] or (fmt("%.1f", data.distanceLast / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")) + tmp = data.distanceLast < 1000 and floor(data.distanceLast + 0.5) .. units[data.dist_unit] or (frmt("%.1f", data.distanceLast / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")) text(LEFT_POS + 2, BOTTOM - 16, tmp, SMLSIZE + data.telemFlags) end @@ -499,7 +494,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic end local val = math.floor((data.showMax and data.cellMin or data.cell) * 100 + 0.5) * 0.01 - text(X1 - 3, TOP + 42, fmt(config[1].v == 0 and "%.2fV" or "%.1fV", config[1].v == 0 and val or (data.showMax and data.battMin or data.batt)), MIDSIZE + RIGHT + tmp) + text(X1 - 3, TOP + 42, frmt(config[1].v == 0 and "%.2fV" or "%.1fV", config[1].v == 0 and val or (data.showMax and data.battMin or data.batt)), MIDSIZE + RIGHT + tmp) text(0, TOP + 51, labels[2], SMLSIZE) if data.bl ~= val then local red = val >= config[2].v and max(floor((4.2 - val) / (4.2 - config[2].v) * 255), 0) or 255 @@ -528,17 +523,18 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic text(X1 + 9, TOP + 1, labels[4], SMLSIZE) text(X2, TOP + 12, floor(tmp + 0.5) .. units[data.alt_unit], MIDSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) tmp2 = data.showMax and data.distanceMax or data.distanceLast - tmp = tmp2 < 1000 and floor(tmp2 + 0.5) .. units[data.dist_unit] or (fmt("%.1f", tmp2 / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")) + tmp = tmp2 < 1000 and floor(tmp2 + 0.5) .. units[data.dist_unit] or (frmt("%.1f", tmp2 / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")) text(X1 + 9, TOP + 44, labels[5], SMLSIZE) text(X2, TOP + 55, tmp, MIDSIZE + RIGHT + data.telemFlags) if data.showCurr then tmp = data.showMax and data.currentMax or data.current text(X1 + 9, TOP + 87, labels[3], SMLSIZE) - text(X2, TOP + 98, (tmp >= 99.5 and floor(tmp + 0.5) or fmt("%.1fA", tmp)), MIDSIZE + RIGHT + data.telemFlags) + text(X2, TOP + 98, (tmp >= 99.5 and floor(tmp + 0.5) or frmt("%.1fA", tmp)), MIDSIZE + RIGHT + data.telemFlags) end -- Box 3 (flight modes, orientation) - text(X2 + 20, TOP, modes[data.modeId].t, modes[data.modeId].f == 3 and WARNING_COLOR or 0) + tmp = (X2 + X3) * 0.5 + 4 + text(tmp, TOP, modes[data.modeId].t, CENTERED + (modes[data.modeId].f == 3 and WARNING_COLOR or 0)) if data.altHold then bmap(icons.lock, X1 + 63, TOP + 4) end @@ -548,12 +544,12 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if data.showHead then if data.showDir or data.headingRef == -1 then - text((X2 + X3) * 0.5, TOP + 18, "N", SMLSIZE) - text(X3 - 4, 211, "E", SMLSIZE + RIGHT) - text(X2 + 10, 211, "W", SMLSIZE) - text(X2 + 78, BOTTOM - 15, floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) + text(tmp, TOP + 18, dir[0], CENTERED + SMLSIZE) + text(X3 - 4, 211, dir[2], SMLSIZE + RIGHT) + text(X2 + 10, 211, dir[6], SMLSIZE) + text(tmp + 4, BOTTOM - 15, floor(data.heading + 0.5) % 360 .. "\64", CENTERED + SMLSIZE + data.telemFlags) end - local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, (X2 + X3) * 0.5 + 4, 219, 25) + local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, tmp, 219, 25) if data.headingHold then fill((x2 + x3) * 0.5 - 2, (y2 + y3) * 0.5 - 2, 5, 5, SOLID) else @@ -581,10 +577,10 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if not data.crsf then text(RIGHT_POS, TOP + 28, floor(data.gpsAlt + 0.5) .. (data.gpsAlt_unit == 10 and "'" or units[data.gpsAlt_unit]), MIDSIZE + tmp) end - text(RIGHT_POS, TOP + 54, config[16].v == 0 and fmt("%.6f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), tmp) - text(RIGHT_POS, TOP + 74, config[16].v == 0 and fmt("%.6f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), tmp) + text(RIGHT_POS, TOP + 54, config[16].v == 0 and frmt("%.6f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), tmp) + text(RIGHT_POS, TOP + 74, config[16].v == 0 and frmt("%.6f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), tmp) tmp = data.showMax and data.speedMax or data.speed - text(RIGHT_POS + 1, TOP + 98, tmp >= 99.5 and floor(tmp + 0.5) .. units[data.speed_unit] or fmt("%.1f", tmp) .. units[data.speed_unit], MIDSIZE + RIGHT + data.telemFlags) + text(RIGHT_POS + 1, TOP + 98, tmp >= 99.5 and floor(tmp + 0.5) .. units[data.speed_unit] or frmt("%.1f", tmp) .. units[data.speed_unit], MIDSIZE + RIGHT + data.telemFlags) -- Dividers color(CUSTOM_COLOR, DKGREY) diff --git a/src/iNav/log.lua b/src/iNav/log.lua index 7a33af4f..334e6d39 100644 --- a/src/iNav/log.lua +++ b/src/iNav/log.lua @@ -99,9 +99,9 @@ local function playLog(data, config, distCalc, date, NEXT, PREV) if pos == nil then pos = string.len(raw) end - local line = string.sub(raw, 0, pos) + local ln = string.sub(raw, 0, pos) raw = string.sub(raw, pos + 1) - record = parseLine(line) + record = parseLine(ln) end end @@ -143,12 +143,14 @@ local function playLog(data, config, distCalc, date, NEXT, PREV) if data.a4_id > -1 then data.cell = toNum(record[label.a4]) end - pos = string.find(record[label.gps], " ") - if pos ~= nil then - gpsTemp = { - lat = toNum(string.sub(record[label.gps], 0, pos - 1)), - lon = toNum(string.sub(record[label.gps], pos + 1)) - } + if record[label.gps] ~= nil then + pos = string.find(record[label.gps], " ") + if pos ~= nil then + gpsTemp = { + lat = toNum(string.sub(record[label.gps], 0, pos - 1)), + lon = toNum(string.sub(record[label.gps], pos + 1)) + } + end end data.telem = true data.telemFlags = 0 diff --git a/src/iNav/menu.lua b/src/iNav/menu.lua index 39a75deb..cc7daba7 100644 --- a/src/iNav/menu.lua +++ b/src/iNav/menu.lua @@ -1,17 +1,12 @@ -local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, getTelemetryUnit, FILE_PATH, SMLCD, FLASH, PREV, NEXT, HORUS, env) +local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, getTelemetryUnit, SMLCD, FLASH, PREV, NEXT, HORUS, text, rect, fill, frmt, env) - local CONFIG_X = HORUS and 90 or (SMLCD and 0 or 46) - local TOP = HORUS and 37 or 11 - local LINE = HORUS and 22 or 9 + local CONFIG_X = HORUS and (data.nv and 10 or 90) or (SMLCD and 0 or 46) + local TOP = HORUS and (data.nv and 107 or 37) or 11 + local HIGH = HORUS and (data.nv and 28 or 22) or 9 local RSIDE = HORUS and 200 or 83 local GPS = HORUS and 45 or 21 - local ROWS = HORUS and 9 or 5 + local ROWS = HORUS and (data.nv and 12 or 9) or 5 local FONT = HORUS and 0 or SMLSIZE - local text = lcd.drawText - local min = math.min - local max = math.max - local floor = math.floor - local format = string.format local offOn = {[0] = "Off", "On"} -- Config options: o=display Order / t=Text / c=Characters / v=default Value / l=Lookup text / d=Decimal / m=Min / x=maX / i=Increment / a=Append text @@ -44,7 +39,7 @@ local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, { t = "AltHold Center FB",l = 1 }, -- 26 { t = "Battery Capacity", m = 150, i = 50, a = "mAh" }, -- 27 { t = "Altitude Graph", l = {[0] = "Off", 1, 2, 3, 4, 5, 6}, a = " Min" }, -- 28 - { t = "Cell Calculation", m = 4.2, i = 0.1, a = "V" }, -- 29 + { t = "Cell Calculation", m = 3.5, i = 0.1, a = "V" }, -- 29 { t = "Aircraft Symbol", a = "" }, -- 30 { t = "Center Map Home", l = 1 }, -- 31 { t = "Orientation", l = {[0] = "Launch", "Compass"} }, -- 32 @@ -58,17 +53,19 @@ local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, end if HORUS then - lcd.setColor(CUSTOM_COLOR, GREY) - lcd.drawFilledRectangle(CONFIG_X - 10, TOP - 7, LCD_W - CONFIG_X * 2 + 20, LINE * (ROWS + 1) + 12, CUSTOM_COLOR) - lcd.setColor(CUSTOM_COLOR, 12678) -- Dark grey - end - if not SMLCD then - lcd.drawRectangle(CONFIG_X - (HORUS and 10 or 5), TOP - (HORUS and 7 or 2), LCD_W - CONFIG_X * 2 + (HORUS and 20 or 10), LINE * (ROWS + 1) + (HORUS and 12 or 1), SOLID) + if not data.nv then + lcd.setColor(CUSTOM_COLOR, GREY) + fill(CONFIG_X - 10, TOP - 7, LCD_W - CONFIG_X * 2 + 20, HIGH * (ROWS + 1) + 12, CUSTOM_COLOR) + end + rect(CONFIG_X - 10, TOP - 7, LCD_W - CONFIG_X * 2 + 20, HIGH * (ROWS + 1) + 12, TEXT_COLOR) + lcd.setColor(CUSTOM_COLOR, data.nv and LIGHTGREY or 12678) -- Dark grey + elseif not SMLCD then + rect(CONFIG_X - 5, TOP - 2, LCD_W - CONFIG_X * 2 + 10, HIGH * (ROWS + 1) + 1, SOLID) end -- Special limit cases - config[19].x = SMLCD and ((config[14].v == 1 or data.crsf) and 1 or 2) or 2 - config[19].v = min(config[19].x, config[19].v) + config[19].x = SMLCD and ((config[14].v == 1 or data.crsf) and 1 or 2) or (data.nv and 1 or 2) + config[19].v = math.min(config[19].x, config[19].v) config[25].x = config[28].v == 0 and 2 or 3 if config[28].v == 0 and config[25].v == 3 then config[25].v = 2 @@ -129,7 +126,7 @@ local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, if data.configSelect == 0 then if event == NEXT or event == EVT_DOWN_REPT or event == EVT_MINUS_REPT then -- Next option data.configStatus = data.configStatus == #config and 1 or data.configStatus + 1 - data.configTop = data.configStatus > min(#config, data.configTop + ROWS) and data.configTop + 1 or (data.configStatus == 1 and 1 or data.configTop) + data.configTop = data.configStatus > math.min(#config, data.configTop + ROWS) and data.configTop + 1 or (data.configStatus == 1 and 1 or data.configTop) elseif event == PREV or event == EVT_UP_REPT or event == EVT_PLUS_REPT then -- Previous option data.configStatus = data.configStatus == 1 and #config or data.configStatus - 1 data.configTop = data.configStatus < data.configTop and data.configTop - 1 or (data.configStatus == #config and #config - ROWS or data.configTop) @@ -139,10 +136,10 @@ local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, end -- Delete invisible menus - local bottom = min(#config, data.configTop + ROWS) - for line = 1, #config do - if line < data.configTop or line > bottom then - config2[config[line].z] = nil + local bottom = math.min(#config, data.configTop + ROWS) + for i = 1, #config do + if i < data.configTop or i > bottom then + config2[config[i].z] = nil end end collectgarbage() @@ -153,22 +150,22 @@ local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, local i = config2[z].i == nil and 1 or config2[z].i if event == EVT_EXIT_BREAK then data.configSelect = 0 - elseif event == NEXT or event == EVT_UP_REPT or event == EVT_PLUS_REPT then - config[z].v = min(floor(config[z].v * 10 + i * 10) * 0.1, config[z].x == nil and 1 or config[z].x) - elseif event == PREV or event == EVT_DOWN_REPT or event == EVT_MINUS_REPT then - config[z].v =max(floor(config[z].v * 10 - i * 10) * 0.1, config2[z].m == nil and 0 or config2[z].m) + elseif event == NEXT or event == EVT_DOWN_REPT or event == EVT_MINUS_REPT then + config[z].v = math.min(math.floor(config[z].v * 10 + i * 10) * 0.1, config[z].x == nil and 1 or config[z].x) + elseif event == PREV or event == EVT_UP_REPT or event == EVT_PLUS_REPT then + config[z].v =math.max(math.floor(config[z].v * 10 - i * 10) * 0.1, config2[z].m == nil and 0 or config2[z].m) end -- Special cases if event ~= 0 and event ~= nil then if z == 2 then -- Cell low > critical - config[2].v = max(config[2].v, config[3].v + 0.1) + config[2].v = math.max(config[2].v, config[3].v + 0.1) elseif z == 3 then -- Cell critical < low - config[3].v = min(config[3].v, config[2].v - 0.1) + config[3].v = math.min(config[3].v, config[2].v - 0.1) elseif z == 18 then -- Fuel low > critical - config[18].v = max(config[18].v, config[17].v + 1) + config[18].v = math.max(config[18].v, config[17].v + 1) elseif z == 17 then -- Fuel critical < low - config[17].v = min(config[17].v, config[18].v - 1) + config[17].v = math.min(config[17].v, config[18].v - 1) elseif z == 20 then -- Speed sensor local tmp = config[20].v == 0 and "GSpd" or "ASpd" data.speed_id = getTelemetryId(tmp) @@ -179,27 +176,27 @@ local function view(data, config, units, lang, event, gpsDegMin, getTelemetryId, data.alt[i] = 0 end elseif i > 1 then - config[z].v = floor(config[z].v / i) * i + config[z].v = math.floor(config[z].v / i) * i end end end -- Print screen - for line = data.configTop, bottom do - local y = (line - data.configTop) * LINE + TOP - local z = config[line].z - local tmp = (data.configStatus == line and INVERS + data.configSelect or 0) + (config[z].d ~= nil and PREC1 or 0) + for i = data.configTop, bottom do + local y = (i - data.configTop) * HIGH + TOP + local z = config[i].z + local tmp = (data.configStatus == i and INVERS + data.configSelect or 0) if config2[z].p == 1 and HORUS then tmp = tmp + CUSTOM_COLOR end text(CONFIG_X, y, config2[z].t, FONT + ((config2[z].p == 1 and HORUS) and CUSTOM_COLOR or 0)) if config2[z].p == nil then if config2[z].l == nil then - text(CONFIG_X + RSIDE, y, (config[z].d ~= nil and format("%.1f", config[z].v) or config[z].v) .. config2[z].a, FONT + tmp) + text(CONFIG_X + RSIDE, y, (config[z].d ~= nil and frmt("%.1f", config[z].v) or config[z].v) .. config2[z].a, FONT + tmp) else if config2[z].l == 0 then if config[z].v == 0 then - config2[z].l = { [0] = format("%10.6f %11.6f", data.lastLock.lat, data.lastLock.lon) } + config2[z].l = { [0] = frmt("%10.6f %11.6f", data.lastLock.lat, data.lastLock.lon) } else config2[z].l = { gpsDegMin(data.lastLock.lat, true) .. " " .. gpsDegMin(data.lastLock.lon, false) } end diff --git a/src/iNav/modes.lua b/src/iNav/modes.lua index ef38585a..9a02f1c9 100644 --- a/src/iNav/modes.lua +++ b/src/iNav/modes.lua @@ -1,3 +1,5 @@ +local HORUS = ... + -- Modes: t=text / f=flags for text / w=wave file local modes = { { t = "! TELEM !", f = 3 }, @@ -15,8 +17,10 @@ local modes = { { t = " CRUISE", f = 0, w = "cruzmd" } } -local units = { [0] = "", "V", "A", "mA", "kts", "m/s", "f/s", "kmh", "MPH", "m", LCD_W >= 480 and "ft" or "'" } +local units = { [0] = "", "V", "A", "mA", "kts", "m/s", "f/s", "kmh", "MPH", "m", HORUS and "ft" or "'" } local labels = { "Fuel", "Battery", "Current", "Altitude", "Distance" } -return modes, units, labels \ No newline at end of file +local dir = { [0] = "N", "NE", "E", "SE", "S", "SW", "W", "NW" } + +return modes, units, labels, dir \ No newline at end of file diff --git a/src/iNav/nirvana.lua b/src/iNav/nirvana.lua new file mode 100644 index 00000000..4f1402cd --- /dev/null +++ b/src/iNav/nirvana.lua @@ -0,0 +1,624 @@ +local function view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) + + local rgb = lcd.RGB + local SKY = 13660 --rgb(50, 171, 230) + local GROUND = 37799 --rgb(148, 118, 58) + --local SKY2 = 8943 --rgb(32, 92, 122) + --local GROUND2 = 20996 --rgb(81, 65, 36) + local MAP = 13478 --rgb(50, 150, 50) + --local DKMAP = 544 --rgb(0, 70, 0) + local LIGHTMAP = rgb(50, 200, 50) + --local DATA = 12942 --rgb(50, 82, 115) + local DKGREY = 33874 --rgb(98, 106, 115) + local RIGHT_POS = 270 + local X_CNTR = 134 --(RIGHT_POS + LEFT_POS [0]) / 2 - 1 + local HEADING_DEG = 190 + local PIXEL_DEG = RIGHT_POS / HEADING_DEG --(RIGHT_POS - LEFT_POS [0]) / HEADING_DEG + local TOP = 20 + local BOTTOM = 146 + local Y_CNTR = 83 --(TOP + BOTTOM) / 2 + local DEGV = 160 + local CENTERED = 4 + --local TINSIZE = 256 + --local VERTICAL = 16384 + local tmp, tmp2, top2, bot2, pitch, roll, roll1, upsideDown + local bmap = lcd.drawBitmap + local color = lcd.setColor + local floor = math.floor + local min = math.min + local max = math.max + local abs = math.abs + local rad = math.rad + local deg = math.deg + local sin = math.sin + local cos = math.cos + + function intersect(s1, e1, s2, e2) + local d = (s1.x - e1.x) * (s2.y - e2.y) - (s1.y - e1.y) * (s2.x - e2.x) + local a = s1.x * e1.y - s1.y * e1.x + local b = s2.x * e2.y - s2.y * e2.x + local x = (a * (s2.x - e2.x) - (s1.x - e1.x) * b) / d + local y = (a * (s2.y - e2.y) - (s1.y - e1.y) * b) / d + if x < min(s2.x, e2.x) - 1 or x > max(s2.x, e2.x) + 1 or y < min(s2.y, e2.y) - 1 or y > max(s2.y, e2.y) + 1 then + return nil, nil + end + return floor(x + 0.5), floor(y + 0.5) + end + + local function pitchLadder(r, adj) + --[[ Caged mode + local x = sin(roll1) * r + local y = cos(roll1) * r + local p = cos(rad(pitch - adj)) * DEGV + local x1, y1, x2, y2 = X_CNTR - x, Y_CNTR + y - p, X_CNTR + x, Y_CNTR - y - p + ]] + -- Uncaged mode + local p = sin(rad(adj)) * DEGV + local y = (Y_CNTR - cos(rad(pitch)) * DEGV) - sin(roll1) * p + if y > top2 and y < bot2 then + local x = X_CNTR - cos(roll1) * p + local xd = sin(roll1) * r + local yd = cos(roll1) * r + local x1, y1, x2, y2 = x - xd, y + yd, x + xd, y - yd + if (y1 > top2 or y2 > top2) and (y1 < bot2 or y2 < bot2) and x1 >= 0 and x2 >= 0 then + color(CUSTOM_COLOR, r == 20 and WHITE or LIGHTGREY) + line(x1, y1, x2, y2, SOLID, CUSTOM_COLOR) + if r == 20 and y1 > top2 and y1 < bot2 then + text(x1, y1 - 8, upsideDown and -adj or adj, SMLSIZE + RIGHT) + end + end + end + end + + local function tics(v, p) + tmp = floor((v + 25) * 0.1) * 10 + for i = tmp - 40, tmp, 5 do + local tmp2 = Y_CNTR + ((v - i) * 3) - 9 + if tmp2 > 10 and tmp2 < BOTTOM - 8 then + line(p, tmp2 + 8, p + 2, tmp2 + 8, SOLID, TEXT_COLOR) + --[[ Currently disabled due to preformance + if config[28].v == 0 and i % 10 == 0 and (i >= 0 or p > X_CNTR) and tmp2 < BOTTOM - 23 then + text(p + (p > X_CNTR and -1 or 4), tmp2, i, SMLSIZE + (p > X_CNTR and RIGHT or 0) + TEXT_COLOR) + end + ]] + end + end + end + + -- Setup + bmap(icons.bg, 0, TOP) + + -- Calculate orientation + if data.pitchRoll then + pitch = (abs(data.roll) > 900 and -1 or 1) * (270 - data.pitch * 0.1) % 180 + roll = (270 - data.roll * 0.1) % 180 + upsideDown = abs(data.roll) > 900 + else + pitch = 90 - deg(math.atan2(data.accx * (data.accz >= 0 and -1 or 1), math.sqrt(data.accy * data.accy + data.accz * data.accz))) + roll = 90 - deg(math.atan2(data.accy * (data.accz >= 0 and 1 or -1), math.sqrt(data.accx * data.accx + data.accz * data.accz))) + upsideDown = data.accz < 0 + end + roll1 = rad(roll) + top2 = config[33].v == 0 and TOP or TOP + 20 + bot2 = BOTTOM - 16 + local i = { {}, {} } + local tl = { x = 1, y = TOP } + local tr = { x = RIGHT_POS - 2, y = TOP } + local bl = { x = 1, y = BOTTOM - 1 } + local br = { x = RIGHT_POS - 2, y = BOTTOM - 1 } + local skip = false + + -- Calculate horizon (uses simple "caged" mode for less math) + local x = sin(roll1) * 200 + local y = cos(roll1) * 200 + local p = cos(rad(pitch)) * DEGV + local h1 = { x = X_CNTR + x, y = Y_CNTR - y - p } + local h2 = { x = X_CNTR - x, y = Y_CNTR + y - p } + + -- Find intersections between horizon and edges of attitude indicator + local x1, y1 = intersect(h1, h2, tl, bl) + local x2, y2 = intersect(h1, h2, tr, br) + if x1 and x2 then + i[1].x, i[1].y = x1, y1 + i[2].x, i[2].y = x2, y2 + else + local x3, y3 = intersect(h1, h2, bl, br) + local x4, y4 = intersect(h1, h2, tl, tr) + if x3 and x4 then + i[1].x, i[1].y = x3, y3 + i[2].x, i[2].y = x4, y4 + elseif (x1 or x2) and (x3 or x4) then + i[1].x, i[1].y = x1 and x1 or x2, y1 and y1 or y2 + i[2].x, i[2].y = x3 and x3 or x4, y3 and y3 or y4 + else + skip = true + end + end + + -- Draw ground + color(CUSTOM_COLOR, GROUND) + if skip then + -- Must be going down hard! + if (pitch - 90) * (upsideDown and -1 or 1) < 0 then + fill(tl.x, tl.y, br.x - tl.x + 1, br.y - tl.y + 1, CUSTOM_COLOR) + end + else + local trix, triy + + -- Find right angle coordinates of triangle + if upsideDown then + trix = roll > 90 and max(i[1].x, i[2].x) or min(i[1].x, i[2].x) + triy = min(i[1].y, i[2].y) + else + trix = roll > 90 and min(i[1].x, i[2].x) or max(i[1].x, i[2].x) + triy = max(i[1].y, i[2].y) + end + + -- Find rectangle(s) and fill + if upsideDown then + if triy > tl.y then + fill(tl.x, tl.y, br.x - tl.x + 1, triy - tl.y, CUSTOM_COLOR) + end + if roll > 90 and trix < br.x then + fill(trix, triy, br.x - trix + 1, br.y - triy + 1, CUSTOM_COLOR) + elseif roll <= 90 and trix > tl.x then + fill(tl.x, triy, trix - tl.x, br.y - triy + 1, CUSTOM_COLOR) + end + else + if triy < br.y then + fill(tl.x, triy + 1, br.x - tl.x + 1, br.y - triy, CUSTOM_COLOR) + end + if roll > 90 and trix > tl.x then + fill(tl.x, tl.y, trix - tl.x, triy - tl.y + 1, CUSTOM_COLOR) + elseif roll <= 90 and trix < br.x then + fill(trix, tl.y, br.x - trix + 1, triy - tl.y + 1, CUSTOM_COLOR) + end + end + + -- Fill remaining triangle + local height = i[1].y - triy + local top = i[1].y + if height == 0 then + height = i[2].y - triy + top = i[2].y + end + local inc = 1 + if height ~= 0 then + local width = abs(i[1].x - trix) + local tx1 = i[1].x + local tx2 = trix + if width == 0 then + width = abs(i[2].x - trix) + tx1 = i[2].x + tx2 = trix + end + inc = abs(height) < 10 and 1 or (abs(height) < 20 and 2 or ((abs(height) < width and abs(roll - 90) < 55) and 3 or 5)) + local steps = height > 0 and inc or -inc + local slope = width / height * inc + local s = slope > 0 and 0 or inc - 1 + slope = abs(slope) * (tx1 < tx2 and 1 or -1) + for y = triy, top, steps do + if abs(steps) == 1 then + line(tx1, y, tx2, y, SOLID, CUSTOM_COLOR) + else + if tx1 < tx2 then + --if tx1 < tx2 and tx2 - tx1 + 1 > 0 then + fill(tx1, y - s, tx2 - tx1 + 1, inc, CUSTOM_COLOR) + else + --elseif tx1 > tx2 and tx1 - tx2 + 1 > 0 then + fill(tx2, y - s, tx1 - tx2 + 1, inc, CUSTOM_COLOR) + end + end + tx1 = tx1 + slope + end + end + + -- Smooth horizon + if not upsideDown and inc <= 3 then + if inc > 1 then + if inc > 2 then + line(i[1].x, i[1].y + 2, i[2].x, i[2].y + 2, SOLID, CUSTOM_COLOR) + end + line(i[1].x, i[1].y + 1, i[2].x, i[2].y + 1, SOLID, CUSTOM_COLOR) + color(CUSTOM_COLOR, SKY) + line(i[1].x, i[1].y - 1, i[2].x, i[2].y - 1, SOLID, CUSTOM_COLOR) + if inc > 2 then + line(i[1].x, i[1].y - 2, i[2].x, i[2].y - 2, SOLID, CUSTOM_COLOR) + end + if 90 - roll > 25 then + line(i[1].x, i[1].y - 3, i[2].x, i[2].y - 3, SOLID, CUSTOM_COLOR) + end + end + color(CUSTOM_COLOR, LIGHTGREY) + line(i[1].x, i[1].y, i[2].x, i[2].y, SOLID, CUSTOM_COLOR) + end + end + + -- Pitch ladder + if data.telem then + tmp = pitch - 90 + local tmp2 = max(min((tmp >= 0 and floor(tmp * 0.2) or math.ceil(tmp * 0.2)) * 5, 30), -30) + for x = tmp2 - 20, tmp2 + 20, 5 do + if x ~= 0 and (x % 10 == 0 or (x > -30 and x < 30)) then + pitchLadder(x % 10 == 0 and 20 or 15, x) + end + end + if not data.showMax then + text(X_CNTR - 60, Y_CNTR - 10, frmt("%.0f", upsideDown and -tmp or tmp) .. "\64", SMLSIZE + RIGHT) + end + end + + -- Speed & altitude tics + if config[28].v == 0 then + tics(data.speed, 1) + tics(data.altitude, RIGHT_POS - 4) + text(42, Y_CNTR - 25, units[data.speed_unit], SMLSIZE + RIGHT) + text(RIGHT_POS - 4, Y_CNTR - 25, "Alt " .. units[data.alt_unit], SMLSIZE + RIGHT) + end + + -- Compass + if data.showHead then + for i = 0, 348.75, 11.25 do + tmp = floor(((i - data.heading + (361 + HEADING_DEG * 0.5)) % 360) * PIXEL_DEG - 2.5) + if tmp >= 9 and tmp <= RIGHT_POS - 12 then + if i % 45 == 0 then + text(tmp, bot2, dir[i / 45], CENTERED + SMLSIZE) + else + line(tmp, BOTTOM - 4, tmp, BOTTOM - 1, SOLID, 0) + end + end + end + end + + -- Calculate the maximum distance for scaling home location and map + local maxDist = max(min(data.distanceMax, data.distanceLast * 6), data.distRef * 10) + + -- Home direction + if data.showHead and data.armed and data.telem and data.gpsHome ~= false then + if data.distanceLast >= data.distRef then + local bearing = calcBearing(data.gpsHome, data.gpsLatLon) + 540 % 360 + if config[15].v == 1 then + -- HUD method + local d = 1 - data.distanceLast / maxDist + local w = HEADING_DEG / (d + 1) + local h = floor((((upsideDown and data.heading - bearing or bearing - data.heading) + (361 + w * 0.5)) % 360) * (RIGHT_POS / w) - 0.5) + --local p = sin(math.atan(data.altitude / data.distanceLast) - math.atan(data.altitude / max(maxDist, data.altitude * 0.25))) * (upsideDown and DEGV or -DEGV) + --local p = sin(rad(d * max(15 + (pitch - 90) * 0.5, 0))) * (upsideDown and DEGV or -DEGV) + local p = sin(math.atan(data.altitude / data.distanceLast * 0.5)) * (upsideDown and DEGV or -DEGV) + local x = (X_CNTR - cos(roll1) * p) + (sin(roll1) * (h - X_CNTR)) - 9 + local y = ((Y_CNTR - cos(rad(pitch)) * DEGV) - sin(roll1) * p) - (cos(roll1) * (h - X_CNTR)) - 9 + if x >= 0 and x < RIGHT_POS - 17 then + local s = floor(d * 2 + 0.5) + bmap(icons.home[s], x, min(max(y, s == 2 and TOP or 15), BOTTOM - (s == 2 and 35 or 30))) + end + --[[ + if x >= 0 and y >= TOP and x < RIGHT_POS - 17 and y < BOTTOM - 17 then + bmap(icons.home[floor(d * 2 + 0.5)], x, y) + end + ]] + else + -- Bottom-fixed method + local home = floor(((bearing - data.heading + (361 + HEADING_DEG * 0.5)) % 360) * PIXEL_DEG - 2.5) + if home >= 3 and home <= RIGHT_POS - 6 then + bmap(icons.home[1], home - 7, BOTTOM - 31) + end + end + end + -- Flight path vector + if data.fpv_id > -1 and data.speed >= 8 then + tmp = (data.fpv - data.heading + 360) % 360 + if tmp >= 302 or tmp <= 57 then + local fpv = floor(((data.fpv - data.heading + (361 + HEADING_DEG * 0.5)) % 360) * PIXEL_DEG - 0.5) + --local p = sin(rad(data.vspeed_id == -1 and pitch - 90 or math.log(1 + min(abs(0.6 * (data.vspeed_unit == 6 and data.vspeed * 0.3048 or data.vspeed)), 10)) * (data.vspeed < 0 and -5 or 5))) * DEGV + local p = sin(data.vspeed_id == -1 and rad(pitch - 90) or (math.tan(data.vspeed / (data.speed * (data.speed_unit == 8 and 1.4667 or 0.2778))))) * DEGV + local x = (X_CNTR - cos(roll1) * p) + (sin(roll1) * (fpv - X_CNTR)) - 9 + local y = ((Y_CNTR - cos(rad(pitch)) * DEGV) - sin(roll1) * p) - (cos(roll1) * (fpv - X_CNTR)) - 6 + if y > TOP and y < bot2 and x >= 0 then + bmap(icons.fpv, x, y) + end + end + end + end + + -- View overlay + bmap(icons.fg, 1, 20) + --line(0, BOTTOM, RIGHT_POS, BOTTOM, SOLID, DKGREY) + color(CUSTOM_COLOR, MAP) + line(0, BOTTOM + 1, RIGHT_POS, BOTTOM + 1, SOLID, CUSTOM_COLOR) + + -- Speed & altitude + tmp = data.showMax and data.speedMax or data.speed + text(41, Y_CNTR - 9, tmp >= 99.5 and floor(tmp + 0.5) or frmt("%.1f", tmp), SMLSIZE + RIGHT + data.telemFlags) + tmp = data.showMax and data.altitudeMax or data.altitude + text(RIGHT_POS, Y_CNTR - 9, floor(tmp + 0.5), SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + if data.altHold then + bmap(icons.lock, RIGHT_POS - 55, Y_CNTR - 5) + end + + -- Heading + if data.showHead then + text(X_CNTR + 20, bot2, floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) + end + + -- Roll scale + if config[33].v == 1 then + bmap(icons.roll, 43, 20) + if roll > 30 and roll < 150 and not upsideDown then + local x1, y1, x2, y2, x3, y3 = calcDir(rad(roll - 90), rad(roll + 55), rad(roll - 235), X_CNTR - (cos(roll1) * 75), 79 - (sin(roll1) * 40), 7) + color(CUSTOM_COLOR, YELLOW) + line(x1, y1, x2, y2, SOLID, CUSTOM_COLOR) + line(x1, y1, x3, y3, SOLID, CUSTOM_COLOR) + line(x2, y2, x3, y3, SOLID, CUSTOM_COLOR) + end + end + + --[[ Variometer + if config[7].v % 2 == 1 then + color(CUSTOM_COLOR, DKGREY) + fill(RIGHT_POS, TOP, 10, BOTTOM - 20, CUSTOM_COLOR) + color(CUSTOM_COLOR, LIGHTGREY) + line(RIGHT_POS + 10, TOP, RIGHT_POS + 10, BOTTOM - 1, SOLID, CUSTOM_COLOR) + color(CUSTOM_COLOR, GREY) + line(RIGHT_POS, Y_CNTR - 1, RIGHT_POS + 9, Y_CNTR - 1, SOLID, CUSTOM_COLOR) + if data.telem then + color(CUSTOM_COLOR, YELLOW) + tmp = math.log(1 + min(abs(0.6 * (data.vspeed_unit == 6 and data.vspeed * 0.3048 or data.vspeed)), 10)) * (data.vspeed < 0 and -1 or 1) + local y1 = Y_CNTR - (tmp * 0.416667 * (Y_CNTR - 21)) + local y2 = Y_CNTR - (tmp * 0.384615 * (Y_CNTR - 21)) + line(RIGHT_POS, y1 - 1, RIGHT_POS + 9, y2 - 1, SOLID, CUSTOM_COLOR) + line(RIGHT_POS, y1, RIGHT_POS + 9, y2, SOLID, CUSTOM_COLOR) + end + if data.startup == 0 then + text(RIGHT_POS + 13, TOP - 1, frmt(abs(data.vspeed) >= 9.95 and "%.0f" or "%.1f", data.vspeed) .. units[data.vspeed_unit], SMLSIZE + data.telemFlags) + end + end + ]] + + -- RSSI/LQ + tmp = (not data.telem or data.rssi < data.rssiLow) and FLASH or 0 + local val = data.showMax and data.rssiMin or data.rssiLast + text(LCD_W + 1, BOTTOM - 19, val .. (data.crsf and "%" or "dB"), RIGHT + tmp) + text(LCD_W - 43, TOP, data.crsf and " LQ" or "RSSI", SMLSIZE) + if data.rl ~= val then + local red = val >= data.rssiLow and max(floor((100 - val) / (100 - data.rssiLow) * 255), 0) or 255 + local green = val < data.rssiLow and max(floor((val - data.rssiCrit) / (data.rssiLow - data.rssiCrit) * 255), 0) or 255 + data.rc = rgb(red, green, 60) + data.rl = val + end + color(CUSTOM_COLOR, data.rc) + rect(LCD_W - 32, TOP + 20, 15, 88, CUSTOM_COLOR) + local h = math.floor(max(1, (min(val, 100) * 0.01) * 86)) + fill(LCD_W - 31, TOP + 107 - h, 13, h, CUSTOM_COLOR) + + -- Calc orientation + tmp = data.headingRef + if data.showDir or data.headingRef == -1 then + tmp = 0 + end + local r1 = rad(data.heading - tmp) + local r2 = rad(data.heading - tmp + 145) + local r3 = rad(data.heading - tmp - 145) + + -- Radar + local LEFT_POS = 0 + RIGHT_POS = 319 + X_CNTR = 159 --(RIGHT_POS + LEFT_POS) * 0.5 - 1 + BOTTOM = 297 --BOTTOM + 150 + Y_CNTR = 222 --BOTTOM - 75 + if data.startup == 0 then + -- Launch/north-based orientation + if data.showDir or data.headingRef == -1 then + text(LEFT_POS + 2, Y_CNTR - 9, dir[6], SMLSIZE) + text(RIGHT_POS, Y_CNTR - 9, dir[2], SMLSIZE + RIGHT) + end + local cx, cy, d + + -- Altitude graph + if config[28].v > 0 then + local factor = 30 / (data.altMax - data.altMin) + color(CUSTOM_COLOR, LIGHTMAP) + for i = 1, 60 do + cx = RIGHT_POS - 60 + i + cy = floor(BOTTOM - (data.alt[((data.altCur - 2 + i) % 60) + 1] - data.altMin) * factor + 0.5) + if cy < BOTTOM then + line(cx, cy, cx, BOTTOM - 1, SOLID, CUSTOM_COLOR) + end + if (i - 1) % (60 / config[28].v) == 0 then + color(CUSTOM_COLOR, BLACK) + line(cx, BOTTOM - 30, cx, BOTTOM - 1, DOTTED, CUSTOM_COLOR) + color(CUSTOM_COLOR, LIGHTMAP) + end + end + if data.altMin < -1 then + cy = BOTTOM - (-data.altMin * factor) + color(CUSTOM_COLOR, BLACK) + line(RIGHT_POS - 58, cy, RIGHT_POS - 1, cy, DOTTED, CUSTOM_COLOR) + if cy < 142 then + text(RIGHT_POS - 59, cy - 8, "0", SMLSIZE + RIGHT) + end + end + text(RIGHT_POS + 2, BOTTOM - 46, floor(data.altMax + 0.5) .. units[data.alt_unit], SMLSIZE + RIGHT) + end + + if data.gpsHome ~= false then + -- Craft location + tmp2 = config[31].v == 1 and 65 or 130 + d = data.distanceLast >= data.distRef and min(max(data.distanceLast / maxDist * tmp2, 7), tmp2) or 1 + local bearing = calcBearing(data.gpsHome, data.gpsLatLon) - tmp + local rad1 = rad(bearing) + cx = floor(sin(rad1) * d + 0.5) + cy = floor(cos(rad1) * d + 0.5) + -- Home position + local hx = X_CNTR + 2 + local hy = Y_CNTR + if config[31].v ~= 1 then + hx = hx - (d > 9 and cx * 0.5 or 0) + hy = hy + (d > 9 and cy * 0.5 or 0) + end + if d >= 12 then + --bmap(icons.home, hx - 4, hy - 5) + bmap(icons.home[1], hx - 8, hy - 10) + elseif d > 1 then + fill(hx - 1, hy - 1, 3, 3, SOLID) + end + -- Shift craft location + cx = d == 1 and X_CNTR + 2 or cx + hx + cy = d == 1 and Y_CNTR or hy - cy + else + cx = X_CNTR + 2 + cy = Y_CNTR + d = 1 + end + -- Orientation + local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, cx, cy, 8) + color(CUSTOM_COLOR, LIGHTGREY) + line(x2, y2, x3, y3, SOLID, CUSTOM_COLOR) + line(x1, y1, x2, y2, SOLID, TEXT_COLOR) + line(x1, y1, x3, y3, SOLID, TEXT_COLOR) + tmp = data.distanceLast < 1000 and floor(data.distanceLast + 0.5) .. units[data.dist_unit] or (frmt("%.1f", data.distanceLast / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")) + text(LEFT_POS + 2, BOTTOM - 16, tmp, SMLSIZE + data.telemFlags) + end + + -- Startup message + if data.startup == 2 then + color(CUSTOM_COLOR, BLACK) + text(X_CNTR - 78, 192, "Lua Telemetry", MIDSIZE + CUSTOM_COLOR) + text(X_CNTR - 38, 222, "v" .. VERSION, MIDSIZE + CUSTOM_COLOR) + text(X_CNTR - 79, 191, "Lua Telemetry", MIDSIZE) + text(X_CNTR - 39, 221, "v" .. VERSION, MIDSIZE) + end + + -- Data + local X1 = 140 - 149 + local X2 = 234 - 144 + local X3 = 346 - 149 + TOP = BOTTOM + 60 + BOTTOM = 479 + + -- Box 1 (fuel, battery) + tmp = (not data.telem or data.cell < config[3].v or (data.showFuel and config[23].v == 0 and data.fuel <= config[17].v)) and FLASH or 0 + local btop = TOP - 54 + local bleft = 0 + local bwidth = 150 + local bright = 150 + if data.showFuel then + if config[23].v > 0 or (data.crsf and data.showMax) then + text(bright, btop, (data.crsf and data.fuelRaw or data.fuel) .. data.fUnit[data.crsf and 1 or config[23].v], MIDSIZE + RIGHT + tmp) + else + text(bright, btop, data.fuel .. "%", MIDSIZE + RIGHT + tmp) + if data.fl ~= data.fuel then + local red = data.fuel >= config[18].v and max(floor((100 - data.fuel) / (100 - config[18].v) * 255), 0) or 255 + local green = data.fuel < config[18].v and max(floor((data.fuel - config[17].v) / (config[18].v - config[17].v) * 255), 0) or 255 + data.fc = rgb(red, green, 60) + data.fl = data.fuel + end + color(CUSTOM_COLOR, data.fc) + --lcd.drawGauge(bleft, btop + 26, bwidth, 15, min(data.fuel, 99), 100, CUSTOM_COLOR) + rect(bleft, btop + 27, bwidth, 15, CUSTOM_COLOR) + local w = max(1, (min(data.fuel, 100) * 0.01) * (bwidth - 2)) + fill(1, btop + 28, w, 13, CUSTOM_COLOR) + end + text(bleft, btop + ((config[23].v > 0 or (data.crsf and data.showMax)) and 23 or 9), labels[1], SMLSIZE) + end + + bleft = 170 + bright = LCD_W - 1 + val = math.floor((data.showMax and data.cellMin or data.cell) * 100 + 0.5) * 0.01 + text(bright, btop, frmt(config[1].v == 0 and "%.2fV" or "%.1fV", config[1].v == 0 and val or (data.showMax and data.battMin or data.batt)), MIDSIZE + RIGHT + tmp) + text(bleft, btop + 9, labels[2], SMLSIZE) + if data.bl ~= val then + local red = val >= config[2].v and max(floor((4.2 - val) / (4.2 - config[2].v) * 255), 0) or 255 + local green = val < config[2].v and max(floor((val - config[3].v) / (config[2].v - config[3].v) * 255), 0) or 255 + data.bc = rgb(red, green, 60) + data.bl = val + end + color(CUSTOM_COLOR, data.bc) + --lcd.drawGauge(bleft, btop + 27, bwidth, 15, min(max(val - config[3].v + 0.1, 0) * (100 / (4.2 - config[3].v + 0.1)), 99), 100, CUSTOM_COLOR) + rect(bleft, btop + 27, bwidth, 15, CUSTOM_COLOR) + local w = max(1, (min(max(val - config[3].v + 0.1, 0) * (100 / (4.2 - config[3].v + 0.1)), 100) * 0.01) * (bwidth - 2)) + fill(bleft + 1, btop + 28, w, 13, CUSTOM_COLOR) + + -- Box 2 (altitude, distance, current) + tmp = data.showMax and data.altitudeMax or data.altitude + text(X1 + 9, TOP + 1, labels[4], SMLSIZE) + text(X2, TOP + 12, floor(tmp + 0.5) .. units[data.alt_unit], MIDSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + tmp2 = data.showMax and data.distanceMax or data.distanceLast + tmp = tmp2 < 1000 and floor(tmp2 + 0.5) .. units[data.dist_unit] or (frmt("%.1f", tmp2 / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")) + text(X1 + 9, TOP + 44, labels[5], SMLSIZE) + text(X2, TOP + 55, tmp, MIDSIZE + RIGHT + data.telemFlags) + if data.showCurr then + tmp = data.showMax and data.currentMax or data.current + text(X1 + 9, TOP + 87, labels[3], SMLSIZE) + text(X2, TOP + 98, (tmp >= 99.5 and floor(tmp + 0.5) or frmt("%.1fA", tmp)), MIDSIZE + RIGHT + data.telemFlags) + end + + -- Box 3 (flight modes, orientation) + tmp = (X2 + X3) * 0.5 + 4 + text(tmp, TOP, modes[data.modeId].t, CENTERED + (modes[data.modeId].f == 3 and WARNING_COLOR or 0)) + if data.altHold then + bmap(icons.lock, X1 + 63, TOP + 4) + end + if data.headFree then + text(X2 + 7, TOP + 19, "HF", FLASH) + end + + if data.showHead then + if data.showDir or data.headingRef == -1 then + text(tmp, TOP + 18, dir[0], CENTERED + SMLSIZE) + text(X3 - 4, 421, dir[2], SMLSIZE + RIGHT) + text(X2 + 10, 421, dir[6], SMLSIZE) + text(tmp + 4, BOTTOM - 15, floor(data.heading + 0.5) % 360 .. "\64", CENTERED + SMLSIZE + data.telemFlags) + end + local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, tmp, 429, 25) + if data.headingHold then + fill((x2 + x3) * 0.5 - 2, (y2 + y3) * 0.5 - 2, 5, 5, SOLID) + else + color(CUSTOM_COLOR, DKGREY) + line(x2, y2, x3, y3, SOLID, CUSTOM_COLOR) + end + line(x1, y1, x2, y2, SOLID, TEXT_COLOR) + line(x1, y1, x3, y3, SOLID, TEXT_COLOR) + end + + -- Box 4 (GPS info, speed) + if data.crsf then + if data.tpwr then + text(RIGHT_POS, TOP, data.tpwr .. "mW", RIGHT + MIDSIZE + data.telemFlags) + end + text(RIGHT_POS + 1, TOP + 28, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) + else + tmp = ((data.armed or data.modeId == 6) and data.hdop < 11 - config[21].v * 2) or not data.telem + text(X3 + 48, TOP, (data.hdop == 0 and not data.gpsFix) and "-- --" or (9 - data.hdop) * 0.5 + 0.8, MIDSIZE + RIGHT + (tmp and FLASH or 0)) + text(X3 + 11, TOP + 24, "HDOP", SMLSIZE) + text(RIGHT_POS + 1, TOP, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) + end + hdopGraph(X3 + 65, TOP + (data.crsf and 51 or 23)) + tmp = RIGHT + ((not data.telem or not data.gpsFix) and FLASH or 0) + if not data.crsf then + text(RIGHT_POS, TOP + 28, floor(data.gpsAlt + 0.5) .. (data.gpsAlt_unit == 10 and "'" or units[data.gpsAlt_unit]), MIDSIZE + tmp) + end + text(RIGHT_POS, TOP + 54, config[16].v == 0 and frmt("%.6f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), SMLSIZE + tmp) + text(RIGHT_POS, TOP + 74, config[16].v == 0 and frmt("%.6f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), SMLSIZE + tmp) + tmp = data.showMax and data.speedMax or data.speed + text(RIGHT_POS + 1, TOP + 98, tmp >= 99.5 and floor(tmp + 0.5) .. units[data.speed_unit] or frmt("%.1f", tmp) .. units[data.speed_unit], MIDSIZE + RIGHT + data.telemFlags) + + -- Dividers + color(CUSTOM_COLOR, DKGREY) + line(X2 + 3, TOP, X2 + 3, BOTTOM, SOLID, CUSTOM_COLOR) + line(X3 + 3, TOP, X3 + 3, BOTTOM, SOLID, CUSTOM_COLOR) + line(X3 + 3, TOP + 95, RIGHT_POS, TOP + 95, SOLID, CUSTOM_COLOR) + if data.crsf then + line(X3 + 3, TOP + 28, RIGHT_POS, TOP + 28, SOLID, CUSTOM_COLOR) + end + line(0, TOP - 1, LCD_W - 1, TOP - 1, SOLID, CUSTOM_COLOR) + + if data.showMax then + color(CUSTOM_COLOR, YELLOW) + fill(240, TOP - 211, 80, 20, CUSTOM_COLOR) + color(CUSTOM_COLOR, BLACK) + text(319, TOP - 211, "Min/Max", CUSTOM_COLOR + RIGHT) + end + +end + +return view \ No newline at end of file diff --git a/src/iNav/other.lua b/src/iNav/other.lua index 67fbb0a3..62329561 100644 --- a/src/iNav/other.lua +++ b/src/iNav/other.lua @@ -7,6 +7,9 @@ data.fm_id = getTelemetryId("FM") > -1 and getTelemetryId("FM") or getTelemetryI -- Testing Crossfire --if data.simu then data.fm_id = 1 end +-- Testing Nirvana +--data.nv = true + if data.fm_id > -1 then crsf = loadScript(FILE_PATH .. "crsf.luac", env)(config, data, getTelemetryId, FLASH) collectgarbage() @@ -42,7 +45,7 @@ end if config[6].v == -1 then config[6].v = data.alt_unit == 10 and 400 or 120 end -config[19].x = SMLCD and ((config[14].v == 1 or data.crsf) and 1 or 2) or 2 +config[19].x = SMLCD and ((config[14].v == 1 or data.crsf) and 1 or 2) or (data.nv and 1 or 2) config[19].v = math.min(config[19].x, config[19].v) config[20].v = data.pitot and config[20].v or 0 if config[28].v == 0 then diff --git a/src/iNav/pilot.lua b/src/iNav/pilot.lua index edd348c9..556ced51 100644 --- a/src/iNav/pilot.lua +++ b/src/iNav/pilot.lua @@ -1,8 +1,8 @@ -local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH) +local function view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) local LEFT_POS = SMLCD and 0 or 36 local RIGHT_POS = SMLCD and LCD_W - 31 or LCD_W - 53 - local X_CNTR = (RIGHT_POS + LEFT_POS) / 2 - 2 + local X_CNTR = (RIGHT_POS + LEFT_POS) * 0.5 - 2 local HEADING_DEG = SMLCD and 170 or 190 local PIXEL_DEG = (RIGHT_POS - LEFT_POS) / HEADING_DEG local gpsFlags = SMLSIZE + RIGHT + ((not data.telem or not data.gpsFix) and FLASH or 0) @@ -24,9 +24,9 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local yd = math.cos(roll1) * r local x1, y1, x2, y2 = x - xd, y + yd, x + xd, y - yd if (y1 > 15 or y2 > 15) and (y1 < 56 or y2 < 56) then - lcd.drawLine(x1, y1, x2, y2, SMLCD and DOTTED or (adj % 10 == 0 and SOLID or DOTTED), SMLCD and 0 or (adj > 0 and GREY_DEFAULT or 0)) + line(x1, y1, x2, y2, SMLCD and DOTTED or (adj % 10 == 0 and SOLID or DOTTED), SMLCD and 0 or (adj > 0 and GREY_DEFAULT or 0)) if not SMLCD and adj % 10 == 0 and adj ~= 0 and y1 > 15 and y1 < 56 then - lcd.drawText(x1 - 2, y1 - 3, math.abs(adj), SMLSIZE + RIGHT) + text(x1 - 2, y1 - 3, math.abs(adj), SMLSIZE + RIGHT) end end end @@ -41,14 +41,14 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic for x = LEFT_POS + 1, RIGHT_POS - 1 do local yy = y + 0.5 if (not upsideDown and yy < 64) or (upsideDown and yy > 7) then - lcd.drawLine(x, math.min(math.max(yy, 8), 63), x, upsideDown and 8 or 63, SOLID, SMLCD and 0 or GREY_DEFAULT) + line(x, math.min(math.max(yy, 8), 63), x, upsideDown and 8 or 63, SOLID, SMLCD and 0 or GREY_DEFAULT) end y = y + a end elseif (y1 > 15 or y2 > 15) and (y1 < 56 or y2 < 56) then - lcd.drawLine(x1, y1, x2, y2, SMLCD and DOTTED or (adj % 10 == 0 and SOLID or DOTTED), SMLCD and 0 or (adj > 0 and GREY_DEFAULT or 0)) + line(x1, y1, x2, y2, SMLCD and DOTTED or (adj % 10 == 0 and SOLID or DOTTED), SMLCD and 0 or (adj > 0 and GREY_DEFAULT or 0)) if not SMLCD and adj % 10 == 0 and adj ~= 0 and y1 > 15 and y1 < 56 then - lcd.drawText(x1 - 2, y1 - 3, math.abs(adj), SMLSIZE + RIGHT) + text(x1 - 2, y1 - 3, math.abs(adj), SMLSIZE + RIGHT) end end ]] @@ -57,7 +57,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local function tics(v, p) for i = v % 10 + 8, 56, 10 do if i < 31 or i > 41 then - lcd.drawLine(p, i, p + 1, i, SOLID, 0) + line(p, i, p + 1, i, SOLID, 0) end end end @@ -65,9 +65,9 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic -- Startup message if data.startup == 2 then if not SMLCD then - lcd.drawText(50, 20, "INAV Lua Telemetry") + text(50, 20, "INAV Lua Telemetry") end - lcd.drawText(X_CNTR - 12, SMLCD and 20 or 42, "v" .. VERSION) + text(X_CNTR - 12, SMLCD and 20 or 42, "v" .. VERSION) end -- Orientation @@ -78,12 +78,12 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local r3 = math.rad(data.heading - data.headingRef - 145) local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, x, 21, 7) if data.headingHold then - lcd.drawFilledRectangle((x2 + x3) * 0.5 - 1, (y2 + y3) * 0.5 - 1, 3, 3, SOLID) + fill((x2 + x3) * 0.5 - 1, (y2 + y3) * 0.5 - 1, 3, 3, SOLID) else - lcd.drawLine(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) + line(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) end - lcd.drawLine(x1, y1, x2, y2, SOLID, FORCE) - lcd.drawLine(x1, y1, x3, y3, SOLID, FORCE) + line(x1, y1, x2, y2, SOLID, FORCE) + line(x1, y1, x3, y3, SOLID, FORCE) end -- Attitude part 1 (pitch ladder) @@ -108,7 +108,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic end if not data.showMax then tmp2 = tmp >= 0 and (tmp < 1 and 0 or math.floor(tmp + 0.5)) or (tmp > -1 and 0 or math.ceil(tmp - 0.5)) - lcd.drawText(X_CNTR - (SMLCD and 14 or 24), 33, math.abs(tmp2) .. (SMLCD and "" or "\64"), SMLSIZE + RIGHT) + text(X_CNTR - (SMLCD and 14 or 24), 33, math.abs(tmp2) .. (SMLCD and "" or "\64"), SMLSIZE + RIGHT) end end @@ -121,15 +121,15 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic end if home >= LEFT_POS - (SMLCD and 0 or 7) and home <= RIGHT_POS - 1 then tmp = (home > X_CNTR - 15 and home < X_CNTR + 10) and 49 or 50 - lcd.drawFilledRectangle(home, tmp - 1, 7, 8, ERASE) + fill(home, tmp - 1, 7, 8, ERASE) if data.distanceLast < data.distRef then - lcd.drawText(home + 1, tmp, " ", SMLSIZE + FLASH) + text(home + 1, tmp, " ", SMLSIZE + FLASH) end icons.home(home, tmp) end elseif data.showMax then - lcd.drawText(LEFT_POS + 21, 33, "\192", SMLSIZE) - lcd.drawText(RIGHT_POS - 22, 33, "\192", SMLSIZE + RIGHT) + text(LEFT_POS + 21, 33, "\192", SMLSIZE) + text(RIGHT_POS - 22, 33, "\192", SMLSIZE + RIGHT) end -- Heading part 1 @@ -137,185 +137,220 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic for i = 0, 348.75, 11.25 do tmp = math.floor(LEFT_POS + ((i - data.heading + (361 + HEADING_DEG * 0.5)) % 360) * PIXEL_DEG - 2.5) if tmp >= LEFT_POS and tmp <= RIGHT_POS then - if i % 90 == 0 then - lcd.drawText(tmp - 2, 57, i == 0 and "N" or (i == 90 and "E" or (i == 180 and "S" or "W")), SMLSIZE) - elseif i % 45 == 0 then - lcd.drawText(tmp - 4, 57, i == 45 and "NE" or (i == 135 and "SE" or (i == 225 and "SW" or "NW")), SMLSIZE) - elseif tmp < X_CNTR - 10 or tmp > X_CNTR + 9 then - lcd.drawLine(tmp, 62, tmp, 63, SOLID, FORCE) + if i % 45 == 0 then + text(tmp - (i % 90 == 0 and 2 or 4), 57, dir[i / 45], SMLSIZE) + else + line(tmp, 62, tmp, 63, SOLID, FORCE) end end end - lcd.drawFilledRectangle(RIGHT_POS, 49, 6, 14, ERASE) + fill(RIGHT_POS, 49, 6, 14, ERASE) end -- Battery and GPS overlay if SMLCD then icons.home(LEFT_POS + 4, 42) tmp = data.showMax and data.distanceMax or data.distanceLast - lcd.drawText(LEFT_POS + 12, 42, tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (string.format("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")), SMLSIZE + data.telemFlags) + text(LEFT_POS + 12, 42, tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (frmt("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")), SMLSIZE + data.telemFlags) tmp = (not data.telem or data.cell < config[3].v or (data.showCurr and config[23].v == 0 and data.fuel <= config[17].v)) and FLASH or 0 if data.showFuel then if config[23].v > 0 or (data.crsf and data.showMax) then - lcd.drawText(RIGHT_POS - 2, 9, (data.crsf and data.fuelRaw or data.fuel), SMLSIZE + RIGHT + tmp) + text(RIGHT_POS - 2, 9, (data.crsf and data.fuelRaw or data.fuel), SMLSIZE + RIGHT + tmp) else - lcd.drawText(RIGHT_POS - 7, 8, data.fuel, MIDSIZE + RIGHT + tmp) - lcd.drawText(RIGHT_POS - 2, 13, "%", SMLSIZE + RIGHT + tmp) + text(RIGHT_POS - 7, 8, data.fuel, MIDSIZE + RIGHT + tmp) + text(RIGHT_POS - 2, 13, "%", SMLSIZE + RIGHT + tmp) end end - lcd.drawText(RIGHT_POS - 7, 19, string.format(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), MIDSIZE + RIGHT + tmp) - lcd.drawText(RIGHT_POS - 2, 24, "V", SMLSIZE + RIGHT + tmp) + text(RIGHT_POS - 7, 19, frmt(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), MIDSIZE + RIGHT + tmp) + text(RIGHT_POS - 2, 24, "V", SMLSIZE + RIGHT + tmp) if data.showDir then - lcd.drawText(RIGHT_POS - 2, 42, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(RIGHT_POS - 2, 49, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(RIGHT_POS - 2, 42, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(RIGHT_POS - 2, 49, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) elseif data.showCurr then - lcd.drawText(RIGHT_POS - 2, 42, string.format("%.1fA", data.showMax and data.currentMax or data.current), SMLSIZE + RIGHT + data.telemFlags) + text(RIGHT_POS - 2, 42, frmt("%.1fA", data.showMax and data.currentMax or data.current), SMLSIZE + RIGHT + data.telemFlags) end elseif not data.armed and data.startup == 0 then - lcd.drawText(LEFT_POS + 19, 24, "Spd", SMLSIZE + RIGHT) - lcd.drawText(RIGHT_POS - 2, 24, "Alt", SMLSIZE + RIGHT) + text(LEFT_POS + 19, 24, "Spd", SMLSIZE + RIGHT) + text(RIGHT_POS - 2, 24, "Alt", SMLSIZE + RIGHT) end -- Flight modes tmp = X_CNTR - (SMLCD and 16 or 19) - lcd.drawLine(tmp, 9, tmp, 15, SOLID, ERASE) - lcd.drawText(tmp + 1, 9, modes[data.modeId].t, SMLSIZE + modes[data.modeId].f) + line(tmp, 9, tmp, 15, SOLID, ERASE) + text(tmp + 1, 9, modes[data.modeId].t, SMLSIZE + modes[data.modeId].f) if data.headFree then - lcd.drawText(tmp, 9, "HF", SMLSIZE + FLASH + RIGHT) + text(tmp, 9, "HF", SMLSIZE + FLASH + RIGHT) end if data.altHold then icons.lock(RIGHT_POS - 28, 33) end -- Attitude part 2 (artificial horizon) - lcd.drawFilledRectangle(X_CNTR - 1, 34, 3, 3, ERASE) + fill(X_CNTR - 1, 34, 3, 3, ERASE) local x = math.sin(roll1) * 200 local y = math.cos(roll1) * 200 local p = math.cos(math.rad(pitch)) * 85 local x1, y1, x2, y2 = X_CNTR - x - 2.5, 35 + y - p, X_CNTR + x - 2.5, 35 - y - p local a = (y2 - y1) / (x2 - x1 + .001) local y = y1 - ((x1 - LEFT_POS + 1) * a) + --[[ Old slower method for x = LEFT_POS + 1, RIGHT_POS - 1 do local yy = y + 0.5 if (not upsideDown and yy < 64) or (upsideDown and yy > 7) then - lcd.drawLine(x, math.min(math.max(yy, 8), 63), x, upsideDown and 8 or 63, SOLID, SMLCD and 0 or GREY_DEFAULT) - --[[ Faster? + line(x, math.min(math.max(yy, 8), 63), x, upsideDown and 8 or 63, SOLID, SMLCD and 0 or GREY_DEFAULT) + end + y = y + a + end + ]] + -- Faster method + local width = math.min(math.max(4 - math.ceil(math.abs(roll - 90) * 0.05), 1), 3) + for x = LEFT_POS + 1, RIGHT_POS - 1, width do + local yy = y + 0.5 + if (not upsideDown and yy < 64) or (upsideDown and yy > 7) then local t = upsideDown and 8 or math.min(math.max(yy, 8), 63) local h = upsideDown and math.min(math.max(yy, 8), 64) - t or 65 - t - lcd.drawFilledRectangle(x, t, 3, h, GREY_DEFAULT) - ]] + fill(x, t, width, h, GREY_DEFAULT) end - y = y + a + y = y + a * width + end + --[[ Even faster? + local width = math.min(math.max(4 - math.ceil(math.abs(roll - 90) * 0.05), 1), 3) + local lastx = -1 + for x = LEFT_POS + 1, RIGHT_POS - 1, width do + if upsideDown then + if y > 8 then + local h = math.min(math.max(y + 0.5, 8), 64) - 8 + if roll > 90 and h == 56 then + lastx = x + break + end + fill(x, 8, width, h, GREY_DEFAULT) + end + else + if y < 64 then + local t = math.min(math.max(y + 0.5, 8), 63) + if roll < 90 and t == 8 then + lastx = x + break + end + fill(x, t, width, 65 - t, GREY_DEFAULT) + end + end + y = y + a * width + end + if lastx then + fill(lastx, 8, RIGHT_POS - lastx, 57, GREY_DEFAULT) end + ]] local inside = SMLCD and 6 or 13 local outside = SMLCD and 14 or 24 - lcd.drawLine(X_CNTR - outside, 35, X_CNTR - inside, 35, SOLID, SMLCD and 0 or FORCE) - lcd.drawLine(X_CNTR + outside, 35, X_CNTR + inside, 35, SOLID, SMLCD and 0 or FORCE) - lcd.drawLine(X_CNTR - inside, 36, X_CNTR - inside, SMLCD and 37 or 38, SOLID, SMLCD and 0 or FORCE) - lcd.drawLine(X_CNTR + inside, 36, X_CNTR + inside, SMLCD and 37 or 38, SOLID, SMLCD and 0 or FORCE) - lcd.drawLine(X_CNTR - 1, 35, X_CNTR + 1, 35, SOLID, SMLCD and 0 or FORCE) + line(X_CNTR - outside, 35, X_CNTR - inside, 35, SOLID, SMLCD and 0 or FORCE) + line(X_CNTR + outside, 35, X_CNTR + inside, 35, SOLID, SMLCD and 0 or FORCE) + line(X_CNTR - inside, 36, X_CNTR - inside, SMLCD and 37 or 38, SOLID, SMLCD and 0 or FORCE) + line(X_CNTR + inside, 36, X_CNTR + inside, SMLCD and 37 or 38, SOLID, SMLCD and 0 or FORCE) + line(X_CNTR - 1, 35, X_CNTR + 1, 35, SOLID, SMLCD and 0 or FORCE) if SMLCD then lcd.drawPoint(X_CNTR, 34, 0) lcd.drawPoint(X_CNTR, 36, 0) else - lcd.drawLine(X_CNTR, 34, X_CNTR, 36, SOLID, FORCE) + line(X_CNTR, 34, X_CNTR, 36, SOLID, FORCE) end -- Heading part 2 if data.showHead then - lcd.drawLine(X_CNTR - 9, 56, X_CNTR + 10, 56, SOLID, ERASE) - lcd.drawLine(X_CNTR - 10, 56, X_CNTR - 10, 63, SOLID, ERASE) - lcd.drawText(X_CNTR - 9, 57, " ", SMLSIZE + data.telemFlags) - lcd.drawText(X_CNTR + 11, 57, math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) + line(X_CNTR - 9, 56, X_CNTR + 10, 56, SOLID, ERASE) + line(X_CNTR - 10, 56, X_CNTR - 10, 63, SOLID, ERASE) + text(X_CNTR - 9, 57, " ", SMLSIZE + data.telemFlags) + text(X_CNTR + 11, 57, math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) if not SMLCD then lcd.drawRectangle(X_CNTR - 11, 55, 23, 10, FORCE) end end -- Speed - lcd.drawLine(LEFT_POS, 8, LEFT_POS, 63, SOLID, FORCE) + line(LEFT_POS, 8, LEFT_POS, 63, SOLID, FORCE) tics(data.speed, LEFT_POS + 1) - lcd.drawLine(LEFT_POS + 1, 32, LEFT_POS + 18, 32, SOLID, ERASE) - lcd.drawText(LEFT_POS + 1, 33, " ", SMLSIZE + data.telemFlags) + line(LEFT_POS + 1, 32, LEFT_POS + 18, 32, SOLID, ERASE) + text(LEFT_POS + 1, 33, " ", SMLSIZE + data.telemFlags) tmp = data.showMax and data.speedMax or data.speed - lcd.drawText(LEFT_POS + 19, 33, data.startup == 0 and (tmp >= 99.5 and math.floor(tmp + 0.5) or string.format("%.1f", tmp)) or "Spd", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS + 19, 33, data.startup == 0 and (tmp >= 99.5 and math.floor(tmp + 0.5) or frmt("%.1f", tmp)) or "Spd", SMLSIZE + RIGHT + data.telemFlags) lcd.drawRectangle(LEFT_POS, 31, 20, 10, FORCE) -- Altitude tics(data.altitude, RIGHT_POS - 2) - lcd.drawLine(RIGHT_POS - 21, 32, RIGHT_POS, 32, SOLID, ERASE) + line(RIGHT_POS - 21, 32, RIGHT_POS, 32, SOLID, ERASE) tmp = data.showMax and data.altitudeMax or data.altitude - lcd.drawText(RIGHT_POS - 21, 33, " ", SMLSIZE + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) - lcd.drawText(RIGHT_POS, 33, data.startup == 0 and (math.floor(tmp + 0.5)) or "Alt", SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(RIGHT_POS - 21, 33, " ", SMLSIZE + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(RIGHT_POS, 33, data.startup == 0 and (math.floor(tmp + 0.5)) or "Alt", SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) lcd.drawRectangle(RIGHT_POS - 22, 31, 23, 10, FORCE) -- Variometer if config[7].v % 2 == 1 then - lcd.drawLine(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + (SMLCD and 4 or 6), 8, RIGHT_POS + (SMLCD and 4 or 6), 63, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + 1, 35, RIGHT_POS + (SMLCD and 3 or 5), 35, SMLCD and DOTTED or SOLID, SMLCD and 0 or GREY_DEFAULT) + line(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) + line(RIGHT_POS + (SMLCD and 4 or 6), 8, RIGHT_POS + (SMLCD and 4 or 6), 63, SOLID, FORCE) + line(RIGHT_POS + 1, 35, RIGHT_POS + (SMLCD and 3 or 5), 35, SMLCD and DOTTED or SOLID, SMLCD and 0 or GREY_DEFAULT) if data.armed then tmp = math.log(1 + math.min(math.abs(0.6 * (data.vspeed_unit == 6 and data.vspeed * 0.3048 or data.vspeed)), 10)) * (data.vspeed < 0 and -1 or 1) local y1 = 36 - (tmp * 11) local y2 = 36 - (tmp * 9) - lcd.drawLine(RIGHT_POS + 1, y1 - 1, RIGHT_POS + (SMLCD and 3 or 5), y2 - 1, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + 1, y1, RIGHT_POS + (SMLCD and 3 or 5), y2, SOLID, FORCE) + line(RIGHT_POS + 1, y1 - 1, RIGHT_POS + (SMLCD and 3 or 5), y2 - 1, SOLID, FORCE) + line(RIGHT_POS + 1, y1, RIGHT_POS + (SMLCD and 3 or 5), y2, SOLID, FORCE) end else - lcd.drawLine(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) + line(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) end -- Right data - GPS - lcd.drawText(LCD_W, data.crsf and 20 or 8, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) + text(LCD_W, data.crsf and 20 or 8, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) icons.gps(LCD_W - (SMLCD and 23 or 22), data.crsf and 24 or 12) if data.crsf then - lcd.drawText(LCD_W, SMLCD and 9 or 11, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) + text(LCD_W, SMLCD and 9 or 11, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) else - lcd.drawText(LCD_W + 1, SMLCD and 43 or 24, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) + text(LCD_W + 1, SMLCD and 43 or 24, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) end if SMLCD then if data.crsf == false then - lcd.drawText(LCD_W + 1, config[22].v == 0 and 32 or 22, "HDOP", RIGHT + SMLSIZE) + text(LCD_W + 1, config[22].v == 0 and 32 or 22, "HDOP", RIGHT + SMLSIZE) end hdopGraph(LCD_W - 12, config[22].v == 0 and (data.crsf and 37 or 24) or 31, MIDSIZE, SMLCD) else hdopGraph(LCD_W - 39, data.crsf and 24 or 10, MIDSIZE, SMLCD) if data.crsf == false then - lcd.drawText(LCD_W - (config[22].v == 0 and 24 or 25), config[22].v == 0 and 18 or 20, "HDOP", RIGHT + SMLSIZE) + text(LCD_W - (config[22].v == 0 and 24 or 25), config[22].v == 0 and 18 or 20, "HDOP", RIGHT + SMLSIZE) end - lcd.drawText(LCD_W + 1, 33, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(LCD_W + 1, 42, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) - lcd.drawText(RIGHT_POS + 8, 57, data.crsf and "LQ" or "RSSI", SMLSIZE) + text(LCD_W + 1, 33, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(LCD_W + 1, 42, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(RIGHT_POS + 8, 57, data.crsf and "LQ" or "RSSI", SMLSIZE) end - lcd.drawLine(RIGHT_POS + (config[7].v % 2 == 1 and (SMLCD and 5 or 7) or 0), 50, LCD_W, 50, SOLID, FORCE) + line(RIGHT_POS + (config[7].v % 2 == 1 and (SMLCD and 5 or 7) or 0), 50, LCD_W, 50, SOLID, FORCE) local rssiFlags = RIGHT + ((not data.telem or data.rssi < data.rssiLow) and FLASH or 0) - lcd.drawText(LCD_W - (data.crsf and 6 or 10), 52, math.min(data.showMax and data.rssiMin or data.rssiLast, data.crsf and 100 or 99), MIDSIZE + rssiFlags) - lcd.drawText(LCD_W, 57, data.crsf and "%" or "dB", SMLSIZE + rssiFlags) + text(LCD_W - (data.crsf and 6 or 10), 52, math.min(data.showMax and data.rssiMin or data.rssiLast, data.crsf and 100 or 99), MIDSIZE + rssiFlags) + text(LCD_W, 57, data.crsf and "%" or "dB", SMLSIZE + rssiFlags) -- Left data - Battery if not SMLCD then - lcd.drawFilledRectangle(LEFT_POS - 7, 49, 7, 14, ERASE) + fill(LEFT_POS - 7, 49, 7, 14, ERASE) tmp = (not data.telem or data.cell < config[3].v or (data.showFuel and config[23].v == 0 and data.fuel <= config[17].v)) and FLASH or 0 if data.showFuel then if config[23].v > 0 or (data.crsf and data.showMax) then - lcd.drawText(LEFT_POS, data.showCurr and 8 or 10, (data.crsf and data.fuelRaw or data.fuel), MIDSIZE + RIGHT + tmp) - lcd.drawText(LEFT_POS, data.showCurr and 20 or 23, data.fUnit[data.crsf and 1 or config[23].v], SMLSIZE + RIGHT + tmp) + text(LEFT_POS, data.showCurr and 8 or 10, (data.crsf and data.fuelRaw or data.fuel), MIDSIZE + RIGHT + tmp) + text(LEFT_POS, data.showCurr and 20 or 23, data.fUnit[data.crsf and 1 or config[23].v], SMLSIZE + RIGHT + tmp) else - lcd.drawText(LEFT_POS - 5, data.showCurr and 8 or 12, data.fuel, DBLSIZE + RIGHT + tmp) - lcd.drawText(LEFT_POS, data.showCurr and 17 or 21, "%", SMLSIZE + RIGHT + tmp) + text(LEFT_POS - 5, data.showCurr and 8 or 12, data.fuel, DBLSIZE + RIGHT + tmp) + text(LEFT_POS, data.showCurr and 17 or 21, "%", SMLSIZE + RIGHT + tmp) end end - lcd.drawText(LEFT_POS - 5, data.showCurr and 25 or 32, string.format(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), DBLSIZE + RIGHT + tmp) - lcd.drawText(LEFT_POS, data.showCurr and 34 or 41, "V", SMLSIZE + RIGHT + tmp) + text(LEFT_POS - 5, data.showCurr and 25 or 32, frmt(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), DBLSIZE + RIGHT + tmp) + text(LEFT_POS, data.showCurr and 34 or 41, "V", SMLSIZE + RIGHT + tmp) if data.showCurr then tmp = data.showMax and data.currentMax or data.current - lcd.drawText(LEFT_POS - 5, 42, tmp >= 99.5 and math.floor(tmp + 0.5) or string.format("%.1f", tmp), MIDSIZE + RIGHT + data.telemFlags) - lcd.drawText(LEFT_POS, 47, "A", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS - 5, 42, tmp >= 99.5 and math.floor(tmp + 0.5) or frmt("%.1f", tmp), MIDSIZE + RIGHT + data.telemFlags) + text(LEFT_POS, 47, "A", SMLSIZE + RIGHT + data.telemFlags) end - lcd.drawLine(0, data.showCurr and 54 or 53, LEFT_POS, data.showCurr and 54 or 53, SOLID, FORCE) + line(0, data.showCurr and 54 or 53, LEFT_POS, data.showCurr and 54 or 53, SOLID, FORCE) icons.home(0, 57) tmp = data.showMax and data.distanceMax or data.distanceLast - lcd.drawText(LEFT_POS, 57, tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (string.format("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")), SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS, 57, tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (frmt("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi")), SMLSIZE + RIGHT + data.telemFlags) end end diff --git a/src/iNav/radar.lua b/src/iNav/radar.lua index f3a2cfaa..ae487a80 100644 --- a/src/iNav/radar.lua +++ b/src/iNav/radar.lua @@ -1,25 +1,25 @@ -local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH) +local function view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) local LEFT_DIV = 36 local LEFT_POS = SMLCD and LEFT_DIV or 73 local RIGHT_POS = SMLCD and LCD_W - 31 or LCD_W - 53 - local X_CNTR = (RIGHT_POS + LEFT_POS) / 2 - 1 + local X_CNTR = (RIGHT_POS + LEFT_POS) * 0.5 - 1 local gpsFlags = SMLSIZE + RIGHT + ((not data.telem or not data.gpsFix) and FLASH or 0) local tmp, pitch -- Startup message if data.startup == 2 then if not SMLCD then - lcd.drawText(LEFT_POS + 8, 28, "Lua Telemetry") + text(LEFT_POS + 8, 28, "Lua Telemetry") end - lcd.drawText(X_CNTR - 10, SMLCD and 29 or 40, "v" .. VERSION) + text(X_CNTR - 10, SMLCD and 29 or 40, "v" .. VERSION) end -- Flight modes tmp = X_CNTR - (SMLCD and 16 or 19) - lcd.drawText(tmp + 1, 9, modes[data.modeId].t, SMLSIZE + modes[data.modeId].f) + text(tmp + 1, 9, modes[data.modeId].t, SMLSIZE + modes[data.modeId].f) if data.headFree then - lcd.drawText(tmp, 9, "HF", SMLSIZE + FLASH + RIGHT) + text(tmp, 9, "HF", SMLSIZE + FLASH + RIGHT) end -- Pitch calculation @@ -34,41 +34,41 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if SMLCD then if data.showDir and (not data.armed or not data.telem) then -- GPS coords - lcd.drawText(RIGHT_POS, 50, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(RIGHT_POS, 57, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(RIGHT_POS, 50, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(RIGHT_POS, 57, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) else -- Distance tmp = data.showMax and data.distanceMax or data.distanceLast - lcd.drawText(LEFT_POS + 25, 57, data.startup > 0 and "Dist " or (tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (string.format("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi"))), SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS + 25, 57, data.startup > 0 and "Dist " or (tmp < 1000 and math.floor(tmp + 0.5) .. units[data.dist_unit] or (frmt("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)) .. (data.dist_unit == 9 and "km" or "mi"))), SMLSIZE + RIGHT + data.telemFlags) -- Altitude tmp = data.showMax and data.altitudeMax or data.altitude - lcd.drawText(RIGHT_POS, 57, data.startup > 0 and "Alt" or (math.floor(tmp + 0.5) .. units[data.alt_unit]), SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(RIGHT_POS, 57, data.startup > 0 and "Alt" or (math.floor(tmp + 0.5) .. units[data.alt_unit]), SMLSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) if data.altHold then icons.lock(RIGHT_POS - 6, 50) end end -- Pitch if data.startup == 0 then - lcd.drawText(LEFT_POS + 15, 17, pitch .. (math.abs(pitch) < 10 and "\64" or ""), SMLSIZE + RIGHT + data.telemFlags) - lcd.drawLine(LEFT_POS + 1, 17, LEFT_POS + 1, 24, SOLID, ERASE) + text(LEFT_POS + 15, 17, pitch .. (math.abs(pitch) < 10 and "\64" or ""), SMLSIZE + RIGHT + data.telemFlags) + line(LEFT_POS + 1, 17, LEFT_POS + 1, 24, SOLID, ERASE) else - lcd.drawText(LEFT_POS + 2, 17, "Ptch", SMLSIZE) + text(LEFT_POS + 2, 17, "Ptch", SMLSIZE) end elseif data.showDir or data.headingRef == -1 then -- Heading - lcd.drawText(X_CNTR + 14 - (data.heading < 100 and 3 or 0) - (data.heading < 10 and 3 or 0), 57, math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) + text(X_CNTR + 14 - (data.heading < 100 and 3 or 0) - (data.heading < 10 and 3 or 0), 57, math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE + RIGHT + data.telemFlags) end -- Min/Max if not data.showDir and data.showMax then - lcd.drawText(RIGHT_POS, 9, "\192", SMLSIZE + RIGHT) + text(RIGHT_POS, 9, "\192", SMLSIZE + RIGHT) end -- Radar if data.startup == 0 then tmp = data.headingRef if data.showDir or data.headingRef == -1 then - lcd.drawText(LEFT_POS + 2, 33, "W", SMLSIZE) - lcd.drawText(RIGHT_POS, 33, "E", SMLSIZE + RIGHT) + text(LEFT_POS + 2, 33, dir[6], SMLSIZE) + text(RIGHT_POS, 33, dir[2], SMLSIZE + RIGHT) tmp = 0 end local cx, cy, d @@ -88,7 +88,7 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if d >= 9 then icons.home(hx - 3, hy - 3) elseif d > 1 then - lcd.drawFilledRectangle(hx - 1, hy - 1, 3, 3, SOLID) + fill(hx - 1, hy - 1, 3, 3, SOLID) end -- Shift craft location cx = d == 1 and X_CNTR + 2 or cx + hx @@ -105,106 +105,106 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, cx, cy, d == 1 and 8 or 5) if data.headingHold then if d == 1 then - lcd.drawFilledRectangle((x2 + x3) * 0.5 - 1.5, (y2 + y3) * 0.5 - 1.5, 4, 4, SOLID) + fill((x2 + x3) * 0.5 - 1.5, (y2 + y3) * 0.5 - 1.5, 4, 4, SOLID) else - lcd.drawFilledRectangle((x2 + x3) * 0.5 - 1, (y2 + y3) * 0.5 - 1, 3, 3, SOLID) + fill((x2 + x3) * 0.5 - 1, (y2 + y3) * 0.5 - 1, 3, 3, SOLID) end else - lcd.drawLine(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) + line(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) end - lcd.drawLine(x1, y1, x2, y2, SOLID, FORCE) - lcd.drawLine(x1, y1, x3, y3, SOLID, FORCE) + line(x1, y1, x2, y2, SOLID, FORCE) + line(x1, y1, x3, y3, SOLID, FORCE) end -- Variometer if config[7].v % 2 == 1 then - lcd.drawLine(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + (SMLCD and 4 or 6), 8, RIGHT_POS + (SMLCD and 4 or 6), 63, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + 1, 35, RIGHT_POS + (SMLCD and 3 or 5), 35, SMLCD and DOTTED or SOLID, SMLCD and 0 or GREY_DEFAULT) + line(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) + line(RIGHT_POS + (SMLCD and 4 or 6), 8, RIGHT_POS + (SMLCD and 4 or 6), 63, SOLID, FORCE) + line(RIGHT_POS + 1, 35, RIGHT_POS + (SMLCD and 3 or 5), 35, SMLCD and DOTTED or SOLID, SMLCD and 0 or GREY_DEFAULT) if data.armed then tmp = math.log(1 + math.min(math.abs(0.6 * (data.vspeed_unit == 6 and data.vspeed * 0.3048 or data.vspeed)), 10)) * (data.vspeed < 0 and -1 or 1) local y1 = 36 - (tmp * 11) local y2 = 36 - (tmp * 9) - lcd.drawLine(RIGHT_POS + 1, y1 - 1, RIGHT_POS + (SMLCD and 3 or 5), y2 - 1, SOLID, FORCE) - lcd.drawLine(RIGHT_POS + 1, y1, RIGHT_POS + (SMLCD and 3 or 5), y2, SOLID, FORCE) + line(RIGHT_POS + 1, y1 - 1, RIGHT_POS + (SMLCD and 3 or 5), y2 - 1, SOLID, FORCE) + line(RIGHT_POS + 1, y1, RIGHT_POS + (SMLCD and 3 or 5), y2, SOLID, FORCE) end else - lcd.drawLine(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) + line(RIGHT_POS, 8, RIGHT_POS, 63, SOLID, FORCE) end -- Right data - GPS - lcd.drawText(LCD_W, data.crsf and 20 or 8, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) + text(LCD_W, data.crsf and 20 or 8, data.satellites % 100, MIDSIZE + RIGHT + data.telemFlags) icons.gps(LCD_W - (SMLCD and 23 or 22), data.crsf and 24 or 12) if data.crsf then - lcd.drawText(LCD_W, SMLCD and 9 or 11, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) + text(LCD_W, SMLCD and 9 or 11, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) else - lcd.drawText(LCD_W + 1, SMLCD and 43 or 24, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) + text(LCD_W + 1, SMLCD and 43 or 24, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) end if SMLCD then if data.crsf == false then - lcd.drawText(LCD_W + 1, config[22].v == 0 and 32 or 22, "HDOP", RIGHT + SMLSIZE) + text(LCD_W + 1, config[22].v == 0 and 32 or 22, "HDOP", RIGHT + SMLSIZE) end hdopGraph(LCD_W - 12, config[22].v == 0 and (data.crsf and 37 or 24) or 31, MIDSIZE, SMLCD) else hdopGraph(LCD_W - 39, data.crsf and 24 or 10, MIDSIZE, SMLCD) if data.crsf == false then - lcd.drawText(LCD_W - (config[22].v == 0 and 24 or 25), config[22].v == 0 and 18 or 20, "HDOP", RIGHT + SMLSIZE) + text(LCD_W - (config[22].v == 0 and 24 or 25), config[22].v == 0 and 18 or 20, "HDOP", RIGHT + SMLSIZE) end - lcd.drawText(LCD_W + 1, 33, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(LCD_W + 1, 42, config[16].v == 0 and string.format("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) - lcd.drawText(RIGHT_POS + 8, 57, data.crsf and "LQ" or "RSSI", SMLSIZE) + text(LCD_W + 1, 33, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(LCD_W + 1, 42, config[16].v == 0 and frmt("%.5f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(RIGHT_POS + 8, 57, data.crsf and "LQ" or "RSSI", SMLSIZE) end - lcd.drawLine(RIGHT_POS + (config[7].v % 2 == 1 and (SMLCD and 5 or 7) or 0), 50, LCD_W, 50, SOLID, FORCE) + line(RIGHT_POS + (config[7].v % 2 == 1 and (SMLCD and 5 or 7) or 0), 50, LCD_W, 50, SOLID, FORCE) local rssiFlags = RIGHT + ((not data.telem or data.rssi < data.rssiLow) and FLASH or 0) - lcd.drawText(LCD_W - (data.crsf and 6 or 10), 52, math.min(data.showMax and data.rssiMin or data.rssiLast, data.crsf and 100 or 99), MIDSIZE + rssiFlags) - lcd.drawText(LCD_W, 57, data.crsf and "%" or "dB", SMLSIZE + rssiFlags) + text(LCD_W - (data.crsf and 6 or 10), 52, math.min(data.showMax and data.rssiMin or data.rssiLast, data.crsf and 100 or 99), MIDSIZE + rssiFlags) + text(LCD_W, 57, data.crsf and "%" or "dB", SMLSIZE + rssiFlags) -- Left data - Battery - lcd.drawLine(LEFT_DIV, 8, LEFT_DIV, 63, SOLID, FORCE) + line(LEFT_DIV, 8, LEFT_DIV, 63, SOLID, FORCE) tmp = (not data.telem or data.cell < config[3].v or (data.showFuel and config[23].v == 0 and data.fuel <= config[17].v)) and FLASH or 0 if data.showFuel then if config[23].v > 0 or (data.crsf and data.showMax) then - lcd.drawText(LEFT_DIV, data.showCurr and 8 or 10, (data.crsf and data.fuelRaw or data.fuel), MIDSIZE + RIGHT + tmp) - lcd.drawText(LEFT_DIV, data.showCurr and 20 or 23, data.fUnit[data.crsf and 1 or config[23].v], SMLSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 8 or 10, (data.crsf and data.fuelRaw or data.fuel), MIDSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 20 or 23, data.fUnit[data.crsf and 1 or config[23].v], SMLSIZE + RIGHT + tmp) else - lcd.drawText(LEFT_DIV - 5, data.showCurr and 8 or 12, data.fuel, DBLSIZE + RIGHT + tmp) - lcd.drawText(LEFT_DIV, data.showCurr and 17 or 21, "%", SMLSIZE + RIGHT + tmp) + text(LEFT_DIV - 5, data.showCurr and 8 or 12, data.fuel, DBLSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 17 or 21, "%", SMLSIZE + RIGHT + tmp) end end - lcd.drawText(LEFT_DIV - 5, data.showCurr and 25 or 32, string.format(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), DBLSIZE + RIGHT + tmp) - lcd.drawText(LEFT_DIV, data.showCurr and 34 or 41, "V", SMLSIZE + RIGHT + tmp) + text(LEFT_DIV - 5, data.showCurr and 25 or 32, frmt(config[1].v == 0 and "%.2f" or "%.1f", config[1].v == 0 and (data.showMax and data.cellMin or data.cell) or (data.showMax and data.battMin or data.batt)), DBLSIZE + RIGHT + tmp) + text(LEFT_DIV, data.showCurr and 34 or 41, "V", SMLSIZE + RIGHT + tmp) if data.showCurr then tmp = data.showMax and data.currentMax or data.current - lcd.drawText(LEFT_DIV - 5, 42, tmp >= 99.5 and math.floor(tmp + 0.5) or string.format("%.1f", tmp), MIDSIZE + RIGHT + data.telemFlags) - lcd.drawText(LEFT_DIV, 47, "A", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV - 5, 42, tmp >= 99.5 and math.floor(tmp + 0.5) or frmt("%.1f", tmp), MIDSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV, 47, "A", SMLSIZE + RIGHT + data.telemFlags) end - lcd.drawLine(0, data.showCurr and 55 or 53, LEFT_DIV, data.showCurr and 55 or 53, SOLID, FORCE) + line(0, data.showCurr and 55 or 53, LEFT_DIV, data.showCurr and 55 or 53, SOLID, FORCE) tmp = data.showMax and data.speedMax or data.speed - lcd.drawText(LEFT_DIV, data.showCurr and 57 or 56, tmp >= 99.5 and math.floor(tmp + 0.5) .. units[data.speed_unit] or string.format("%.1f", tmp) .. units[data.speed_unit], SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV, data.showCurr and 57 or 56, tmp >= 99.5 and math.floor(tmp + 0.5) .. units[data.speed_unit] or frmt("%.1f", tmp) .. units[data.speed_unit], SMLSIZE + RIGHT + data.telemFlags) -- Left data - Wide screen if not SMLCD then - lcd.drawLine(LEFT_POS, 8, LEFT_POS, 63, SOLID, FORCE) + line(LEFT_POS, 8, LEFT_POS, 63, SOLID, FORCE) -- Altitude tmp = data.showMax and data.altitudeMax or data.altitude local tmp2 = data.alt_unit == 9 and 6 or 2 - lcd.drawText(LEFT_DIV + 2, 9, "Alt", SMLSIZE) - lcd.drawText(LEFT_POS - tmp2, data.alt_unit == 9 and 21 or 17, units[data.alt_unit], SMLSIZE + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) - lcd.drawText(LEFT_POS - tmp2, 16, math.floor(tmp + 0.5), MIDSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(LEFT_DIV + 2, 9, "Alt", SMLSIZE) + text(LEFT_POS - tmp2, data.alt_unit == 9 and 21 or 17, units[data.alt_unit], SMLSIZE + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) + text(LEFT_POS - tmp2, 16, math.floor(tmp + 0.5), MIDSIZE + RIGHT + ((not data.telem or tmp + 0.5 >= config[6].v) and FLASH or 0)) if data.altHold then icons.lock(LEFT_POS - 6, 9) end -- Distance tmp = data.showMax and data.distanceMax or data.distanceLast tmp2 = data.dist_unit == 9 and (tmp < 1000 and 6 or 11) or (tmp < 1000 and 2 or 10) - lcd.drawText(LEFT_DIV + 2, 30, "Dist", SMLSIZE) - lcd.drawText(LEFT_POS - tmp2, (data.dist_unit == 9 or tmp >= 1000) and 42 or 38, tmp < 1000 and units[data.dist_unit] or (data.dist_unit == 9 and "km" or "mi"), SMLSIZE + data.telemFlags) - lcd.drawText(LEFT_POS - tmp2, 37, tmp < 1000 and math.floor(tmp + 0.5) or string.format("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)), MIDSIZE + RIGHT + data.telemFlags) + text(LEFT_DIV + 2, 30, "Dist", SMLSIZE) + text(LEFT_POS - tmp2, (data.dist_unit == 9 or tmp >= 1000) and 42 or 38, tmp < 1000 and units[data.dist_unit] or (data.dist_unit == 9 and "km" or "mi"), SMLSIZE + data.telemFlags) + text(LEFT_POS - tmp2, 37, tmp < 1000 and math.floor(tmp + 0.5) or frmt("%.1f", tmp / (data.dist_unit == 9 and 1000 or 5280)), MIDSIZE + RIGHT + data.telemFlags) --Pitch - lcd.drawLine(LEFT_DIV, 50, LEFT_POS, 50, SOLID, FORCE) - lcd.drawText(LEFT_DIV + 5, 54, pitch > 0 and "\194" or (pitch == 0 and "->" or "\195"), SMLSIZE) - lcd.drawText(LEFT_POS, 53, "\64", SMLSIZE + RIGHT + data.telemFlags) - lcd.drawText(LEFT_POS - 4, 52, pitch, MIDSIZE + RIGHT + data.telemFlags) + line(LEFT_DIV, 50, LEFT_POS, 50, SOLID, FORCE) + text(LEFT_DIV + 5, 54, pitch > 0 and "\194" or (pitch == 0 and "->" or "\195"), SMLSIZE) + text(LEFT_POS, 53, "\64", SMLSIZE + RIGHT + data.telemFlags) + text(LEFT_POS - 4, 52, pitch, MIDSIZE + RIGHT + data.telemFlags) end end diff --git a/src/iNav/reset.lua b/src/iNav/reset.lua index 2e3f353b..57bcfcc0 100644 --- a/src/iNav/reset.lua +++ b/src/iNav/reset.lua @@ -37,9 +37,9 @@ for i = 1, 60 do data.alt[i] = 0 end ---[[ FPS stuff +--[[ Show FPS ]] data.fpsStart = getTime() data.frames = 0 -]] + return 0 \ No newline at end of file diff --git a/src/iNav/save.lua b/src/iNav/save.lua index 2dd97df0..2a7bfa52 100644 --- a/src/iNav/save.lua +++ b/src/iNav/save.lua @@ -1,4 +1,4 @@ -local config, data, FILE_PATH = ... +local config, data, frmt, FILE_PATH = ... local fh = io.open(FILE_PATH .. "cfg/" .. model.getInfo().name .. ".dat", "w") --[[ @@ -9,12 +9,11 @@ else ]] if fh ~= nil then local floor = math.floor - local format = string.format - for line = 1, #config do - if config[line].d == nil then - io.write(fh, string.format("%0" .. config[line].c .. "d", config[line].v)) + for i = 1, #config do + if config[i].d == nil then + io.write(fh, frmt("%0" .. config[i].c .. "d", config[i].v)) else - io.write(fh, floor(config[line].v * 10)) + io.write(fh, floor(config[i].v * 10)) end end io.close(fh) diff --git a/src/iNav/view.lua b/src/iNav/view.lua index 22ab2774..867d669f 100644 --- a/src/iNav/view.lua +++ b/src/iNav/view.lua @@ -1,4 +1,4 @@ -local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH) +local function view(data, config, modes, dir, units, labels, gpsDegMin, hdopGraph, icons, calcBearing, calcDir, VERSION, SMLCD, FLASH, FILE_PATH, text, line, rect, fill, frmt) local RIGHT_POS = SMLCD and 129 or 195 local GAUGE_WIDTH = SMLCD and 82 or 149 @@ -11,63 +11,63 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local r2 = math.rad(h + w) local r3 = math.rad(h - w) local x1, y1, x2, y2, x3, y3 = calcDir(r1, r2, r3, x, y, r) - lcd.drawLine(x1, y1, x2, y2, SOLID, FORCE) - lcd.drawLine(x1, y1, x3, y3, SOLID, FORCE) + line(x1, y1, x2, y2, SOLID, FORCE) + line(x1, y1, x3, y3, SOLID, FORCE) if data.headingHold then - lcd.drawFilledRectangle((x2 + x3) * 0.5 - 1.5, (y2 + y3) * 0.5 - 1.5, 4, 4, SOLID) + fill((x2 + x3) * 0.5 - 1.5, (y2 + y3) * 0.5 - 1.5, 4, 4, SOLID) else - lcd.drawLine(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) + line(x2, y2, x3, y3, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) end end - local function drawData(txt, y, dir, vc, vm, max, ext, frac, flags) - if data.showMax and dir > 0 then + local function drawData(txt, y, d, vc, vm, max, ext, frac, flags) + if data.showMax and d > 0 then vc = vm - lcd.drawText(0, y, string.sub(txt, 1, 3), SMLSIZE) - lcd.drawText(15, y, dir == 1 and "\192" or "\193", SMLSIZE) + text(0, y, string.sub(txt, 1, 3), SMLSIZE) + text(15, y, d == 1 and "\192" or "\193", SMLSIZE) else - lcd.drawText(0, y, txt, SMLSIZE) + text(0, y, txt, SMLSIZE) end local tmp = (frac ~= 0 or vc < max) and ext or "" if frac ~= 0 and vc + 0.5 < max then - lcd.drawText(21, y, string.format(frac, vc) .. tmp, SMLSIZE + flags) + text(21, y, frmt(frac, vc) .. tmp, SMLSIZE + flags) else - lcd.drawText(21, y, math.floor(vc + 0.5) .. tmp, SMLSIZE + flags) + text(21, y, math.floor(vc + 0.5) .. tmp, SMLSIZE + flags) end end -- Startup message if data.startup == 2 then if not SMLCD then - lcd.drawText(53, 9, "INAV Lua Telemetry") + text(53, 9, "INAV Lua Telemetry") end - lcd.drawText(SMLCD and 51 or 91, 17, "v" .. VERSION) + text(SMLCD and 51 or 91, 17, "v" .. VERSION) end -- GPS local gpsFlags = SMLSIZE + RIGHT + ((not data.telem or not data.gpsFix) and FLASH or 0) tmp = RIGHT_POS - (gpsFlags == SMLSIZE + RIGHT and 0 or 1) - lcd.drawText(tmp, 25, config[16].v == 0 and string.format(SMLCD and "%.5f" or "%.6f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) - lcd.drawText(tmp, 33, config[16].v == 0 and string.format(SMLCD and "%.5f" or "%.6f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) + text(tmp, 25, config[16].v == 0 and frmt(SMLCD and "%.5f" or "%.6f", data.gpsLatLon.lat) or gpsDegMin(data.gpsLatLon.lat, true), gpsFlags) + text(tmp, 33, config[16].v == 0 and frmt(SMLCD and "%.5f" or "%.6f", data.gpsLatLon.lon) or gpsDegMin(data.gpsLatLon.lon, false), gpsFlags) if data.crsf then - lcd.drawText(RIGHT_POS - (data.telem and 0 or 1), 9, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) + text(RIGHT_POS - (data.telem and 0 or 1), 9, data.tpwr < 1000 and data.tpwr .. "mW" or data.tpwr * 0.001 .. "W", SMLSIZE + RIGHT + data.telemFlags) else - lcd.drawText(tmp, 17, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) + text(tmp, 17, math.floor(data.gpsAlt + 0.5) .. units[data.gpsAlt_unit], gpsFlags) end hdopGraph(RIGHT_POS - 30, data.crsf and 17 or 9, SMLSIZE, SMLCD) icons.gps(RIGHT_POS - 17, data.crsf and 17 or 9) - lcd.drawText(RIGHT_POS - (data.telem and 0 or 1), data.crsf and 17 or 9, data.satellites % 100, SMLSIZE + RIGHT + data.telemFlags) + text(RIGHT_POS - (data.telem and 0 or 1), data.crsf and 17 or 9, data.satellites % 100, SMLSIZE + RIGHT + data.telemFlags) -- Directionals if data.showHead and data.startup == 0 then if data.telem then local indicatorDisplayed = false if data.showDir or data.headingRef == -1 or not SMLCD then - lcd.drawText(X_CNTR_1 - 2, 9, "N " .. math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE) - lcd.drawText(X_CNTR_1 + 10, 21, "E", SMLSIZE) - lcd.drawText(X_CNTR_1 - 14, 21, "W", SMLSIZE) + text(X_CNTR_1 - 2, 9, dir[0] .. " " .. math.floor(data.heading + 0.5) % 360 .. "\64", SMLSIZE) + text(X_CNTR_1 + 10, 21, dir[2], SMLSIZE) + text(X_CNTR_1 - 14, 21, dir[6], SMLSIZE) if not SMLCD then - lcd.drawText(X_CNTR_1 - 2, 32, "S", SMLSIZE) + text(X_CNTR_1 - 2, 32, dir[4], SMLSIZE) end drawDirection(data.heading, 140, 7, X_CNTR_1, 23) indicatorDisplayed = true @@ -84,17 +84,17 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic local rad1 = math.rad(bearing) local x1 = math.floor(math.sin(rad1) * 10 + 0.5) + X_CNTR_2 local y1 = 19 - math.floor(math.cos(rad1) * 10 + 0.5) - lcd.drawLine(X_CNTR_2, 19, x1, y1, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) - lcd.drawFilledRectangle(x1 - 1, y1 - 1, 3, 3, ERASE) - lcd.drawFilledRectangle(x1 - 1, y1 - 1, 3, 3, SOLID) + line(X_CNTR_2, 19, x1, y1, SMLCD and DOTTED or SOLID, FORCE + (SMLCD and 0 or GREY_DEFAULT)) + fill(x1 - 1, y1 - 1, 3, 3, ERASE) + fill(x1 - 1, y1 - 1, 3, 3, SOLID) end end end -- Flight mode - lcd.drawText((SMLCD and 46 or 83) + (modes[data.modeId].f == FLASH and 1 or 0), 33, modes[data.modeId].t, (SMLCD and SMLSIZE or 0) + modes[data.modeId].f) + text((SMLCD and 46 or 83) + (modes[data.modeId].f == FLASH and 1 or 0), 33, modes[data.modeId].t, (SMLCD and SMLSIZE or 0) + modes[data.modeId].f) if data.headFree then - lcd.drawText(RIGHT_POS - 41, 9, "HF", FLASH + SMLSIZE) + text(RIGHT_POS - 41, 9, "HF", FLASH + SMLSIZE) end -- Data & gauges @@ -111,43 +111,43 @@ local function view(data, config, modes, units, labels, gpsDegMin, hdopGraph, ic if config[23].v == 0 then lcd.drawGauge(46, 41, GAUGE_WIDTH, 7, math.min(data.fuel, 99), 100) if data.fuel == 0 then - lcd.drawLine(47, 42, 47, 46, SOLID, ERASE) + line(47, 42, 47, 46, SOLID, ERASE) end end end tmp = 100 / (4.2 - config[3].v + 0.1) lcd.drawGauge(46, data.showCurr and 49 or 41, GAUGE_WIDTH, data.showCurr and 7 or 15, math.min(math.max(data.cell - config[3].v + 0.1, 0) * tmp, 98), 100) tmp = (GAUGE_WIDTH - 2) * (math.min(math.max(data.cellMin - config[3].v + 0.1, 0) * tmp, 99) * 0.01) + 47 - lcd.drawLine(tmp, data.showCurr and 50 or 42, tmp, 54, SOLID, ERASE) + line(tmp, data.showCurr and 50 or 42, tmp, 54, SOLID, ERASE) lcd.drawGauge(46, 57, GAUGE_WIDTH, 7, math.max(math.min((data.rssiLast - data.rssiCrit) / (100 - data.rssiCrit) * 100, 98), 0), 100) tmp = (GAUGE_WIDTH - 2) * (math.max(math.min((data.rssiMin - data.rssiCrit) / (100 - data.rssiCrit) * 100, 99), 0) * 0.01) + 47 - lcd.drawLine(tmp, 58, tmp, 62, SOLID, ERASE) + line(tmp, 58, tmp, 62, SOLID, ERASE) if not SMLCD then local w = config[7].v % 2 == 1 and 7 or 15 local l = config[7].v % 2 == 1 and 205 or 197 - lcd.drawRectangle(l, 9, w, 48, SOLID) + rect(l, 9, w, 48, SOLID) tmp = math.max(math.min(math.ceil(data.altitude / config[6].v * 46), 46), 0) - lcd.drawFilledRectangle(l + 1, 56 - tmp, w - 2, tmp, INVERS) + fill(l + 1, 56 - tmp, w - 2, tmp, INVERS) tmp = 56 - math.max(math.min(math.ceil(data.altitudeMax / config[6].v * 46), 46), 0) - lcd.drawLine(l + 1, tmp, l + w - 2, tmp, SOLID, GREY_DEFAULT) - lcd.drawText(l + 1, 58, config[7].v % 2 == 1 and "A" or "Alt", SMLSIZE) + line(l + 1, tmp, l + w - 2, tmp, SOLID, GREY_DEFAULT) + text(l + 1, 58, config[7].v % 2 == 1 and "A" or "Alt", SMLSIZE) end -- Variometer if config[7].v % 2 == 1 then local varioSpeed = math.log(1 + math.min(math.abs(0.6 * (data.vspeed_unit == 6 and data.vspeed * 0.3048 or data.vspeed)), 10)) * 0.416667 * (data.vspeed < 0 and -1 or 1) if SMLCD and data.armed and not data.showDir and data.startup == 0 then - lcd.drawLine(X_CNTR_2 + 17, 21, X_CNTR_2 + 19, 21, SOLID, FORCE) - lcd.drawLine(X_CNTR_2 + 18, 21, X_CNTR_2 + 18, 21 - (varioSpeed * 12 - 0.5), SOLID, FORCE) + line(X_CNTR_2 + 17, 21, X_CNTR_2 + 19, 21, SOLID, FORCE) + line(X_CNTR_2 + 18, 21, X_CNTR_2 + 18, 21 - (varioSpeed * 12 - 0.5), SOLID, FORCE) elseif not SMLCD then - lcd.drawRectangle(197, 9, 7, 48, SOLID) - lcd.drawText(198, 58, "V", SMLSIZE) + rect(197, 9, 7, 48, SOLID) + text(198, 58, "V", SMLSIZE) if data.armed then tmp = 33 - math.floor(varioSpeed * 23 - 0.5) if tmp > 33 then - lcd.drawFilledRectangle(198, 33, 5, tmp - 33, INVERS) + fill(198, 33, 5, tmp - 33, INVERS) else - lcd.drawFilledRectangle(198, tmp - 1, 5, 33 - tmp + 2, INVERS) + fill(198, tmp - 1, 5, 33 - tmp + 2, INVERS) end end end