mirror of
https://github.com/opentx/opentx.git
synced 2025-07-23 08:15:17 +03:00
290 lines
9.1 KiB
C++
290 lines
9.1 KiB
C++
/*
|
|
* Authors (alphabetical order)
|
|
* - Andre Bernet <bernet.andre@gmail.com>
|
|
* - Andreas Weitl
|
|
* - Bertrand Songis <bsongis@gmail.com>
|
|
* - Bryan J. Rentoul (Gruvin) <gruvin@gmail.com>
|
|
* - Cameron Weeks <th9xer@gmail.com>
|
|
* - Erez Raviv
|
|
* - Gabriel Birkus
|
|
* - Jean-Pierre Parisy
|
|
* - Karl Szmutny
|
|
* - Michael Blandford
|
|
* - Michal Hlavinka
|
|
* - Pat Mackenzie
|
|
* - Philip Moss
|
|
* - Rob Thomson
|
|
* - Thomas Husterer
|
|
*
|
|
* opentx is based on code named
|
|
* gruvin9x by Bryan J. Rentoul: http://code.google.com/p/gruvin9x/,
|
|
* er9x by Erez Raviv: http://code.google.com/p/er9x/,
|
|
* and the original (and ongoing) project by
|
|
* Thomas Husterer, th9x: http://code.google.com/p/th9x/
|
|
*
|
|
* 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"
|
|
|
|
#define MODELSIZE_POS_X 170
|
|
#define MODELSEL_W 133
|
|
|
|
void onModelSelectMenu(const char *result)
|
|
{
|
|
int8_t sub = m_posVert;
|
|
|
|
if (result == STR_SELECT_MODEL || result == STR_CREATE_MODEL) {
|
|
selectModel(sub);
|
|
}
|
|
else if (result == STR_COPY_MODEL) {
|
|
s_copyMode = COPY_MODE;
|
|
s_copyTgtOfs = 0;
|
|
s_copySrcRow = -1;
|
|
}
|
|
else if (result == STR_MOVE_MODEL) {
|
|
s_copyMode = MOVE_MODE;
|
|
s_copyTgtOfs = 0;
|
|
s_copySrcRow = -1;
|
|
}
|
|
else if (result == STR_BACKUP_MODEL) {
|
|
storageCheck(true); // force writing of current model data before this is changed
|
|
POPUP_WARNING(eeBackupModel(sub));
|
|
}
|
|
else if (result == STR_RESTORE_MODEL || result == STR_UPDATE_LIST) {
|
|
if (!sdListFiles(MODELS_PATH, MODELS_EXT, MENU_LINE_LENGTH-1, NULL)) {
|
|
POPUP_WARNING(STR_NO_MODELS_ON_SD);
|
|
s_menu_flags = 0;
|
|
}
|
|
}
|
|
else if (result == STR_DELETE_MODEL) {
|
|
POPUP_CONFIRMATION(STR_DELETEMODEL);
|
|
SET_WARNING_INFO(modelHeaders[sub].name, sizeof(g_model.header.name), ZCHAR);
|
|
}
|
|
else {
|
|
// The user choosed a file on SD to restore
|
|
storageCheck(true);
|
|
POPUP_WARNING(eeRestoreModel(sub, (char *)result));
|
|
if (!s_warning && g_eeGeneral.currModel == sub) {
|
|
eeLoadModel(sub);
|
|
}
|
|
}
|
|
}
|
|
|
|
void menuModelSelect(uint8_t event)
|
|
{
|
|
if (s_warning_result) {
|
|
s_warning_result = 0;
|
|
storageCheck(true);
|
|
eeDeleteModel(m_posVert); // delete file
|
|
s_copyMode = 0;
|
|
event = EVT_ENTRY_UP;
|
|
}
|
|
|
|
uint8_t _event_ = ((event==EVT_KEY_BREAK(KEY_ENTER) || event==EVT_KEY_LONG(KEY_ENTER)) ? 0 : event);
|
|
|
|
if ((s_copyMode && EVT_KEY_MASK(event) == KEY_EXIT) || event == EVT_KEY_BREAK(KEY_EXIT)) {
|
|
_event_ -= KEY_EXIT;
|
|
}
|
|
|
|
int8_t oldSub = m_posVert;
|
|
|
|
check_submenu_simple(NULL, _event_, MAX_MODELS);
|
|
|
|
if (s_editMode > 0) s_editMode = 0;
|
|
|
|
int sub = m_posVert;
|
|
|
|
switch (event)
|
|
{
|
|
case EVT_ENTRY:
|
|
m_posVert = sub = g_eeGeneral.currModel;
|
|
if (sub >= NUM_BODY_LINES) s_pgOfs = sub-(NUM_BODY_LINES-1);
|
|
s_copyMode = 0;
|
|
s_editMode = EDIT_MODE_INIT;
|
|
break;
|
|
|
|
case EVT_KEY_LONG(KEY_EXIT):
|
|
if (s_copyMode && s_copyTgtOfs == 0 && g_eeGeneral.currModel != sub && eeModelExists(sub)) {
|
|
POPUP_CONFIRMATION(STR_DELETEMODEL);
|
|
SET_WARNING_INFO(modelHeaders[sub].name, sizeof(g_model.header.name), ZCHAR);
|
|
killEvents(event);
|
|
break;
|
|
}
|
|
// no break
|
|
case EVT_KEY_BREAK(KEY_EXIT):
|
|
if (s_copyMode) {
|
|
sub = m_posVert = (s_copyMode == MOVE_MODE || s_copySrcRow<0) ? (MAX_MODELS+sub+s_copyTgtOfs) % MAX_MODELS : s_copySrcRow;
|
|
s_copyMode = 0;
|
|
}
|
|
else {
|
|
if (m_posVert != g_eeGeneral.currModel) {
|
|
sub = m_posVert = g_eeGeneral.currModel;
|
|
s_pgOfs = 0;
|
|
}
|
|
else if (event != EVT_KEY_LONG(KEY_EXIT)) {
|
|
popMenu();
|
|
}
|
|
}
|
|
break;
|
|
case EVT_KEY_LONG(KEY_ENTER):
|
|
case EVT_KEY_BREAK(KEY_ENTER):
|
|
s_editMode = 0;
|
|
if (READ_ONLY()) {
|
|
if (g_eeGeneral.currModel != sub && eeModelExists(sub)) {
|
|
selectModel(sub);
|
|
}
|
|
}
|
|
else if (s_copyMode && (s_copyTgtOfs || s_copySrcRow>=0)) {
|
|
displayPopup(s_copyMode==COPY_MODE ? STR_COPYINGMODEL : STR_MOVINGMODEL);
|
|
storageCheck(true); // force writing of current model data before this is changed
|
|
|
|
uint8_t cur = (MAX_MODELS + sub + s_copyTgtOfs) % MAX_MODELS;
|
|
|
|
if (s_copyMode == COPY_MODE) {
|
|
if (!eeCopyModel(cur, s_copySrcRow))
|
|
cur = sub;
|
|
}
|
|
|
|
s_copySrcRow = g_eeGeneral.currModel; // to update the currModel value
|
|
while (sub != cur) {
|
|
uint8_t src = cur;
|
|
cur = (s_copyTgtOfs > 0 ? cur+MAX_MODELS-1 : cur+1) % MAX_MODELS;
|
|
eeSwapModels(src, cur);
|
|
if (src == s_copySrcRow)
|
|
s_copySrcRow = cur;
|
|
else if (cur == s_copySrcRow)
|
|
s_copySrcRow = src;
|
|
}
|
|
|
|
if (s_copySrcRow != g_eeGeneral.currModel) {
|
|
g_eeGeneral.currModel = s_copySrcRow;
|
|
storageDirty(EE_GENERAL);
|
|
}
|
|
|
|
s_copyMode = 0;
|
|
event = EVT_ENTRY_UP;
|
|
}
|
|
else if (event == EVT_KEY_LONG(KEY_ENTER)) {
|
|
s_copyMode = 0;
|
|
killEvents(event);
|
|
if (g_eeGeneral.currModel != sub) {
|
|
if (eeModelExists(sub)) {
|
|
MENU_ADD_ITEM(STR_SELECT_MODEL);
|
|
MENU_ADD_SD_ITEM(STR_BACKUP_MODEL);
|
|
MENU_ADD_ITEM(STR_COPY_MODEL);
|
|
MENU_ADD_ITEM(STR_MOVE_MODEL);
|
|
MENU_ADD_ITEM(STR_DELETE_MODEL);
|
|
}
|
|
else {
|
|
MENU_ADD_ITEM(STR_CREATE_MODEL);
|
|
MENU_ADD_ITEM(STR_RESTORE_MODEL);
|
|
}
|
|
}
|
|
else {
|
|
MENU_ADD_SD_ITEM(STR_BACKUP_MODEL);
|
|
MENU_ADD_ITEM(STR_COPY_MODEL);
|
|
MENU_ADD_ITEM(STR_MOVE_MODEL);
|
|
}
|
|
menuHandler = onModelSelectMenu;
|
|
}
|
|
else if (eeModelExists(sub)) {
|
|
s_copyMode = (s_copyMode == COPY_MODE ? MOVE_MODE : COPY_MODE);
|
|
s_copyTgtOfs = 0;
|
|
s_copySrcRow = -1;
|
|
}
|
|
break;
|
|
|
|
case EVT_KEY_BREAK(KEY_PAGE):
|
|
case EVT_KEY_LONG(KEY_PAGE):
|
|
chainMenu(event == EVT_KEY_BREAK(KEY_PAGE) ? menuModelSetup : menuTabModel[DIM(menuTabModel)-1]);
|
|
killEvents(event);
|
|
break;
|
|
|
|
case EVT_KEY_FIRST(KEY_UP):
|
|
case EVT_KEY_REPT(KEY_UP):
|
|
case EVT_KEY_FIRST(KEY_DOWN):
|
|
case EVT_KEY_REPT(KEY_DOWN):
|
|
if (s_copyMode) {
|
|
int8_t next_ofs = s_copyTgtOfs + oldSub - m_posVert;
|
|
if (next_ofs == MAX_MODELS || next_ofs == -MAX_MODELS)
|
|
next_ofs = 0;
|
|
|
|
if (s_copySrcRow < 0 && s_copyMode==COPY_MODE) {
|
|
s_copySrcRow = oldSub;
|
|
// find a hole (in the first empty slot above / below)
|
|
sub = eeFindEmptyModel(s_copySrcRow, event==EVT_KEY_FIRST(KEY_DOWN) || event==EVT_KEY_REPT(KEY_DOWN));
|
|
if (sub < 0) {
|
|
// no free room for duplicating the model
|
|
AUDIO_ERROR();
|
|
sub = oldSub;
|
|
s_copyMode = 0;
|
|
}
|
|
next_ofs = 0;
|
|
m_posVert = sub;
|
|
}
|
|
s_copyTgtOfs = next_ofs;
|
|
}
|
|
break;
|
|
}
|
|
|
|
lcd_puts(27*FW-(LEN_FREE-4)*FW, 0, STR_FREE);
|
|
lcd_outdezAtt(20*FW, 0, EeFsGetFree(), 0);
|
|
lcd_puts(21*FW, 0, STR_BYTES);
|
|
|
|
displayScreenIndex(e_ModelSelect, DIM(menuTabModel), 0);
|
|
drawFilledRect(0, 0, LCD_W, FH, SOLID, FILL_WHITE|GREY_DEFAULT);
|
|
|
|
TITLE(STR_MENUMODELSEL);
|
|
|
|
for (uint8_t i=0; i<NUM_BODY_LINES; i++) {
|
|
coord_t y = MENU_HEADER_HEIGHT + 1 + i*FH;
|
|
uint8_t k = i+s_pgOfs;
|
|
|
|
lcd_outdezNAtt(3*FW+2, y, k+1, LEADING0+((!s_copyMode && sub==k) ? INVERS : 0), 2);
|
|
|
|
if (s_copyMode == MOVE_MODE || (s_copyMode == COPY_MODE && s_copySrcRow >= 0)) {
|
|
if (k == sub) {
|
|
if (s_copyMode == COPY_MODE) {
|
|
k = s_copySrcRow;
|
|
lcd_putc(MODELSEL_W-FW, y, '+');
|
|
}
|
|
else {
|
|
k = sub + s_copyTgtOfs;
|
|
}
|
|
}
|
|
else if (s_copyTgtOfs < 0 && ((k < sub && k >= sub+s_copyTgtOfs) || (k-MAX_MODELS < sub && k-MAX_MODELS >= sub+s_copyTgtOfs)))
|
|
k += 1;
|
|
else if (s_copyTgtOfs > 0 && ((k > sub && k <= sub+s_copyTgtOfs) || (k+MAX_MODELS > sub && k+MAX_MODELS <= sub+s_copyTgtOfs)))
|
|
k += MAX_MODELS-1;
|
|
}
|
|
|
|
k %= MAX_MODELS;
|
|
|
|
if (eeModelExists(k)) {
|
|
putsModelName(4*FW, y, modelHeaders[k].name, k, 0);
|
|
lcd_outdezAtt(20*FW, y, eeModelSize(k), 0);
|
|
if (k==g_eeGeneral.currModel && (s_copyMode!=COPY_MODE || s_copySrcRow<0 || i+s_pgOfs!=(vertpos_t)sub))
|
|
lcd_putc(1, y, '*');
|
|
}
|
|
|
|
if (s_copyMode && (vertpos_t)sub==i+s_pgOfs) {
|
|
drawFilledRect(9, y, MODELSEL_W-1-9, 7);
|
|
lcdDrawRect(8, y-1, MODELSEL_W-1-7, 9, s_copyMode == COPY_MODE ? SOLID : DOTTED);
|
|
}
|
|
}
|
|
|
|
if (event == EVT_ENTRY || sub != oldSub) {
|
|
loadModelBitmap(modelHeaders[sub].bitmap, modelBitmap);
|
|
}
|
|
|
|
lcd_bmp(22*FW+2, 2*FH+FH/2, modelBitmap);
|
|
}
|