mirror of
https://github.com/Next-Flip/Momentum-Firmware.git
synced 2026-05-14 15:08:35 -07:00
Merge branch 'Next-Flip:dev' into dev
This commit is contained in:
Submodule applications/external updated: 3da3a125fe...3dbd298b07
@@ -163,6 +163,7 @@ MomentumApp* momentum_app_alloc() {
|
|||||||
MomentumApp* app = malloc(sizeof(MomentumApp));
|
MomentumApp* app = malloc(sizeof(MomentumApp));
|
||||||
app->gui = furi_record_open(RECORD_GUI);
|
app->gui = furi_record_open(RECORD_GUI);
|
||||||
app->dialogs = furi_record_open(RECORD_DIALOGS);
|
app->dialogs = furi_record_open(RECORD_DIALOGS);
|
||||||
|
app->expansion = furi_record_open(RECORD_EXPANSION);
|
||||||
app->notification = furi_record_open(RECORD_NOTIFICATION);
|
app->notification = furi_record_open(RECORD_NOTIFICATION);
|
||||||
|
|
||||||
// View Dispatcher and Scene Manager
|
// View Dispatcher and Scene Manager
|
||||||
@@ -361,6 +362,7 @@ void momentum_app_free(MomentumApp* app) {
|
|||||||
|
|
||||||
// Records
|
// Records
|
||||||
furi_record_close(RECORD_NOTIFICATION);
|
furi_record_close(RECORD_NOTIFICATION);
|
||||||
|
furi_record_close(RECORD_EXPANSION);
|
||||||
furi_record_close(RECORD_DIALOGS);
|
furi_record_close(RECORD_DIALOGS);
|
||||||
furi_record_close(RECORD_GUI);
|
furi_record_close(RECORD_GUI);
|
||||||
free(app);
|
free(app);
|
||||||
|
|||||||
@@ -25,6 +25,7 @@
|
|||||||
#include <flipper_application/flipper_application.h>
|
#include <flipper_application/flipper_application.h>
|
||||||
#include <notification/notification_app.h>
|
#include <notification/notification_app.h>
|
||||||
#include <power/power_service/power.h>
|
#include <power/power_service/power.h>
|
||||||
|
#include <expansion/expansion.h>
|
||||||
#include <rgb_backlight.h>
|
#include <rgb_backlight.h>
|
||||||
#include <m-array.h>
|
#include <m-array.h>
|
||||||
#include <momentum/namespoof.h>
|
#include <momentum/namespoof.h>
|
||||||
@@ -35,6 +36,7 @@ ARRAY_DEF(CharList, char*)
|
|||||||
typedef struct {
|
typedef struct {
|
||||||
Gui* gui;
|
Gui* gui;
|
||||||
DialogsApp* dialogs;
|
DialogsApp* dialogs;
|
||||||
|
Expansion* expansion;
|
||||||
NotificationApp* notification;
|
NotificationApp* notification;
|
||||||
SceneManager* scene_manager;
|
SceneManager* scene_manager;
|
||||||
ViewDispatcher* view_dispatcher;
|
ViewDispatcher* view_dispatcher;
|
||||||
@@ -58,6 +60,7 @@ typedef struct {
|
|||||||
char subghz_freq_buffer[7];
|
char subghz_freq_buffer[7];
|
||||||
bool subghz_extend;
|
bool subghz_extend;
|
||||||
RgbColor lcd_color;
|
RgbColor lcd_color;
|
||||||
|
Rgb565Color vgm_color;
|
||||||
char device_name[FURI_HAL_VERSION_ARRAY_NAME_LENGTH];
|
char device_name[FURI_HAL_VERSION_ARRAY_NAME_LENGTH];
|
||||||
int32_t dolphin_level;
|
int32_t dolphin_level;
|
||||||
int32_t dolphin_angry;
|
int32_t dolphin_angry;
|
||||||
|
|||||||
@@ -18,4 +18,6 @@ ADD_SCENE(momentum_app, misc, Misc)
|
|||||||
ADD_SCENE(momentum_app, misc_screen, MiscScreen)
|
ADD_SCENE(momentum_app, misc_screen, MiscScreen)
|
||||||
ADD_SCENE(momentum_app, misc_screen_color, MiscScreenColor)
|
ADD_SCENE(momentum_app, misc_screen_color, MiscScreenColor)
|
||||||
ADD_SCENE(momentum_app, misc_dolphin, MiscDolphin)
|
ADD_SCENE(momentum_app, misc_dolphin, MiscDolphin)
|
||||||
|
ADD_SCENE(momentum_app, misc_vgm, MiscVgm)
|
||||||
|
ADD_SCENE(momentum_app, misc_vgm_color, MiscVgmColor)
|
||||||
ADD_SCENE(momentum_app, misc_rename, MiscRename)
|
ADD_SCENE(momentum_app, misc_rename, MiscRename)
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
enum VarItemListIndex {
|
enum VarItemListIndex {
|
||||||
VarItemListIndexScreen,
|
VarItemListIndexScreen,
|
||||||
VarItemListIndexDolphin,
|
VarItemListIndexDolphin,
|
||||||
|
VarItemListIndexVgm,
|
||||||
VarItemListIndexChangeDeviceName,
|
VarItemListIndexChangeDeviceName,
|
||||||
VarItemListIndexChargeCap,
|
VarItemListIndexChargeCap,
|
||||||
VarItemListIndexShowMomentumIntro,
|
VarItemListIndexShowMomentumIntro,
|
||||||
@@ -36,6 +37,9 @@ void momentum_app_scene_misc_on_enter(void* context) {
|
|||||||
item = variable_item_list_add(var_item_list, "Dolphin", 0, NULL, app);
|
item = variable_item_list_add(var_item_list, "Dolphin", 0, NULL, app);
|
||||||
variable_item_set_current_value_text(item, ">");
|
variable_item_set_current_value_text(item, ">");
|
||||||
|
|
||||||
|
item = variable_item_list_add(var_item_list, "VGM Options", 0, NULL, app);
|
||||||
|
variable_item_set_current_value_text(item, ">");
|
||||||
|
|
||||||
variable_item_list_add(var_item_list, "Change Device Name", 0, NULL, app);
|
variable_item_list_add(var_item_list, "Change Device Name", 0, NULL, app);
|
||||||
|
|
||||||
char cap_str[6];
|
char cap_str[6];
|
||||||
@@ -77,6 +81,10 @@ bool momentum_app_scene_misc_on_event(void* context, SceneManagerEvent event) {
|
|||||||
scene_manager_set_scene_state(app->scene_manager, MomentumAppSceneMiscDolphin, 0);
|
scene_manager_set_scene_state(app->scene_manager, MomentumAppSceneMiscDolphin, 0);
|
||||||
scene_manager_next_scene(app->scene_manager, MomentumAppSceneMiscDolphin);
|
scene_manager_next_scene(app->scene_manager, MomentumAppSceneMiscDolphin);
|
||||||
break;
|
break;
|
||||||
|
case VarItemListIndexVgm:
|
||||||
|
scene_manager_set_scene_state(app->scene_manager, MomentumAppSceneMiscVgm, 0);
|
||||||
|
scene_manager_next_scene(app->scene_manager, MomentumAppSceneMiscVgm);
|
||||||
|
break;
|
||||||
case VarItemListIndexChangeDeviceName:
|
case VarItemListIndexChangeDeviceName:
|
||||||
scene_manager_set_scene_state(app->scene_manager, MomentumAppSceneMiscRename, 0);
|
scene_manager_set_scene_state(app->scene_manager, MomentumAppSceneMiscRename, 0);
|
||||||
scene_manager_next_scene(app->scene_manager, MomentumAppSceneMiscRename);
|
scene_manager_next_scene(app->scene_manager, MomentumAppSceneMiscRename);
|
||||||
|
|||||||
@@ -55,6 +55,10 @@ static void momentum_app_scene_misc_screen_lcd_color_changed(VariableItem* item,
|
|||||||
variable_item_set_current_value_text(item, lcd_colors[index].name);
|
variable_item_set_current_value_text(item, lcd_colors[index].name);
|
||||||
rgb_backlight_set_color(led, &lcd_colors[index].color);
|
rgb_backlight_set_color(led, &lcd_colors[index].color);
|
||||||
app->save_backlight = true;
|
app->save_backlight = true;
|
||||||
|
if(momentum_settings.vgm_color_mode == VgmColorModeRgbBacklight) {
|
||||||
|
expansion_disable(app->expansion);
|
||||||
|
expansion_enable(app->expansion);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
static void momentum_app_scene_misc_screen_lcd_color_0_changed(VariableItem* item) {
|
static void momentum_app_scene_misc_screen_lcd_color_0_changed(VariableItem* item) {
|
||||||
momentum_app_scene_misc_screen_lcd_color_changed(item, 0);
|
momentum_app_scene_misc_screen_lcd_color_changed(item, 0);
|
||||||
|
|||||||
@@ -43,6 +43,10 @@ bool momentum_app_scene_misc_screen_color_on_event(void* context, SceneManagerEv
|
|||||||
scene_manager_get_scene_state(app->scene_manager, MomentumAppSceneMiscScreenColor),
|
scene_manager_get_scene_state(app->scene_manager, MomentumAppSceneMiscScreenColor),
|
||||||
&app->lcd_color);
|
&app->lcd_color);
|
||||||
app->save_backlight = true;
|
app->save_backlight = true;
|
||||||
|
if(momentum_settings.vgm_color_mode == VgmColorModeRgbBacklight) {
|
||||||
|
expansion_disable(app->expansion);
|
||||||
|
expansion_enable(app->expansion);
|
||||||
|
}
|
||||||
scene_manager_previous_scene(app->scene_manager);
|
scene_manager_previous_scene(app->scene_manager);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
|||||||
@@ -0,0 +1,173 @@
|
|||||||
|
#include "../momentum_app.h"
|
||||||
|
|
||||||
|
enum VarItemListIndex {
|
||||||
|
VarItemListIndexColors,
|
||||||
|
VarItemListIndexForeground,
|
||||||
|
VarItemListIndexBackground,
|
||||||
|
};
|
||||||
|
|
||||||
|
void momentum_app_scene_misc_vgm_var_item_list_callback(void* context, uint32_t index) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
view_dispatcher_send_custom_event(app->view_dispatcher, index);
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* const colors_names[VgmColorModeCount] = {
|
||||||
|
"Default",
|
||||||
|
"Custom",
|
||||||
|
"RGB Backlight",
|
||||||
|
};
|
||||||
|
static void momentum_app_scene_misc_vgm_colors_changed(VariableItem* item) {
|
||||||
|
MomentumApp* app = variable_item_get_context(item);
|
||||||
|
uint8_t index = variable_item_get_current_value_index(item);
|
||||||
|
variable_item_set_current_value_text(item, colors_names[index]);
|
||||||
|
momentum_settings.vgm_color_mode = index;
|
||||||
|
app->save_settings = true;
|
||||||
|
variable_item_set_locked(
|
||||||
|
variable_item_list_get(app->var_item_list, VarItemListIndexForeground),
|
||||||
|
index != VgmColorModeCustom,
|
||||||
|
NULL);
|
||||||
|
variable_item_set_locked(
|
||||||
|
variable_item_list_get(app->var_item_list, VarItemListIndexBackground),
|
||||||
|
index != VgmColorModeCustom,
|
||||||
|
NULL);
|
||||||
|
expansion_disable(app->expansion);
|
||||||
|
expansion_enable(app->expansion);
|
||||||
|
}
|
||||||
|
|
||||||
|
static const struct {
|
||||||
|
char* name;
|
||||||
|
Rgb565Color color;
|
||||||
|
} vgm_colors[] = {
|
||||||
|
{"Orange", {0xFC00}}, {"Black", {0x0000}}, {"Red", {0xF800}}, {"Maroon", {0x8000}},
|
||||||
|
{"Yellow", {0xFFE0}}, {"Olive", {0x8400}}, {"Lime", {0x07E0}}, {"Green", {0x0400}},
|
||||||
|
{"Aqua", {0x07EF}}, {"Cyan", {0x069A}}, {"Azure", {0x03FF}}, {"Teal", {0x0410}},
|
||||||
|
{"Blue", {0x001F}}, {"Navy", {0x0010}}, {"Purple", {0x8010}}, {"Fuchsia", {0xF81F}},
|
||||||
|
{"Pink", {0xA8F5}}, {"Brown", {0xA145}}, {"White", {0xFFFF}},
|
||||||
|
};
|
||||||
|
static const size_t vgm_colors_count = COUNT_OF(vgm_colors);
|
||||||
|
static void momentum_app_scene_misc_vgm_foreground_changed(VariableItem* item) {
|
||||||
|
MomentumApp* app = variable_item_get_context(item);
|
||||||
|
uint8_t index = variable_item_get_current_value_index(item);
|
||||||
|
variable_item_set_current_value_text(item, vgm_colors[index].name);
|
||||||
|
momentum_settings.vgm_color_fg = vgm_colors[index].color;
|
||||||
|
app->save_settings = true;
|
||||||
|
if(momentum_settings.vgm_color_mode == VgmColorModeCustom) {
|
||||||
|
expansion_disable(app->expansion);
|
||||||
|
expansion_enable(app->expansion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
static void momentum_app_scene_misc_vgm_background_changed(VariableItem* item) {
|
||||||
|
MomentumApp* app = variable_item_get_context(item);
|
||||||
|
uint8_t index = variable_item_get_current_value_index(item);
|
||||||
|
variable_item_set_current_value_text(item, vgm_colors[index].name);
|
||||||
|
momentum_settings.vgm_color_bg = vgm_colors[index].color;
|
||||||
|
app->save_settings = true;
|
||||||
|
if(momentum_settings.vgm_color_mode == VgmColorModeCustom) {
|
||||||
|
expansion_disable(app->expansion);
|
||||||
|
expansion_enable(app->expansion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void momentum_app_scene_misc_vgm_on_enter(void* context) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
VariableItemList* var_item_list = app->var_item_list;
|
||||||
|
VariableItem* item;
|
||||||
|
uint8_t value_index;
|
||||||
|
|
||||||
|
item = variable_item_list_add(
|
||||||
|
var_item_list,
|
||||||
|
"VGM Colors",
|
||||||
|
VgmColorModeCount,
|
||||||
|
momentum_app_scene_misc_vgm_colors_changed,
|
||||||
|
app);
|
||||||
|
value_index = momentum_settings.vgm_color_mode;
|
||||||
|
variable_item_set_current_value_index(item, value_index);
|
||||||
|
variable_item_set_current_value_text(item, colors_names[value_index]);
|
||||||
|
|
||||||
|
item = variable_item_list_add(
|
||||||
|
var_item_list,
|
||||||
|
"Foreground",
|
||||||
|
vgm_colors_count,
|
||||||
|
momentum_app_scene_misc_vgm_foreground_changed,
|
||||||
|
app);
|
||||||
|
Rgb565Color color = momentum_settings.vgm_color_fg;
|
||||||
|
bool found = false;
|
||||||
|
for(size_t i = 0; i < vgm_colors_count; i++) {
|
||||||
|
if(rgb565cmp(&color, &vgm_colors[i].color) != 0) continue;
|
||||||
|
value_index = i;
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
variable_item_set_current_value_index(item, found ? value_index : vgm_colors_count);
|
||||||
|
if(found) {
|
||||||
|
variable_item_set_current_value_text(item, vgm_colors[value_index].name);
|
||||||
|
} else {
|
||||||
|
char str[5];
|
||||||
|
snprintf(str, sizeof(str), "%04X", color.value);
|
||||||
|
variable_item_set_current_value_text(item, str);
|
||||||
|
}
|
||||||
|
variable_item_set_locked(
|
||||||
|
item, momentum_settings.vgm_color_mode != VgmColorModeCustom, "Need Custom\nColors!");
|
||||||
|
|
||||||
|
item = variable_item_list_add(
|
||||||
|
var_item_list,
|
||||||
|
"Background",
|
||||||
|
vgm_colors_count,
|
||||||
|
momentum_app_scene_misc_vgm_background_changed,
|
||||||
|
app);
|
||||||
|
color = momentum_settings.vgm_color_bg;
|
||||||
|
found = false;
|
||||||
|
for(size_t i = 0; i < vgm_colors_count; i++) {
|
||||||
|
if(rgb565cmp(&color, &vgm_colors[i].color) != 0) continue;
|
||||||
|
value_index = i;
|
||||||
|
found = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
variable_item_set_current_value_index(item, found ? value_index : vgm_colors_count);
|
||||||
|
if(found) {
|
||||||
|
variable_item_set_current_value_text(item, vgm_colors[value_index].name);
|
||||||
|
} else {
|
||||||
|
char str[5];
|
||||||
|
snprintf(str, sizeof(str), "%04X", color.value);
|
||||||
|
variable_item_set_current_value_text(item, str);
|
||||||
|
}
|
||||||
|
variable_item_set_locked(
|
||||||
|
item, momentum_settings.vgm_color_mode != VgmColorModeCustom, "Need Custom\nColors!");
|
||||||
|
|
||||||
|
variable_item_list_set_enter_callback(
|
||||||
|
var_item_list, momentum_app_scene_misc_vgm_var_item_list_callback, app);
|
||||||
|
|
||||||
|
variable_item_list_set_selected_item(
|
||||||
|
var_item_list, scene_manager_get_scene_state(app->scene_manager, MomentumAppSceneMiscVgm));
|
||||||
|
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, MomentumAppViewVarItemList);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool momentum_app_scene_misc_vgm_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
|
scene_manager_set_scene_state(app->scene_manager, MomentumAppSceneMiscVgm, event.event);
|
||||||
|
consumed = true;
|
||||||
|
switch(event.event) {
|
||||||
|
case VarItemListIndexForeground:
|
||||||
|
case VarItemListIndexBackground:
|
||||||
|
scene_manager_set_scene_state(
|
||||||
|
app->scene_manager,
|
||||||
|
MomentumAppSceneMiscVgmColor,
|
||||||
|
event.event - VarItemListIndexForeground);
|
||||||
|
scene_manager_next_scene(app->scene_manager, MomentumAppSceneMiscVgmColor);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void momentum_app_scene_misc_vgm_on_exit(void* context) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
variable_item_list_reset(app->var_item_list);
|
||||||
|
}
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
#include "../momentum_app.h"
|
||||||
|
|
||||||
|
enum ByteInputResult {
|
||||||
|
ByteInputResultOk,
|
||||||
|
};
|
||||||
|
|
||||||
|
void momentum_app_scene_misc_vgm_color_byte_input_callback(void* context) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
|
||||||
|
view_dispatcher_send_custom_event(app->view_dispatcher, ByteInputResultOk);
|
||||||
|
}
|
||||||
|
|
||||||
|
void momentum_app_scene_misc_vgm_color_on_enter(void* context) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
ByteInput* byte_input = app->byte_input;
|
||||||
|
|
||||||
|
byte_input_set_header_text(byte_input, "Set VGM Color (RGB565)");
|
||||||
|
|
||||||
|
if(scene_manager_get_scene_state(app->scene_manager, MomentumAppSceneMiscVgmColor)) {
|
||||||
|
app->vgm_color = momentum_settings.vgm_color_bg;
|
||||||
|
} else {
|
||||||
|
app->vgm_color = momentum_settings.vgm_color_fg;
|
||||||
|
}
|
||||||
|
app->vgm_color.value = __REVSH(app->vgm_color.value);
|
||||||
|
|
||||||
|
byte_input_set_result_callback(
|
||||||
|
byte_input,
|
||||||
|
momentum_app_scene_misc_vgm_color_byte_input_callback,
|
||||||
|
NULL,
|
||||||
|
app,
|
||||||
|
(void*)&app->vgm_color,
|
||||||
|
sizeof(app->vgm_color));
|
||||||
|
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, MomentumAppViewByteInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool momentum_app_scene_misc_vgm_color_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
|
consumed = true;
|
||||||
|
switch(event.event) {
|
||||||
|
case ByteInputResultOk:
|
||||||
|
app->vgm_color.value = __REVSH(app->vgm_color.value);
|
||||||
|
if(scene_manager_get_scene_state(app->scene_manager, MomentumAppSceneMiscVgmColor)) {
|
||||||
|
momentum_settings.vgm_color_bg = app->vgm_color;
|
||||||
|
} else {
|
||||||
|
momentum_settings.vgm_color_fg = app->vgm_color;
|
||||||
|
}
|
||||||
|
app->save_settings = true;
|
||||||
|
if(momentum_settings.vgm_color_mode == VgmColorModeCustom) {
|
||||||
|
expansion_disable(app->expansion);
|
||||||
|
expansion_enable(app->expansion);
|
||||||
|
}
|
||||||
|
scene_manager_previous_scene(app->scene_manager);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void momentum_app_scene_misc_vgm_color_on_exit(void* context) {
|
||||||
|
MomentumApp* app = context;
|
||||||
|
byte_input_set_result_callback(app->byte_input, NULL, NULL, NULL, NULL, 0);
|
||||||
|
byte_input_set_header_text(app->byte_input, "");
|
||||||
|
}
|
||||||
@@ -73,14 +73,14 @@ static const MfClassicKeyPair troika_4k_keys[] = {
|
|||||||
{.a = 0x7A38E3511A38, .b = 0xAB16584C972A}, //30
|
{.a = 0x7A38E3511A38, .b = 0xAB16584C972A}, //30
|
||||||
{.a = 0x7545DF809202, .b = 0xECF751084A80}, //31
|
{.a = 0x7545DF809202, .b = 0xECF751084A80}, //31
|
||||||
{.a = 0x5125974CD391, .b = 0xD3EAFB5DF46D}, //32
|
{.a = 0x5125974CD391, .b = 0xD3EAFB5DF46D}, //32
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //33
|
{.a = 0x7A86AA203788, .b = 0xE41242278CA2}, //33
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //34
|
{.a = 0xAFCEF64C9913, .b = 0x9DB96DCA4324}, //34
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //35
|
{.a = 0x04EAA462F70B, .b = 0xAC17B93E2FAE}, //35
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //36
|
{.a = 0xE734C210F27E, .b = 0x29BA8C3E9FDA}, //36
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //37
|
{.a = 0xD5524F591EED, .b = 0x5DAF42861B4D}, //37
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //38
|
{.a = 0xE4821A377B75, .b = 0xE8709E486465}, //38
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //39
|
{.a = 0x518DC6EEA089, .b = 0x97C64AC98CA4}, //39
|
||||||
{.a = 0xFFFFFFFFFFFF, .b = 0xFFFFFFFFFFFF}, //40
|
{.a = 0xBB52F8CCE07F, .b = 0x6B6119752C70}, //40
|
||||||
};
|
};
|
||||||
|
|
||||||
static bool troika_get_card_config(TroikaCardConfig* config, MfClassicType type) {
|
static bool troika_get_card_config(TroikaCardConfig* config, MfClassicType type) {
|
||||||
|
|||||||
14
applications/system/findmy/application.fam
Normal file
14
applications/system/findmy/application.fam
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
App(
|
||||||
|
appid="findmy",
|
||||||
|
name="FindMy Flipper",
|
||||||
|
apptype=FlipperAppType.EXTERNAL,
|
||||||
|
entry_point="findmy_main",
|
||||||
|
requires=["gui"],
|
||||||
|
stack_size=1 * 1024,
|
||||||
|
order=35,
|
||||||
|
fap_icon="location_icon.png",
|
||||||
|
fap_category="Bluetooth",
|
||||||
|
fap_author="@MatthewKuKanich",
|
||||||
|
fap_version="1.0",
|
||||||
|
fap_description="BLE FindMy Location Beacon",
|
||||||
|
)
|
||||||
164
applications/system/findmy/findmy.c
Normal file
164
applications/system/findmy/findmy.c
Normal file
@@ -0,0 +1,164 @@
|
|||||||
|
#include "findmy_i.h"
|
||||||
|
|
||||||
|
static bool findmy_custom_event_callback(void* context, uint32_t event) {
|
||||||
|
furi_assert(context);
|
||||||
|
FindMy* app = context;
|
||||||
|
return scene_manager_handle_custom_event(app->scene_manager, event);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool findmy_back_event_callback(void* context) {
|
||||||
|
furi_assert(context);
|
||||||
|
FindMy* app = context;
|
||||||
|
return scene_manager_handle_back_event(app->scene_manager);
|
||||||
|
}
|
||||||
|
|
||||||
|
static FindMy* findmy_app_alloc() {
|
||||||
|
FindMy* app = malloc(sizeof(FindMy));
|
||||||
|
|
||||||
|
app->gui = furi_record_open(RECORD_GUI);
|
||||||
|
|
||||||
|
app->view_dispatcher = view_dispatcher_alloc();
|
||||||
|
view_dispatcher_enable_queue(app->view_dispatcher);
|
||||||
|
|
||||||
|
app->scene_manager = scene_manager_alloc(&findmy_scene_handlers, app);
|
||||||
|
|
||||||
|
view_dispatcher_set_event_callback_context(app->view_dispatcher, app);
|
||||||
|
view_dispatcher_set_custom_event_callback(app->view_dispatcher, findmy_custom_event_callback);
|
||||||
|
view_dispatcher_set_navigation_event_callback(
|
||||||
|
app->view_dispatcher, findmy_back_event_callback);
|
||||||
|
|
||||||
|
app->findmy_main = findmy_main_alloc(app);
|
||||||
|
view_dispatcher_add_view(
|
||||||
|
app->view_dispatcher, FindMyViewMain, findmy_main_get_view(app->findmy_main));
|
||||||
|
|
||||||
|
app->byte_input = byte_input_alloc();
|
||||||
|
view_dispatcher_add_view(
|
||||||
|
app->view_dispatcher, FindMyViewByteInput, byte_input_get_view(app->byte_input));
|
||||||
|
|
||||||
|
app->var_item_list = variable_item_list_alloc();
|
||||||
|
view_dispatcher_add_view(
|
||||||
|
app->view_dispatcher,
|
||||||
|
FindMyViewVarItemList,
|
||||||
|
variable_item_list_get_view(app->var_item_list));
|
||||||
|
|
||||||
|
view_dispatcher_attach_to_gui(app->view_dispatcher, app->gui, ViewDispatcherTypeFullscreen);
|
||||||
|
|
||||||
|
app->beacon_active = false;
|
||||||
|
findmy_main_update_active(app->findmy_main, app->beacon_active);
|
||||||
|
app->broadcast_interval = 5;
|
||||||
|
findmy_main_update_interval(app->findmy_main, app->broadcast_interval);
|
||||||
|
app->transmit_power = 6;
|
||||||
|
app->apple = true;
|
||||||
|
findmy_main_update_apple(app->findmy_main, app->apple);
|
||||||
|
|
||||||
|
return app;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void findmy_app_free(FindMy* app) {
|
||||||
|
furi_assert(app);
|
||||||
|
|
||||||
|
view_dispatcher_remove_view(app->view_dispatcher, FindMyViewVarItemList);
|
||||||
|
variable_item_list_free(app->var_item_list);
|
||||||
|
|
||||||
|
view_dispatcher_remove_view(app->view_dispatcher, FindMyViewByteInput);
|
||||||
|
byte_input_free(app->byte_input);
|
||||||
|
|
||||||
|
view_dispatcher_remove_view(app->view_dispatcher, FindMyViewMain);
|
||||||
|
findmy_main_free(app->findmy_main);
|
||||||
|
|
||||||
|
view_dispatcher_free(app->view_dispatcher);
|
||||||
|
scene_manager_free(app->scene_manager);
|
||||||
|
|
||||||
|
furi_record_close(RECORD_GUI);
|
||||||
|
|
||||||
|
free(app);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void findmy_start(FindMy* app) {
|
||||||
|
furi_hal_bt_extra_beacon_stop(); // Stop any running beacon
|
||||||
|
|
||||||
|
app->config.min_adv_interval_ms = app->broadcast_interval * 1000; // Converting s to ms
|
||||||
|
app->config.max_adv_interval_ms = (app->broadcast_interval * 1000) + 150;
|
||||||
|
app->config.adv_channel_map = GapAdvChannelMapAll;
|
||||||
|
app->config.adv_power_level = GapAdvPowerLevel_0dBm + app->transmit_power;
|
||||||
|
app->config.address_type = GapAddressTypePublic;
|
||||||
|
|
||||||
|
uint8_t mac[EXTRA_BEACON_MAC_ADDR_SIZE] = {0x4D, 0x61, 0x74, 0x4B, 0x75, 0x4B};
|
||||||
|
furi_hal_bt_reverse_mac_addr(mac);
|
||||||
|
memcpy(&app->config.address, mac, sizeof(app->config.address));
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_set_config(&app->config));
|
||||||
|
|
||||||
|
uint8_t data[EXTRA_BEACON_MAX_DATA_SIZE];
|
||||||
|
uint8_t* it = data;
|
||||||
|
|
||||||
|
// For Apple AirTags
|
||||||
|
*it++ = 0x1E; // Length
|
||||||
|
*it++ = 0xFF; // Manufacturer Specific Data
|
||||||
|
*it++ = 0x4C; // Company ID (Apple, Inc.)
|
||||||
|
*it++ = 0x00; // State
|
||||||
|
*it++ = 0x12; // Data - Public Key without the MAC address
|
||||||
|
*it++ = 0x81; // ...
|
||||||
|
*it++ = 0xB9; // ...
|
||||||
|
*it++ = 0x02; // First 2 bits are the version, the rest is the battery level
|
||||||
|
*it++ = 0x7E; // Hint (0x00)
|
||||||
|
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_set_data(data, it - data));
|
||||||
|
}
|
||||||
|
|
||||||
|
int32_t findmy_main(void* p) {
|
||||||
|
UNUSED(p);
|
||||||
|
FindMy* app = findmy_app_alloc();
|
||||||
|
|
||||||
|
findmy_start(app);
|
||||||
|
|
||||||
|
scene_manager_next_scene(app->scene_manager, FindMySceneMain);
|
||||||
|
|
||||||
|
view_dispatcher_run(app->view_dispatcher);
|
||||||
|
|
||||||
|
findmy_app_free(app);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_change_broadcast_interval(FindMy* app, uint8_t value) {
|
||||||
|
if(value > 10 || value < 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
app->broadcast_interval = value;
|
||||||
|
findmy_main_update_interval(app->findmy_main, app->broadcast_interval);
|
||||||
|
if(app->beacon_active) {
|
||||||
|
// Always check if beacon is active before changing config
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_stop());
|
||||||
|
}
|
||||||
|
app->config.min_adv_interval_ms = app->broadcast_interval * 1000;
|
||||||
|
app->config.max_adv_interval_ms = app->config.min_adv_interval_ms + 150;
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_set_config(&app->config));
|
||||||
|
if(app->beacon_active) {
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_start());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_change_transmit_power(FindMy* app, uint8_t value) {
|
||||||
|
if(value > 6) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
app->transmit_power = value;
|
||||||
|
if(app->beacon_active) {
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_stop());
|
||||||
|
}
|
||||||
|
app->config.adv_power_level = GapAdvPowerLevel_0dBm + app->transmit_power;
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_set_config(&app->config));
|
||||||
|
if(app->beacon_active) {
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_start());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_toggle_beacon(FindMy* app) {
|
||||||
|
app->beacon_active = !app->beacon_active;
|
||||||
|
findmy_main_update_active(app->findmy_main, app->beacon_active);
|
||||||
|
findmy_main_update_apple(app->findmy_main, app->apple);
|
||||||
|
if(app->beacon_active) {
|
||||||
|
furi_hal_bt_extra_beacon_start();
|
||||||
|
} else {
|
||||||
|
furi_hal_bt_extra_beacon_stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
3
applications/system/findmy/findmy.h
Normal file
3
applications/system/findmy/findmy.h
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
typedef struct FindMy FindMy;
|
||||||
42
applications/system/findmy/findmy_i.h
Normal file
42
applications/system/findmy/findmy_i.h
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "findmy.h"
|
||||||
|
#include <furi_hal_bt.h>
|
||||||
|
#include <extra_beacon.h>
|
||||||
|
#include <assets_icons.h>
|
||||||
|
#include <gui/gui.h>
|
||||||
|
#include <gui/scene_manager.h>
|
||||||
|
#include <gui/view_dispatcher.h>
|
||||||
|
#include "views/findmy_main.h"
|
||||||
|
#include <gui/modules/byte_input.h>
|
||||||
|
#include <gui/modules/variable_item_list.h>
|
||||||
|
#include "scenes/findmy_scene.h"
|
||||||
|
|
||||||
|
struct FindMy {
|
||||||
|
Gui* gui;
|
||||||
|
SceneManager* scene_manager;
|
||||||
|
ViewDispatcher* view_dispatcher;
|
||||||
|
|
||||||
|
FindMyMain* findmy_main;
|
||||||
|
ByteInput* byte_input;
|
||||||
|
VariableItemList* var_item_list;
|
||||||
|
|
||||||
|
uint8_t mac_buf[EXTRA_BEACON_MAC_ADDR_SIZE];
|
||||||
|
uint8_t packet_buf[EXTRA_BEACON_MAX_DATA_SIZE];
|
||||||
|
|
||||||
|
GapExtraBeaconConfig config;
|
||||||
|
bool apple;
|
||||||
|
bool beacon_active;
|
||||||
|
uint8_t broadcast_interval;
|
||||||
|
uint8_t transmit_power;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
FindMyViewMain,
|
||||||
|
FindMyViewByteInput,
|
||||||
|
FindMyViewVarItemList,
|
||||||
|
} FindMyView;
|
||||||
|
|
||||||
|
void findmy_change_broadcast_interval(FindMy* app, uint8_t value);
|
||||||
|
void findmy_change_transmit_power(FindMy* app, uint8_t value);
|
||||||
|
void findmy_toggle_beacon(FindMy* app);
|
||||||
BIN
applications/system/findmy/location_icon.png
Normal file
BIN
applications/system/findmy/location_icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 6.6 KiB |
31
applications/system/findmy/scenes/findmy_scene.c
Normal file
31
applications/system/findmy/scenes/findmy_scene.c
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
#include "findmy_scene.h"
|
||||||
|
#include "findmy.h"
|
||||||
|
|
||||||
|
// Generate scene on_enter handlers array
|
||||||
|
#define ADD_SCENE(prefix, name, id) prefix##_scene_##name##_on_enter,
|
||||||
|
void (*const findmy_on_enter_handlers[])(void*) = {
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
};
|
||||||
|
#undef ADD_SCENE
|
||||||
|
|
||||||
|
// Generate scene on_event handlers array
|
||||||
|
#define ADD_SCENE(prefix, name, id) prefix##_scene_##name##_on_event,
|
||||||
|
bool (*const findmy_on_event_handlers[])(void* context, SceneManagerEvent event) = {
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
};
|
||||||
|
#undef ADD_SCENE
|
||||||
|
|
||||||
|
// Generate scene on_exit handlers array
|
||||||
|
#define ADD_SCENE(prefix, name, id) prefix##_scene_##name##_on_exit,
|
||||||
|
void (*const findmy_on_exit_handlers[])(void* context) = {
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
};
|
||||||
|
#undef ADD_SCENE
|
||||||
|
|
||||||
|
// Initialize scene handlers configuration structure
|
||||||
|
const SceneManagerHandlers findmy_scene_handlers = {
|
||||||
|
.on_enter_handlers = findmy_on_enter_handlers,
|
||||||
|
.on_event_handlers = findmy_on_event_handlers,
|
||||||
|
.on_exit_handlers = findmy_on_exit_handlers,
|
||||||
|
.scene_num = FindMySceneNum,
|
||||||
|
};
|
||||||
30
applications/system/findmy/scenes/findmy_scene.h
Normal file
30
applications/system/findmy/scenes/findmy_scene.h
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <gui/scene_manager.h>
|
||||||
|
#include "findmy.h"
|
||||||
|
|
||||||
|
// Generate scene id and total number
|
||||||
|
#define ADD_SCENE(prefix, name, id) FindMyScene##id,
|
||||||
|
typedef enum {
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
FindMySceneNum,
|
||||||
|
} FindMyScene;
|
||||||
|
#undef ADD_SCENE
|
||||||
|
|
||||||
|
extern const SceneManagerHandlers findmy_scene_handlers;
|
||||||
|
|
||||||
|
// Generate scene on_enter handlers declaration
|
||||||
|
#define ADD_SCENE(prefix, name, id) void prefix##_scene_##name##_on_enter(void*);
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
#undef ADD_SCENE
|
||||||
|
|
||||||
|
// Generate scene on_event handlers declaration
|
||||||
|
#define ADD_SCENE(prefix, name, id) \
|
||||||
|
bool prefix##_scene_##name##_on_event(void* context, SceneManagerEvent event);
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
#undef ADD_SCENE
|
||||||
|
|
||||||
|
// Generate scene on_exit handlers declaration
|
||||||
|
#define ADD_SCENE(prefix, name, id) void prefix##_scene_##name##_on_exit(void* context);
|
||||||
|
#include "findmy_scenes.h"
|
||||||
|
#undef ADD_SCENE
|
||||||
98
applications/system/findmy/scenes/findmy_scene_config.c
Normal file
98
applications/system/findmy/scenes/findmy_scene_config.c
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
#include "../findmy_i.h"
|
||||||
|
|
||||||
|
enum VarItemListIndex {
|
||||||
|
VarItemListIndexBroadcastInterval,
|
||||||
|
VarItemListIndexTransmitPower,
|
||||||
|
VarItemListIndexRegisterTag,
|
||||||
|
VarItemListIndexAbout,
|
||||||
|
};
|
||||||
|
|
||||||
|
void findmy_scene_config_broadcast_interval_changed(VariableItem* item) {
|
||||||
|
FindMy* app = variable_item_get_context(item);
|
||||||
|
uint8_t index = variable_item_get_current_value_index(item);
|
||||||
|
findmy_change_broadcast_interval(app, index + 1);
|
||||||
|
char str[5];
|
||||||
|
snprintf(str, sizeof(str), "%ds", app->broadcast_interval);
|
||||||
|
variable_item_set_current_value_text(item, str);
|
||||||
|
variable_item_set_current_value_index(item, app->broadcast_interval - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_transmit_power_changed(VariableItem* item) {
|
||||||
|
FindMy* app = variable_item_get_context(item);
|
||||||
|
uint8_t index = variable_item_get_current_value_index(item);
|
||||||
|
findmy_change_transmit_power(app, index);
|
||||||
|
char str[7];
|
||||||
|
snprintf(str, sizeof(str), "%ddBm", app->transmit_power);
|
||||||
|
variable_item_set_current_value_text(item, str);
|
||||||
|
variable_item_set_current_value_index(item, app->transmit_power);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_callback(void* context, uint32_t index) {
|
||||||
|
furi_assert(context);
|
||||||
|
FindMy* app = context;
|
||||||
|
view_dispatcher_send_custom_event(app->view_dispatcher, index);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_on_enter(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
VariableItemList* var_item_list = app->var_item_list;
|
||||||
|
VariableItem* item;
|
||||||
|
|
||||||
|
item = variable_item_list_add(
|
||||||
|
var_item_list,
|
||||||
|
"Broadcast Interval",
|
||||||
|
10,
|
||||||
|
findmy_scene_config_broadcast_interval_changed,
|
||||||
|
app);
|
||||||
|
// Broadcast Interval is 1-10, so use 0-9 and offset indexes by 1
|
||||||
|
variable_item_set_current_value_index(item, app->broadcast_interval - 1);
|
||||||
|
char broadcast_interval_s[5];
|
||||||
|
snprintf(broadcast_interval_s, sizeof(broadcast_interval_s), "%ds", app->broadcast_interval);
|
||||||
|
variable_item_set_current_value_text(item, broadcast_interval_s);
|
||||||
|
|
||||||
|
item = variable_item_list_add(
|
||||||
|
var_item_list, "Transmit Power", 7, findmy_scene_config_transmit_power_changed, app);
|
||||||
|
variable_item_set_current_value_index(item, app->transmit_power);
|
||||||
|
char transmit_power_s[7];
|
||||||
|
snprintf(transmit_power_s, sizeof(transmit_power_s), "%ddBm", app->transmit_power);
|
||||||
|
variable_item_set_current_value_text(item, transmit_power_s);
|
||||||
|
|
||||||
|
item = variable_item_list_add(var_item_list, "Register Tag", 0, NULL, NULL);
|
||||||
|
item = variable_item_list_add(var_item_list, "Matthew KuKanich, Thanks to Chapoly1305, WillyJL, OpenHaystack, Testers", 1, NULL, NULL);
|
||||||
|
variable_item_set_current_value_text(item, "Credits");
|
||||||
|
|
||||||
|
variable_item_list_set_enter_callback(var_item_list, findmy_scene_config_callback, app);
|
||||||
|
|
||||||
|
variable_item_list_set_selected_item(
|
||||||
|
var_item_list, scene_manager_get_scene_state(app->scene_manager, FindMySceneConfig));
|
||||||
|
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, FindMyViewVarItemList);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool findmy_scene_config_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
FindMy* app = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
|
scene_manager_set_scene_state(app->scene_manager, FindMySceneConfig, event.event);
|
||||||
|
consumed = true;
|
||||||
|
switch(event.event) {
|
||||||
|
case VarItemListIndexRegisterTag:
|
||||||
|
scene_manager_next_scene(app->scene_manager, FindMySceneConfigMac);
|
||||||
|
break;
|
||||||
|
case VarItemListIndexAbout:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_on_exit(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
VariableItemList* var_item_list = app->var_item_list;
|
||||||
|
|
||||||
|
variable_item_list_reset(var_item_list);
|
||||||
|
}
|
||||||
60
applications/system/findmy/scenes/findmy_scene_config_mac.c
Normal file
60
applications/system/findmy/scenes/findmy_scene_config_mac.c
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
#include "../findmy_i.h"
|
||||||
|
|
||||||
|
enum ByteInputResult {
|
||||||
|
ByteInputResultOk,
|
||||||
|
};
|
||||||
|
|
||||||
|
static void findmy_scene_config_mac_callback(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
|
||||||
|
view_dispatcher_send_custom_event(app->view_dispatcher, ByteInputResultOk);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_mac_on_enter(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
ByteInput* byte_input = app->byte_input;
|
||||||
|
|
||||||
|
byte_input_set_header_text(byte_input, "Enter Bluetooth MAC:");
|
||||||
|
|
||||||
|
memcpy(app->mac_buf, &app->config.address, sizeof(app->mac_buf));
|
||||||
|
furi_hal_bt_reverse_mac_addr(app->mac_buf);
|
||||||
|
|
||||||
|
byte_input_set_result_callback(
|
||||||
|
byte_input,
|
||||||
|
findmy_scene_config_mac_callback,
|
||||||
|
NULL,
|
||||||
|
app,
|
||||||
|
app->mac_buf,
|
||||||
|
sizeof(app->mac_buf));
|
||||||
|
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, FindMyViewByteInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool findmy_scene_config_mac_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
FindMy* app = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
|
consumed = true;
|
||||||
|
switch(event.event) {
|
||||||
|
case ByteInputResultOk:
|
||||||
|
furi_hal_bt_reverse_mac_addr(app->mac_buf);
|
||||||
|
memcpy(&app->config.address, app->mac_buf, sizeof(app->config.address));
|
||||||
|
furi_hal_bt_extra_beacon_set_config(&app->config);
|
||||||
|
scene_manager_next_scene(app->scene_manager, FindMySceneConfigPacket);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_mac_on_exit(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
|
||||||
|
byte_input_set_result_callback(app->byte_input, NULL, NULL, NULL, NULL, 0);
|
||||||
|
byte_input_set_header_text(app->byte_input, "");
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,64 @@
|
|||||||
|
#include "../findmy_i.h"
|
||||||
|
|
||||||
|
enum ByteInputResult {
|
||||||
|
ByteInputResultOk,
|
||||||
|
};
|
||||||
|
|
||||||
|
static void findmy_scene_config_packet_callback(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
|
||||||
|
view_dispatcher_send_custom_event(app->view_dispatcher, ByteInputResultOk);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_packet_on_enter(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
ByteInput* byte_input = app->byte_input;
|
||||||
|
|
||||||
|
byte_input_set_header_text(byte_input, "Enter Bluetooth Payload:");
|
||||||
|
|
||||||
|
memset(app->packet_buf, 0, sizeof(app->packet_buf));
|
||||||
|
furi_hal_bt_extra_beacon_get_data(app->packet_buf);
|
||||||
|
|
||||||
|
byte_input_set_result_callback(
|
||||||
|
byte_input,
|
||||||
|
findmy_scene_config_packet_callback,
|
||||||
|
NULL,
|
||||||
|
app,
|
||||||
|
app->packet_buf,
|
||||||
|
sizeof(app->packet_buf));
|
||||||
|
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, FindMyViewByteInput);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool findmy_scene_config_packet_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
FindMy* app = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
|
consumed = true;
|
||||||
|
switch(event.event) {
|
||||||
|
case ByteInputResultOk:
|
||||||
|
scene_manager_search_and_switch_to_previous_scene(
|
||||||
|
app->scene_manager, FindMySceneConfig);
|
||||||
|
furi_check(furi_hal_bt_extra_beacon_set_data(app->packet_buf, sizeof(app->packet_buf)));
|
||||||
|
if (app->packet_buf[0] == 0x1E && app->packet_buf[3] == 0x00) {
|
||||||
|
app->apple = true; // Checks payload data for Apple identifier
|
||||||
|
} else {
|
||||||
|
app->apple = false;
|
||||||
|
}
|
||||||
|
findmy_main_update_apple(app->findmy_main, app->apple);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_config_packet_on_exit(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
|
||||||
|
byte_input_set_result_callback(app->byte_input, NULL, NULL, NULL, NULL, 0);
|
||||||
|
byte_input_set_header_text(app->byte_input, "");
|
||||||
|
}
|
||||||
54
applications/system/findmy/scenes/findmy_scene_main.c
Normal file
54
applications/system/findmy/scenes/findmy_scene_main.c
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
#include "../findmy_i.h"
|
||||||
|
|
||||||
|
void findmy_scene_main_callback(FindMyMainEvent event, void* context) {
|
||||||
|
furi_assert(context);
|
||||||
|
FindMy* app = context;
|
||||||
|
view_dispatcher_send_custom_event(app->view_dispatcher, event);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_main_on_enter(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
|
||||||
|
findmy_main_set_callback(app->findmy_main, findmy_scene_main_callback, app);
|
||||||
|
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, FindMyViewMain);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool findmy_scene_main_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
FindMy* app = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
|
consumed = true;
|
||||||
|
switch(event.event) {
|
||||||
|
case FindMyMainEventToggle:
|
||||||
|
findmy_toggle_beacon(app);
|
||||||
|
break;
|
||||||
|
case FindMyMainEventBackground:
|
||||||
|
furi_hal_bt_extra_beacon_start();
|
||||||
|
view_dispatcher_stop(app->view_dispatcher);
|
||||||
|
break;
|
||||||
|
case FindMyMainEventConfig:
|
||||||
|
scene_manager_next_scene(app->scene_manager, FindMySceneConfig);
|
||||||
|
break;
|
||||||
|
case FindMyMainEventIntervalUp:
|
||||||
|
findmy_change_broadcast_interval(app, app->broadcast_interval + 1);
|
||||||
|
break;
|
||||||
|
case FindMyMainEventIntervalDown:
|
||||||
|
findmy_change_broadcast_interval(app, app->broadcast_interval - 1);
|
||||||
|
break;
|
||||||
|
case FindMyMainEventQuit:
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
consumed = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_scene_main_on_exit(void* context) {
|
||||||
|
FindMy* app = context;
|
||||||
|
UNUSED(app);
|
||||||
|
}
|
||||||
4
applications/system/findmy/scenes/findmy_scenes.h
Normal file
4
applications/system/findmy/scenes/findmy_scenes.h
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
ADD_SCENE(findmy, main, Main)
|
||||||
|
ADD_SCENE(findmy, config, Config)
|
||||||
|
ADD_SCENE(findmy, config_mac, ConfigMac)
|
||||||
|
ADD_SCENE(findmy, config_packet, ConfigPacket)
|
||||||
143
applications/system/findmy/views/findmy_main.c
Normal file
143
applications/system/findmy/views/findmy_main.c
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
#include "findmy_main.h"
|
||||||
|
#include "../findmy_i.h"
|
||||||
|
|
||||||
|
struct FindMyMain {
|
||||||
|
View* view;
|
||||||
|
FindMyMainCallback callback;
|
||||||
|
void* context;
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
bool active;
|
||||||
|
bool apple;
|
||||||
|
uint8_t interval;
|
||||||
|
} FindMyMainModel;
|
||||||
|
|
||||||
|
static void findmy_main_draw_callback(Canvas* canvas, void* _model) {
|
||||||
|
FindMyMainModel* model = _model;
|
||||||
|
canvas_clear(canvas);
|
||||||
|
canvas_set_bitmap_mode(canvas, true);
|
||||||
|
canvas_set_font(canvas, FontPrimary);
|
||||||
|
|
||||||
|
canvas_draw_str(canvas, 4, 11, "FindMy Flipper");
|
||||||
|
canvas_set_font(canvas, FontSecondary);
|
||||||
|
if(model->active) {
|
||||||
|
canvas_draw_str(canvas, 4, 49, "Broadcast Active");
|
||||||
|
canvas_draw_icon(canvas, 78, 42, &I_Ok_btn_9x9);
|
||||||
|
} else {
|
||||||
|
canvas_draw_str(canvas, 4, 49, "Broadcast Inactive");
|
||||||
|
}
|
||||||
|
canvas_set_font(canvas, FontSecondary);
|
||||||
|
canvas_draw_str(canvas, 4, 21, "Press <- to run in background");
|
||||||
|
canvas_set_font(canvas, FontSecondary);
|
||||||
|
char interval_str[20];
|
||||||
|
snprintf(interval_str, sizeof(interval_str), "Ping Interval: %ds", model->interval);
|
||||||
|
canvas_draw_str(canvas, 4, 62, interval_str);
|
||||||
|
canvas_set_font(canvas, FontPrimary);
|
||||||
|
if(model->apple){
|
||||||
|
canvas_draw_str(canvas, 4, 32, "Apple Network");
|
||||||
|
canvas_draw_icon(canvas, 80, 24, &I_Lock_7x8);
|
||||||
|
} else {
|
||||||
|
canvas_draw_str(canvas, 4, 32, "Samsung Network");
|
||||||
|
canvas_draw_icon(canvas, 97, 24, &I_Lock_7x8);
|
||||||
|
}
|
||||||
|
canvas_set_font(canvas, FontSecondary);
|
||||||
|
canvas_draw_str(canvas, 100, 61, "Config");
|
||||||
|
canvas_draw_line(canvas, 100, 51, 127, 51);
|
||||||
|
canvas_draw_line(canvas, 97, 53, 97, 63);
|
||||||
|
canvas_draw_line(canvas, 97, 53, 99, 51);
|
||||||
|
canvas_draw_line(canvas, 3, 52, 87, 52);
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool findmy_main_input_callback(InputEvent* event, void* context) {
|
||||||
|
furi_assert(context);
|
||||||
|
FindMyMain* findmy_main = context;
|
||||||
|
bool consumed = false;
|
||||||
|
|
||||||
|
if(event->type == InputTypePress) {
|
||||||
|
consumed = true;
|
||||||
|
// FIXME: finish implementing handlers in scene side
|
||||||
|
switch(event->key) {
|
||||||
|
case InputKeyBack:
|
||||||
|
findmy_main->callback(FindMyMainEventQuit, findmy_main->context);
|
||||||
|
// furi_hal_bt_extra_beacon_stop();
|
||||||
|
break;
|
||||||
|
case InputKeyOk:
|
||||||
|
findmy_main->callback(FindMyMainEventToggle, findmy_main->context);
|
||||||
|
break;
|
||||||
|
case InputKeyLeft:
|
||||||
|
findmy_main->callback(FindMyMainEventBackground, findmy_main->context);
|
||||||
|
break;
|
||||||
|
case InputKeyRight:
|
||||||
|
findmy_main->callback(FindMyMainEventConfig, findmy_main->context);
|
||||||
|
break;
|
||||||
|
case InputKeyUp:
|
||||||
|
findmy_main->callback(FindMyMainEventIntervalUp, findmy_main->context);
|
||||||
|
break;
|
||||||
|
case InputKeyDown:
|
||||||
|
findmy_main->callback(FindMyMainEventIntervalDown, findmy_main->context);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return consumed;
|
||||||
|
}
|
||||||
|
|
||||||
|
FindMyMain* findmy_main_alloc(FindMy* app) {
|
||||||
|
FindMyMain* findmy_main = malloc(sizeof(FindMyMain));
|
||||||
|
|
||||||
|
findmy_main->view = view_alloc();
|
||||||
|
view_allocate_model(findmy_main->view, ViewModelTypeLocking, sizeof(FindMyMainModel));
|
||||||
|
with_view_model(
|
||||||
|
findmy_main->view,
|
||||||
|
FindMyMainModel * model,
|
||||||
|
{
|
||||||
|
model->active = app->beacon_active;
|
||||||
|
model->apple = app->apple;
|
||||||
|
model->interval = app->broadcast_interval;
|
||||||
|
},
|
||||||
|
false);
|
||||||
|
view_set_context(findmy_main->view, findmy_main);
|
||||||
|
view_set_draw_callback(findmy_main->view, findmy_main_draw_callback);
|
||||||
|
view_set_input_callback(findmy_main->view, findmy_main_input_callback);
|
||||||
|
|
||||||
|
return findmy_main;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_main_free(FindMyMain* findmy_main) {
|
||||||
|
furi_assert(findmy_main);
|
||||||
|
view_free(findmy_main->view);
|
||||||
|
free(findmy_main);
|
||||||
|
}
|
||||||
|
|
||||||
|
View* findmy_main_get_view(FindMyMain* findmy_main) {
|
||||||
|
furi_assert(findmy_main);
|
||||||
|
return findmy_main->view;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_main_set_callback(FindMyMain* findmy_main, FindMyMainCallback callback, void* context) {
|
||||||
|
furi_assert(findmy_main);
|
||||||
|
furi_assert(callback);
|
||||||
|
findmy_main->callback = callback;
|
||||||
|
findmy_main->context = context;
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_main_update_active(FindMyMain* findmy_main, bool active) {
|
||||||
|
furi_assert(findmy_main);
|
||||||
|
with_view_model(
|
||||||
|
findmy_main->view, FindMyMainModel * model, { model->active = active; }, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_main_update_interval(FindMyMain* findmy_main, uint8_t interval) {
|
||||||
|
furi_assert(findmy_main);
|
||||||
|
with_view_model(
|
||||||
|
findmy_main->view, FindMyMainModel * model, { model->interval = interval; }, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
void findmy_main_update_apple(FindMyMain* findmy_main, bool apple) {
|
||||||
|
furi_assert(findmy_main);
|
||||||
|
with_view_model(
|
||||||
|
findmy_main->view, FindMyMainModel * model, { model->apple = apple; }, true);
|
||||||
|
}
|
||||||
29
applications/system/findmy/views/findmy_main.h
Normal file
29
applications/system/findmy/views/findmy_main.h
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "../findmy.h"
|
||||||
|
#include <gui/view.h>
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
FindMyMainEventToggle,
|
||||||
|
FindMyMainEventBackground,
|
||||||
|
FindMyMainEventConfig,
|
||||||
|
FindMyMainEventIntervalUp,
|
||||||
|
FindMyMainEventIntervalDown,
|
||||||
|
FindMyMainEventQuit,
|
||||||
|
} FindMyMainEvent;
|
||||||
|
|
||||||
|
typedef struct FindMyMain FindMyMain;
|
||||||
|
typedef void (*FindMyMainCallback)(FindMyMainEvent event, void* context);
|
||||||
|
|
||||||
|
// Main functionality
|
||||||
|
FindMyMain* findmy_main_alloc(FindMy* app);
|
||||||
|
void findmy_main_free(FindMyMain* findmy_main);
|
||||||
|
View* findmy_main_get_view(FindMyMain* findmy_main);
|
||||||
|
|
||||||
|
// To communicate with scene
|
||||||
|
void findmy_main_set_callback(FindMyMain* findmy_main, FindMyMainCallback callback, void* context);
|
||||||
|
|
||||||
|
// To redraw when info changes
|
||||||
|
void findmy_main_update_active(FindMyMain* findmy_main, bool active);
|
||||||
|
void findmy_main_update_interval(FindMyMain* findmy_main, uint8_t interval);
|
||||||
|
void findmy_main_update_apple(FindMyMain* findmy_main, bool apple);
|
||||||
@@ -176,7 +176,10 @@ static void js_blebeacon_stop(struct mjs* mjs) {
|
|||||||
blebeacon->saved_prev_active = true;
|
blebeacon->saved_prev_active = true;
|
||||||
blebeacon->prev_active = furi_hal_bt_extra_beacon_is_active();
|
blebeacon->prev_active = furi_hal_bt_extra_beacon_is_active();
|
||||||
}
|
}
|
||||||
furi_hal_bt_extra_beacon_stop();
|
if(!furi_hal_bt_extra_beacon_stop()) {
|
||||||
|
ret_int_err(mjs, "Failed stopping beacon");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
mjs_return(mjs, MJS_UNDEFINED);
|
mjs_return(mjs, MJS_UNDEFINED);
|
||||||
}
|
}
|
||||||
@@ -209,7 +212,7 @@ static void js_blebeacon_destroy(void* inst) {
|
|||||||
JsBlebeaconInst* blebeacon = inst;
|
JsBlebeaconInst* blebeacon = inst;
|
||||||
if(!blebeacon->keep_alive) {
|
if(!blebeacon->keep_alive) {
|
||||||
if(furi_hal_bt_extra_beacon_is_active()) {
|
if(furi_hal_bt_extra_beacon_is_active()) {
|
||||||
furi_hal_bt_extra_beacon_stop();
|
furi_check(furi_hal_bt_extra_beacon_stop());
|
||||||
}
|
}
|
||||||
if(blebeacon->saved_prev_cfg && blebeacon->prev_cfg_set) {
|
if(blebeacon->saved_prev_cfg && blebeacon->prev_cfg_set) {
|
||||||
furi_check(furi_hal_bt_extra_beacon_set_config(&blebeacon->prev_cfg));
|
furi_check(furi_hal_bt_extra_beacon_set_config(&blebeacon->prev_cfg));
|
||||||
|
|||||||
@@ -44,6 +44,7 @@ static void flipper_print_version(const char* target, const Version* version) {
|
|||||||
#include <applications/main/archive/helpers/archive_favorites.h>
|
#include <applications/main/archive/helpers/archive_favorites.h>
|
||||||
#include <momentum/namespoof.h>
|
#include <momentum/namespoof.h>
|
||||||
#include <momentum/momentum.h>
|
#include <momentum/momentum.h>
|
||||||
|
#include <assets_icons.h>
|
||||||
|
|
||||||
void flipper_migrate_files() {
|
void flipper_migrate_files() {
|
||||||
Storage* storage = furi_record_open(RECORD_STORAGE);
|
Storage* storage = furi_record_open(RECORD_STORAGE);
|
||||||
@@ -101,9 +102,8 @@ void flipper_migrate_files() {
|
|||||||
static void flipper_boot_status(Canvas* canvas, const char* text) {
|
static void flipper_boot_status(Canvas* canvas, const char* text) {
|
||||||
FURI_LOG_I(TAG, text);
|
FURI_LOG_I(TAG, text);
|
||||||
canvas_reset(canvas);
|
canvas_reset(canvas);
|
||||||
canvas_draw_str_aligned(canvas, 64, 40, AlignCenter, AlignCenter, text);
|
canvas_draw_icon(canvas, 33, 16, &I_Updating_Logo_62x15);
|
||||||
canvas_set_font(canvas, FontPrimary);
|
canvas_draw_str_aligned(canvas, 64, 44, AlignCenter, AlignCenter, text);
|
||||||
canvas_draw_str_aligned(canvas, 64, 24, AlignCenter, AlignCenter, "Momentum is Booting");
|
|
||||||
canvas_commit(canvas);
|
canvas_commit(canvas);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
@@ -143,7 +143,7 @@ void flipper_init() {
|
|||||||
flipper_boot_status(canvas, "Starting Namespoof");
|
flipper_boot_status(canvas, "Starting Namespoof");
|
||||||
namespoof_init();
|
namespoof_init();
|
||||||
|
|
||||||
flipper_boot_status(canvas, "Loading Momentum Settings");
|
flipper_boot_status(canvas, "Loading Settings");
|
||||||
momentum_settings_load();
|
momentum_settings_load();
|
||||||
furi_hal_light_sequence("rgb RB");
|
furi_hal_light_sequence("rgb RB");
|
||||||
|
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <furi_hal_serial_types.h>
|
#include <furi_hal_serial_types.h>
|
||||||
|
#include <toolbox/colors.h>
|
||||||
#include <gui/canvas.h>
|
#include <gui/canvas.h>
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
@@ -43,10 +44,12 @@ typedef enum {
|
|||||||
SpiCount,
|
SpiCount,
|
||||||
} SpiHandle;
|
} SpiHandle;
|
||||||
|
|
||||||
_Static_assert(sizeof(MenuStyle) == sizeof(uint8_t), "enum too big, fix load/save");
|
typedef enum {
|
||||||
_Static_assert(sizeof(BatteryIcon) == sizeof(uint8_t), "enum too big, fix load/save");
|
VgmColorModeDefault,
|
||||||
_Static_assert(sizeof(SpiHandle) == sizeof(uint8_t), "enum too big, fix load/save");
|
VgmColorModeCustom,
|
||||||
_Static_assert(sizeof(FuriHalSerialId) == sizeof(uint8_t), "enum too big, fix load/save");
|
VgmColorModeRgbBacklight,
|
||||||
|
VgmColorModeCount,
|
||||||
|
} VgmColorMode;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
char asset_pack[ASSET_PACKS_NAME_LEN];
|
char asset_pack[ASSET_PACKS_NAME_LEN];
|
||||||
@@ -84,6 +87,9 @@ typedef struct {
|
|||||||
FuriHalSerialId uart_esp_channel;
|
FuriHalSerialId uart_esp_channel;
|
||||||
FuriHalSerialId uart_nmea_channel;
|
FuriHalSerialId uart_nmea_channel;
|
||||||
bool file_naming_prefix_after;
|
bool file_naming_prefix_after;
|
||||||
|
VgmColorMode vgm_color_mode;
|
||||||
|
Rgb565Color vgm_color_fg;
|
||||||
|
Rgb565Color vgm_color_bg;
|
||||||
} MomentumSettings;
|
} MomentumSettings;
|
||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
|
|||||||
@@ -41,13 +41,15 @@ MomentumSettings momentum_settings = {
|
|||||||
.uart_esp_channel = FuriHalSerialIdUsart, // pin 13,14
|
.uart_esp_channel = FuriHalSerialIdUsart, // pin 13,14
|
||||||
.uart_nmea_channel = FuriHalSerialIdUsart, // pin 13,14
|
.uart_nmea_channel = FuriHalSerialIdUsart, // pin 13,14
|
||||||
.file_naming_prefix_after = false, // Before
|
.file_naming_prefix_after = false, // Before
|
||||||
|
.vgm_color_mode = VgmColorModeDefault, // Default
|
||||||
|
.vgm_color_fg.value = 0x0000, // Default Black
|
||||||
|
.vgm_color_bg.value = 0xFC00, // Default Orange
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
momentum_settings_type_str,
|
momentum_settings_type_str,
|
||||||
momentum_settings_type_int,
|
momentum_settings_type_int,
|
||||||
momentum_settings_type_uint,
|
momentum_settings_type_uint,
|
||||||
momentum_settings_type_enum,
|
|
||||||
momentum_settings_type_bool,
|
momentum_settings_type_bool,
|
||||||
} momentum_settings_type;
|
} momentum_settings_type;
|
||||||
|
|
||||||
@@ -60,19 +62,20 @@ static const struct {
|
|||||||
struct {
|
struct {
|
||||||
int32_t i_min;
|
int32_t i_min;
|
||||||
int32_t i_max;
|
int32_t i_max;
|
||||||
|
uint8_t i_sz;
|
||||||
};
|
};
|
||||||
struct {
|
struct {
|
||||||
uint32_t u_min;
|
uint32_t u_min;
|
||||||
uint32_t u_max;
|
uint32_t u_max;
|
||||||
|
uint8_t u_sz;
|
||||||
};
|
};
|
||||||
uint8_t e_cnt;
|
|
||||||
};
|
};
|
||||||
#define clamp(t, min, max) .t##_min = min, .t##_max = max
|
|
||||||
#define setting(t, n) .type = momentum_settings_type##t, .key = #n, .val = &momentum_settings.n
|
#define setting(t, n) .type = momentum_settings_type##t, .key = #n, .val = &momentum_settings.n
|
||||||
#define setting_str(n) setting(_str, n), .str_len = sizeof(momentum_settings.n)
|
#define setting_str(n) setting(_str, n), .str_len = sizeof(momentum_settings.n)
|
||||||
#define setting_int(n, min, max) setting(_int, n), clamp(i, min, max)
|
#define num(t, n, min, max) .t##_min = min, .t##_max = max, .t##_sz = sizeof(momentum_settings.n)
|
||||||
#define setting_uint(n, min, max) setting(_uint, n), clamp(u, min, max)
|
#define setting_int(n, min, max) setting(_int, n), num(i, n, min, max)
|
||||||
#define setting_enum(n, cnt) setting(_enum, n), .e_cnt = cnt
|
#define setting_uint(n, min, max) setting(_uint, n), num(u, n, min, max)
|
||||||
|
#define setting_enum(n, cnt) setting_uint(n, 0, cnt - 1)
|
||||||
#define setting_bool(n) setting(_bool, n)
|
#define setting_bool(n) setting(_bool, n)
|
||||||
} momentum_settings_entries[] = {
|
} momentum_settings_entries[] = {
|
||||||
{setting_str(asset_pack)},
|
{setting_str(asset_pack)},
|
||||||
@@ -110,6 +113,9 @@ static const struct {
|
|||||||
{setting_enum(uart_esp_channel, FuriHalSerialIdMax)},
|
{setting_enum(uart_esp_channel, FuriHalSerialIdMax)},
|
||||||
{setting_enum(uart_nmea_channel, FuriHalSerialIdMax)},
|
{setting_enum(uart_nmea_channel, FuriHalSerialIdMax)},
|
||||||
{setting_bool(file_naming_prefix_after)},
|
{setting_bool(file_naming_prefix_after)},
|
||||||
|
{setting_enum(vgm_color_mode, VgmColorModeCount)},
|
||||||
|
{setting_uint(vgm_color_fg, 0x0000, 0xFFFF)},
|
||||||
|
{setting_uint(vgm_color_bg, 0x0000, 0xFFFF)},
|
||||||
};
|
};
|
||||||
|
|
||||||
void momentum_settings_load() {
|
void momentum_settings_load() {
|
||||||
@@ -131,15 +137,13 @@ void momentum_settings_load() {
|
|||||||
break;
|
break;
|
||||||
case momentum_settings_type_int:
|
case momentum_settings_type_int:
|
||||||
ok = flipper_format_read_int32(file, entry.key, &val_int, 1);
|
ok = flipper_format_read_int32(file, entry.key, &val_int, 1);
|
||||||
if(ok) *(int32_t*)entry.val = CLAMP(val_int, entry.i_max, entry.i_min);
|
val_int = CLAMP(val_int, entry.i_max, entry.i_min);
|
||||||
|
if(ok) memcpy(entry.val, &val_int, entry.i_sz);
|
||||||
break;
|
break;
|
||||||
case momentum_settings_type_uint:
|
case momentum_settings_type_uint:
|
||||||
ok = flipper_format_read_uint32(file, entry.key, &val_uint, 1);
|
ok = flipper_format_read_uint32(file, entry.key, &val_uint, 1);
|
||||||
if(ok) *(uint32_t*)entry.val = CLAMP(val_uint, entry.u_max, entry.u_min);
|
val_uint = CLAMP(val_uint, entry.u_max, entry.u_min);
|
||||||
break;
|
if(ok) memcpy(entry.val, &val_uint, entry.u_sz);
|
||||||
case momentum_settings_type_enum:
|
|
||||||
ok = flipper_format_read_uint32(file, entry.key, &val_uint, 1);
|
|
||||||
if(ok) *(uint8_t*)entry.val = CLAMP(val_uint, entry.e_cnt - 1U, 0U);
|
|
||||||
break;
|
break;
|
||||||
case momentum_settings_type_bool:
|
case momentum_settings_type_bool:
|
||||||
ok = flipper_format_read_bool(file, entry.key, &val_bool, 1);
|
ok = flipper_format_read_bool(file, entry.key, &val_bool, 1);
|
||||||
@@ -164,7 +168,8 @@ void momentum_settings_save() {
|
|||||||
FlipperFormat* file = flipper_format_file_alloc(storage);
|
FlipperFormat* file = flipper_format_file_alloc(storage);
|
||||||
|
|
||||||
if(flipper_format_file_open_always(file, MOMENTUM_SETTINGS_PATH)) {
|
if(flipper_format_file_open_always(file, MOMENTUM_SETTINGS_PATH)) {
|
||||||
uint32_t tmp_enum;
|
int32_t tmp_int;
|
||||||
|
uint32_t tmp_uint;
|
||||||
for(size_t entry_i = 0; entry_i < COUNT_OF(momentum_settings_entries); entry_i++) {
|
for(size_t entry_i = 0; entry_i < COUNT_OF(momentum_settings_entries); entry_i++) {
|
||||||
#define entry momentum_settings_entries[entry_i]
|
#define entry momentum_settings_entries[entry_i]
|
||||||
switch(entry.type) {
|
switch(entry.type) {
|
||||||
@@ -172,14 +177,14 @@ void momentum_settings_save() {
|
|||||||
flipper_format_write_string_cstr(file, entry.key, (char*)entry.val);
|
flipper_format_write_string_cstr(file, entry.key, (char*)entry.val);
|
||||||
break;
|
break;
|
||||||
case momentum_settings_type_int:
|
case momentum_settings_type_int:
|
||||||
flipper_format_write_int32(file, entry.key, (int32_t*)entry.val, 1);
|
tmp_int = 0;
|
||||||
|
memcpy(&tmp_int, entry.val, entry.i_sz);
|
||||||
|
flipper_format_write_int32(file, entry.key, &tmp_int, 1);
|
||||||
break;
|
break;
|
||||||
case momentum_settings_type_uint:
|
case momentum_settings_type_uint:
|
||||||
flipper_format_write_uint32(file, entry.key, (uint32_t*)entry.val, 1);
|
tmp_uint = 0;
|
||||||
break;
|
memcpy(&tmp_uint, entry.val, entry.u_sz);
|
||||||
case momentum_settings_type_enum:
|
flipper_format_write_uint32(file, entry.key, &tmp_uint, 1);
|
||||||
tmp_enum = *(uint8_t*)entry.val;
|
|
||||||
flipper_format_write_uint32(file, entry.key, &tmp_enum, 1);
|
|
||||||
break;
|
break;
|
||||||
case momentum_settings_type_bool:
|
case momentum_settings_type_bool:
|
||||||
flipper_format_write_bool(file, entry.key, (bool*)entry.val, 1);
|
flipper_format_write_bool(file, entry.key, (bool*)entry.val, 1);
|
||||||
|
|||||||
@@ -86,3 +86,7 @@ void rgb2hsv(const RgbColor* rgb, HsvColor* hsv) {
|
|||||||
hsv->h = 171 + 43 * (rgb->r - rgb->g) / (rgbMax - rgbMin);
|
hsv->h = 171 + 43 * (rgb->r - rgb->g) / (rgbMax - rgbMin);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline int rgb565cmp(const Rgb565Color* a, const Rgb565Color* b) {
|
||||||
|
return memcmp(a, b, sizeof(Rgb565Color));
|
||||||
|
}
|
||||||
|
|||||||
@@ -25,6 +25,17 @@ int hsvcmp(const HsvColor* a, const HsvColor* b);
|
|||||||
void hsv2rgb(const HsvColor* hsv, RgbColor* rgb);
|
void hsv2rgb(const HsvColor* hsv, RgbColor* rgb);
|
||||||
void rgb2hsv(const RgbColor* rgb, HsvColor* hsv);
|
void rgb2hsv(const RgbColor* rgb, HsvColor* hsv);
|
||||||
|
|
||||||
|
typedef union {
|
||||||
|
uint16_t value;
|
||||||
|
struct {
|
||||||
|
uint16_t r : 5;
|
||||||
|
uint16_t g : 6;
|
||||||
|
uint16_t b : 5;
|
||||||
|
} FURI_PACKED;
|
||||||
|
} Rgb565Color;
|
||||||
|
|
||||||
|
int rgb565cmp(const Rgb565Color* a, const Rgb565Color* b);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -2958,6 +2958,7 @@ Function,-,rename,int,"const char*, const char*"
|
|||||||
Function,-,renameat,int,"int, const char*, int, const char*"
|
Function,-,renameat,int,"int, const char*, int, const char*"
|
||||||
Function,-,rewind,void,FILE*
|
Function,-,rewind,void,FILE*
|
||||||
Function,+,rgb2hsv,void,"const RgbColor*, HsvColor*"
|
Function,+,rgb2hsv,void,"const RgbColor*, HsvColor*"
|
||||||
|
Function,+,rgb565cmp,int,"const Rgb565Color*, const Rgb565Color*"
|
||||||
Function,+,rgb_backlight_get_color,void,"uint8_t, RgbColor*"
|
Function,+,rgb_backlight_get_color,void,"uint8_t, RgbColor*"
|
||||||
Function,+,rgb_backlight_get_rainbow_interval,uint32_t,
|
Function,+,rgb_backlight_get_rainbow_interval,uint32_t,
|
||||||
Function,+,rgb_backlight_get_rainbow_mode,RGBBacklightRainbowMode,
|
Function,+,rgb_backlight_get_rainbow_mode,RGBBacklightRainbowMode,
|
||||||
|
|||||||
|
@@ -9,6 +9,12 @@
|
|||||||
#include <furi.h>
|
#include <furi.h>
|
||||||
#include <protobuf_version.h>
|
#include <protobuf_version.h>
|
||||||
|
|
||||||
|
#include <momentum/momentum.h>
|
||||||
|
#include <rgb_backlight.h>
|
||||||
|
#include <SK6805.h>
|
||||||
|
#include <colors.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
|
||||||
FURI_WEAK void furi_hal_info_get_api_version(uint16_t* major, uint16_t* minor) {
|
FURI_WEAK void furi_hal_info_get_api_version(uint16_t* major, uint16_t* minor) {
|
||||||
*major = 0;
|
*major = 0;
|
||||||
*minor = 0;
|
*minor = 0;
|
||||||
@@ -64,6 +70,57 @@ void furi_hal_info_get(PropertyValueCallback out, char sep, void* context) {
|
|||||||
property_value_out(
|
property_value_out(
|
||||||
&property_context, "%d", 2, "hardware", "color", furi_hal_version_get_hw_color());
|
&property_context, "%d", 2, "hardware", "color", furi_hal_version_get_hw_color());
|
||||||
|
|
||||||
|
// RGB Settings
|
||||||
|
property_value_out(
|
||||||
|
&property_context,
|
||||||
|
"%d",
|
||||||
|
4,
|
||||||
|
"hardware",
|
||||||
|
"screen",
|
||||||
|
"rgb",
|
||||||
|
"enabled",
|
||||||
|
momentum_settings.rgb_backlight);
|
||||||
|
for(int i = 0; i < SK6805_get_led_count(); i++) {
|
||||||
|
RgbColor rgb;
|
||||||
|
rgb_backlight_get_color(i, &rgb);
|
||||||
|
|
||||||
|
uint32_t led_value = 0;
|
||||||
|
memcpy(((void*)&led_value) + 1, &rgb, sizeof(RgbColor));
|
||||||
|
|
||||||
|
char led_string[5] = {'l', 'e', 'd', '0' + i, '\0'};
|
||||||
|
property_value_out(
|
||||||
|
&property_context, "%06X", 4, "hardware", "screen", "rgb", led_string, __REV(led_value));
|
||||||
|
}
|
||||||
|
|
||||||
|
// VGM Settings
|
||||||
|
property_value_out(
|
||||||
|
&property_context,
|
||||||
|
"%d",
|
||||||
|
4,
|
||||||
|
"hardware",
|
||||||
|
"vgm",
|
||||||
|
"color",
|
||||||
|
"mode",
|
||||||
|
momentum_settings.vgm_color_mode);
|
||||||
|
property_value_out(
|
||||||
|
&property_context,
|
||||||
|
"%04X",
|
||||||
|
4,
|
||||||
|
"hardware",
|
||||||
|
"vgm",
|
||||||
|
"color",
|
||||||
|
"fg",
|
||||||
|
momentum_settings.vgm_color_fg.value);
|
||||||
|
property_value_out(
|
||||||
|
&property_context,
|
||||||
|
"%04X",
|
||||||
|
4,
|
||||||
|
"hardware",
|
||||||
|
"vgm",
|
||||||
|
"color",
|
||||||
|
"bg",
|
||||||
|
momentum_settings.vgm_color_bg.value);
|
||||||
|
|
||||||
if(sep == '.') {
|
if(sep == '.') {
|
||||||
property_value_out(
|
property_value_out(
|
||||||
&property_context,
|
&property_context,
|
||||||
|
|||||||
Reference in New Issue
Block a user