mirror of
https://github.com/opentx/opentx.git
synced 2025-07-24 16:55:20 +03:00
242 lines
9.4 KiB
C++
242 lines
9.4 KiB
C++
/*
|
|
* Copyright (C) OpenTX
|
|
*
|
|
* Based on code named
|
|
* th9x - http://code.google.com/p/th9x
|
|
* er9x - http://code.google.com/p/er9x
|
|
* gruvin9x - http://code.google.com/p/gruvin9x
|
|
*
|
|
* License GPLv2: http://www.gnu.org/licenses/gpl-2.0.html
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
#include "opentx.h"
|
|
|
|
enum LimitsItems {
|
|
ITEM_LIMITS_CH_NAME,
|
|
ITEM_LIMITS_OFFSET,
|
|
ITEM_LIMITS_MIN,
|
|
ITEM_LIMITS_MAX,
|
|
ITEM_LIMITS_DIRECTION,
|
|
ITEM_LIMITS_CURVE,
|
|
#if defined(PPM_CENTER_ADJUSTABLE)
|
|
ITEM_LIMITS_PPM_CENTER,
|
|
#endif
|
|
#if defined(PPM_LIMITS_SYMETRICAL)
|
|
ITEM_LIMITS_SYMETRICAL,
|
|
#endif
|
|
ITEM_LIMITS_COUNT,
|
|
ITEM_LIMITS_MAXROW = ITEM_LIMITS_COUNT-1
|
|
};
|
|
|
|
#define LIMITS_NAME_POS 52
|
|
#define LIMITS_OFFSET_POS 160
|
|
#define LIMITS_MIN_POS 220
|
|
#define LIMITS_DIRECTION_POS 235
|
|
#define LIMITS_MAX_POS 300
|
|
#define LIMITS_REVERT_POS 312
|
|
#define LIMITS_CURVE_POS 340
|
|
#define LIMITS_PPM_CENTER_POS 440
|
|
#define LIMITS_SYMETRICAL_POS 450
|
|
|
|
#define LIMITS_MIN_MAX_OFFSET 1000
|
|
#define CONVERT_US_MIN_MAX(x) (((x)*1280)/250)
|
|
|
|
#if defined(PPM_UNIT_US)
|
|
#define SET_MIN_MAX(x, val) x = ((val)*250)/128
|
|
#define MIN_MAX_DISPLAY(x) CONVERT_US_MIN_MAX(x)
|
|
#else
|
|
#define MIN_MAX_DISPLAY(x) (x)
|
|
#define SET_MIN_MAX(x, val) x = (val)
|
|
#endif
|
|
|
|
void onLimitsMenu(const char *result)
|
|
{
|
|
uint8_t ch = menuVerticalPosition;
|
|
|
|
if (result == STR_RESET) {
|
|
LimitData *ld = limitAddress(ch);
|
|
ld->min = 0;
|
|
ld->max = 0;
|
|
ld->offset = 0;
|
|
ld->ppmCenter = 0;
|
|
ld->revert = false;
|
|
ld->curve = 0;
|
|
storageDirty(EE_MODEL);
|
|
}
|
|
else if (result == STR_COPY_STICKS_TO_OFS) {
|
|
copySticksToOffset(ch);
|
|
storageDirty(EE_MODEL);
|
|
}
|
|
else if (result == STR_COPY_TRIMS_TO_OFS) {
|
|
copyTrimsToOffset(ch);
|
|
storageDirty(EE_MODEL);
|
|
}
|
|
else if (result == STR_COPY_MIN_MAX_TO_OUTPUTS) {
|
|
copyMinMaxToOutputs(ch);
|
|
}
|
|
}
|
|
|
|
bool menuModelLimits(event_t event)
|
|
{
|
|
MENU(STR_MENULIMITS, MODEL_ICONS, menuTabModel, MENU_MODEL_OUTPUTS, MAX_OUTPUT_CHANNELS+1,
|
|
{ NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW,
|
|
NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW,
|
|
NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW,
|
|
NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW, NAVIGATION_LINE_BY_LINE|ITEM_LIMITS_MAXROW,
|
|
0 });
|
|
|
|
uint32_t sub = menuVerticalPosition;
|
|
|
|
if (sub < MAX_OUTPUT_CHANNELS) {
|
|
#if defined(PPM_CENTER_ADJUSTABLE) || defined(PPM_UNIT_US)
|
|
lcdDrawNumber(LCD_W / 2, MENU_TITLE_TOP + 1, PPM_CH_CENTER(sub)+channelOutputs[sub]/2, MENU_TITLE_COLOR, 0, "", STR_US);
|
|
#else
|
|
lcdDrawNumber(LCD_W / 2, MENU_TITLE_TOP + 1, calcRESXto1000(channelOutputs[sub]), PREC1 | MENU_TITLE_COLOR);
|
|
#endif
|
|
}
|
|
|
|
if (sub<MAX_OUTPUT_CHANNELS && menuHorizontalPosition>=0) {
|
|
drawColumnHeader(STR_LIMITS_HEADERS, NULL, menuHorizontalPosition);
|
|
}
|
|
|
|
for (int i=0; i<NUM_BODY_LINES; i++) {
|
|
coord_t y = MENU_CONTENT_TOP + i*FH;
|
|
uint8_t k = i+menuVerticalOffset;
|
|
|
|
if (k==MAX_OUTPUT_CHANNELS) {
|
|
// last line available - add the "copy trim menu" line
|
|
uint8_t attr = (sub==MAX_OUTPUT_CHANNELS) ? INVERS : 0;
|
|
// TODO CENTER attribute
|
|
lcdDrawText(100, y, STR_TRIMS2OFFSETS, NO_HIGHLIGHT() ? 0 : attr);
|
|
if (attr) {
|
|
s_editMode = 0;
|
|
if (event==EVT_KEY_LONG(KEY_ENTER)) {
|
|
START_NO_HIGHLIGHT();
|
|
killEvents(event);
|
|
moveTrimsToOffsets(); // if highlighted and menu pressed - move trims to offsets
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
LimitData *ld = limitAddress(k);
|
|
|
|
int16_t v = (ld->revert) ? -LIMIT_OFS(ld) : LIMIT_OFS(ld);
|
|
char swVal[2] = "-"; // '-', '<', '>'
|
|
if ((channelOutputs[k] - v) > 50) swVal[0] = (ld->revert ? 127 : 126); // Switch to raw inputs? - remove trim!
|
|
if ((channelOutputs[k] - v) < -50) swVal[0] = (ld->revert ? 126 : 127);
|
|
lcdDrawText(LIMITS_DIRECTION_POS, y, swVal);
|
|
|
|
int limit = (g_model.extendedLimits ? LIMIT_EXT_MAX : 1000);
|
|
|
|
putsChn(MENUS_MARGIN_LEFT, y, k+1, (sub==k && menuHorizontalPosition < 0) ? INVERS : 0);
|
|
if (sub==k && menuHorizontalPosition < 0 && event==EVT_KEY_LONG(KEY_ENTER) && !READ_ONLY()) {
|
|
killEvents(event);
|
|
POPUP_MENU_ADD_ITEM(STR_RESET);
|
|
POPUP_MENU_ADD_ITEM(STR_COPY_TRIMS_TO_OFS);
|
|
POPUP_MENU_ADD_ITEM(STR_COPY_STICKS_TO_OFS);
|
|
POPUP_MENU_ADD_ITEM(STR_COPY_MIN_MAX_TO_OUTPUTS);
|
|
POPUP_MENU_START(onLimitsMenu);
|
|
}
|
|
|
|
for (uint8_t j=0; j<ITEM_LIMITS_COUNT; j++) {
|
|
uint8_t attr = ((sub==k && menuHorizontalPosition==j) ? ((s_editMode>0) ? BLINK|INVERS : INVERS) : 0);
|
|
uint8_t active = (attr && s_editMode>0) ;
|
|
switch(j)
|
|
{
|
|
case ITEM_LIMITS_CH_NAME:
|
|
editName(LIMITS_NAME_POS, y, ld->name, sizeof(ld->name), event, attr);
|
|
break;
|
|
|
|
case ITEM_LIMITS_OFFSET:
|
|
if (GV_IS_GV_VALUE(ld->offset, -1000, 1000) || (attr && event == EVT_KEY_LONG(KEY_ENTER))) {
|
|
ld->offset = GVAR_MENU_ITEM(LIMITS_OFFSET_POS, y, ld->offset, -1000, 1000, attr|PREC1|RIGHT, 0, event);
|
|
break;
|
|
}
|
|
|
|
#if defined(PPM_UNIT_US)
|
|
lcdDrawNumber(LIMITS_OFFSET_POS, y, ((int32_t)ld->offset*128) / 25, attr|PREC1|RIGHT);
|
|
#else
|
|
lcdDrawNumber(LIMITS_OFFSET_POS, y, ld->offset, attr|PREC1|RIGHT);
|
|
#endif
|
|
if (active) {
|
|
ld->offset = checkIncDec(event, ld->offset, -1000, 1000, EE_MODEL, NULL, stops1000);
|
|
}
|
|
|
|
// TODO with the contextual menu I think
|
|
// else if (attr && event==EVT_KEY_LONG(KEY_MENU)) {
|
|
// copySticksToOffset(k);
|
|
// s_editMode = 0;
|
|
// }
|
|
break;
|
|
|
|
case ITEM_LIMITS_MIN:
|
|
if (GV_IS_GV_VALUE(ld->min, -GV_RANGELARGE, GV_RANGELARGE) || (attr && event == EVT_KEY_LONG(KEY_ENTER))) {
|
|
ld->min = GVAR_MENU_ITEM(LIMITS_MIN_POS, y, ld->min, -LIMIT_EXT_MAX, LIMIT_EXT_MAX, attr|PREC1|RIGHT, 0, event);
|
|
break;
|
|
}
|
|
lcdDrawNumber(LIMITS_MIN_POS, y, MIN_MAX_DISPLAY(ld->min-LIMITS_MIN_MAX_OFFSET), attr|PREC1|RIGHT);
|
|
if (active) ld->min = LIMITS_MIN_MAX_OFFSET + checkIncDec(event, ld->min-LIMITS_MIN_MAX_OFFSET, -limit, 0, EE_MODEL, NULL, stops1000);
|
|
break;
|
|
|
|
case ITEM_LIMITS_MAX:
|
|
if (GV_IS_GV_VALUE(ld->max, -GV_RANGELARGE, GV_RANGELARGE) || (attr && event == EVT_KEY_LONG(KEY_ENTER))) {
|
|
ld->max = GVAR_MENU_ITEM(LIMITS_MAX_POS, y, ld->max, -LIMIT_EXT_MAX, LIMIT_EXT_MAX, attr|PREC1|RIGHT, 0, event);
|
|
break;
|
|
}
|
|
lcdDrawNumber(LIMITS_MAX_POS, y, MIN_MAX_DISPLAY(ld->max+LIMITS_MIN_MAX_OFFSET), attr|PREC1|RIGHT);
|
|
if (active) ld->max = -LIMITS_MIN_MAX_OFFSET + checkIncDec(event, ld->max+LIMITS_MIN_MAX_OFFSET, 0, +limit, EE_MODEL, NULL, stops1000);
|
|
break;
|
|
|
|
case ITEM_LIMITS_DIRECTION:
|
|
{
|
|
lcdDrawText(LIMITS_REVERT_POS, y, ld->revert ? "\177" : "\176", attr);
|
|
if (active) {
|
|
CHECK_INCDEC_MODELVAR_ZERO(event, ld->revert, 1);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case ITEM_LIMITS_CURVE:
|
|
drawCurveName(LIMITS_CURVE_POS, y, ld->curve, attr);
|
|
if (attr && event==EVT_KEY_LONG(KEY_ENTER) && ld->curve>0) {
|
|
s_currIdxSubMenu = (ld->curve<0 ? -ld->curve-1 : ld->curve-1);
|
|
pushMenu(menuModelCurveOne);
|
|
}
|
|
if (active) {
|
|
CHECK_INCDEC_MODELVAR(event, ld->curve, -MAX_CURVES, +MAX_CURVES);
|
|
}
|
|
break;
|
|
|
|
#if defined(PPM_CENTER_ADJUSTABLE)
|
|
case ITEM_LIMITS_PPM_CENTER:
|
|
lcdDrawNumber(LIMITS_PPM_CENTER_POS, y, PPM_CENTER+ld->ppmCenter, attr|RIGHT);
|
|
if (active) {
|
|
CHECK_INCDEC_MODELVAR(event, ld->ppmCenter, -PPM_CENTER_MAX, +PPM_CENTER_MAX);
|
|
}
|
|
break;
|
|
#endif
|
|
|
|
#if defined(PPM_LIMITS_SYMETRICAL)
|
|
case ITEM_LIMITS_SYMETRICAL:
|
|
lcdDrawText(LIMITS_SYMETRICAL_POS, y, ld->symetrical ? "=" : "\306", attr);
|
|
if (active) {
|
|
CHECK_INCDEC_MODELVAR_ZERO(event, ld->symetrical, 1);
|
|
}
|
|
break;
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|