#include "subbrute_main_view.h"
#include "../subbrute_i.h"
#include
#include
#define STATUS_BAR_Y_SHIFT 14
#define TAG "SubBruteMainView"
#define ITEMS_ON_SCREEN 3
#define ITEMS_INTERVAL 1
#define ITEM_WIDTH 14
#define ITEM_Y 27
#define ITEM_HEIGHT 13
#define TEXT_X 6
#define TEXT_Y 37
#define TEXT_INTERVAL 3
#define TEXT_WIDTH 12
#define ITEM_FRAME_RADIUS 2
struct SubBruteMainView {
View* view;
SubBruteMainViewCallback callback;
void* context;
uint8_t index;
bool is_select_byte;
bool two_bytes;
uint64_t key_from_file;
uint8_t repeat_values[SubBruteAttackTotalCount];
uint8_t window_position;
};
typedef struct {
uint8_t index;
uint8_t repeat_values[SubBruteAttackTotalCount];
uint8_t window_position;
bool is_select_byte;
bool two_bytes;
uint64_t key_from_file;
} SubBruteMainViewModel;
void subbrute_main_view_set_callback(
SubBruteMainView* instance,
SubBruteMainViewCallback callback,
void* context) {
furi_assert(instance);
furi_assert(callback);
instance->callback = callback;
instance->context = context;
}
void subbrute_main_view_center_displayed_key(
Canvas* canvas,
uint64_t key,
uint8_t index,
bool two_bytes) {
uint8_t text_x = TEXT_X;
uint8_t item_x = TEXT_X - ITEMS_INTERVAL;
canvas_set_font(canvas, FontSecondary);
for(int i = 0; i < 8; i++) {
char current_value[3] = {0};
uint8_t byte_value = (uint8_t)(key >> 8 * (7 - i)) & 0xFF;
snprintf(current_value, sizeof(current_value), "%02X", byte_value);
// For two bytes we need to select prev location
if(!two_bytes && i == index) {
canvas_set_color(canvas, ColorBlack);
canvas_draw_rbox(
canvas, item_x - 1, ITEM_Y, ITEM_WIDTH + 1, ITEM_HEIGHT, ITEM_FRAME_RADIUS);
canvas_set_color(canvas, ColorWhite);
canvas_draw_str(canvas, text_x, TEXT_Y, current_value);
} else if(two_bytes && (i == index || i == index - 1)) {
if(i == index) {
canvas_set_color(canvas, ColorBlack);
canvas_draw_rbox(
canvas,
item_x - ITEMS_INTERVAL - ITEM_WIDTH - 1,
ITEM_Y,
ITEM_WIDTH * 2 + ITEMS_INTERVAL * 2 + 1,
ITEM_HEIGHT,
ITEM_FRAME_RADIUS);
canvas_set_color(canvas, ColorWhite);
canvas_draw_str(canvas, text_x, TEXT_Y, current_value);
// Redraw prev element with white
memset(current_value, 0, sizeof(current_value));
byte_value = (uint8_t)(key >> 8 * (7 - i + 1)) & 0xFF;
snprintf(current_value, sizeof(current_value), "%02X", byte_value);
canvas_draw_str(
canvas, text_x - (TEXT_WIDTH + TEXT_INTERVAL), TEXT_Y, current_value);
} else {
canvas_set_color(canvas, ColorWhite);
canvas_draw_str(canvas, text_x, TEXT_Y, current_value);
}
} else {
canvas_set_color(canvas, ColorBlack);
canvas_draw_str(canvas, text_x, TEXT_Y, current_value);
}
text_x = text_x + TEXT_WIDTH + TEXT_INTERVAL;
item_x = item_x + ITEM_WIDTH + ITEMS_INTERVAL;
}
// Return normal color
canvas_set_color(canvas, ColorBlack);
}
void subbrute_main_view_draw_is_byte_selected(Canvas* canvas, SubBruteMainViewModel* model) {
canvas_set_font(canvas, FontSecondary);
canvas_draw_str_aligned(
canvas, 64, 17, AlignCenter, AlignTop, "Please select values to calc:");
subbrute_main_view_center_displayed_key(
canvas, model->key_from_file, model->index, model->two_bytes);
elements_button_center(canvas, "Select");
if(model->index > 0) {
elements_button_left(canvas, " ");
}
if(model->index < 7) {
elements_button_right(canvas, " ");
}
// Switch to another mode
if(model->two_bytes) {
elements_button_up(canvas, "One byte");
} else {
elements_button_up(canvas, "Two bytes");
}
}
void subbrute_main_view_draw_is_ordinary_selected(Canvas* canvas, SubBruteMainViewModel* model) {
uint16_t screen_width = canvas_width(canvas);
uint16_t screen_height = canvas_height(canvas);
// Title
canvas_set_font(canvas, FontPrimary);
canvas_draw_box(canvas, 0, 0, canvas_width(canvas), STATUS_BAR_Y_SHIFT);
canvas_invert_color(canvas);
canvas_draw_str_aligned(canvas, 64, 3, AlignCenter, AlignTop, SUB_BRUTE_FORCER_VERSION);
canvas_invert_color(canvas);
// Menu
canvas_set_color(canvas, ColorBlack);
canvas_set_font(canvas, FontSecondary);
const uint8_t item_height = 16;
const uint8_t string_height_offset = 9;
for(size_t position = 0; position < SubBruteAttackTotalCount; ++position) {
uint8_t item_position = position - model->window_position;
if(item_position < ITEMS_ON_SCREEN) {
if(model->index == position) {
canvas_draw_str_aligned(
canvas,
3,
string_height_offset + (item_position * item_height) + STATUS_BAR_Y_SHIFT,
AlignLeft,
AlignCenter,
subbrute_protocol_name(position));
elements_frame(
canvas, 1, 1 + (item_position * item_height) + STATUS_BAR_Y_SHIFT, 124, 15);
} else {
canvas_draw_str_aligned(
canvas,
4,
string_height_offset + (item_position * item_height) + STATUS_BAR_Y_SHIFT,
AlignLeft,
AlignCenter,
subbrute_protocol_name(position));
}
uint8_t current_repeat_count = model->repeat_values[position];
uint8_t min_repeat_count = subbrute_protocol_repeats_count(position);
if(current_repeat_count > min_repeat_count) {
#ifdef FW_ORIGIN_Official
canvas_set_font(canvas, FontSecondary);
#else
canvas_set_font(canvas, FontBatteryPercent);
#endif
char buffer[10] = {0};
snprintf(buffer, sizeof(buffer), "x%d", current_repeat_count);
uint8_t temp_x_offset_repeats =
current_repeat_count <= SUBBRUTE_PROTOCOL_MAX_REPEATS ? 15 : 18;
canvas_draw_str_aligned(
canvas,
screen_width - temp_x_offset_repeats,
string_height_offset + (item_position * item_height) + STATUS_BAR_Y_SHIFT,
AlignLeft,
AlignCenter,
buffer);
canvas_set_font(canvas, FontSecondary);
}
}
}
elements_scrollbar_pos(
canvas,
screen_width,
STATUS_BAR_Y_SHIFT + 2,
screen_height - STATUS_BAR_Y_SHIFT,
model->index,
SubBruteAttackTotalCount);
}
void subbrute_main_view_draw(Canvas* canvas, SubBruteMainViewModel* model) {
if(model->is_select_byte) {
subbrute_main_view_draw_is_byte_selected(canvas, model);
} else {
subbrute_main_view_draw_is_ordinary_selected(canvas, model);
}
}
bool subbrute_main_view_input_file_protocol(InputEvent* event, SubBruteMainView* instance) {
bool updated = false;
if(event->key == InputKeyLeft) {
if((instance->index > 0 && !instance->two_bytes) ||
(instance->two_bytes && instance->index > 1)) {
instance->index--;
}
updated = true;
} else if(event->key == InputKeyRight) {
if(instance->index < 7) {
instance->index++;
}
updated = true;
} else if(event->key == InputKeyUp) {
instance->two_bytes = !instance->two_bytes;
// Because index is changing
if(instance->two_bytes && instance->index < 7) {
instance->index++;
}
updated = true;
} else if(event->key == InputKeyOk) {
instance->callback(SubBruteCustomEventTypeIndexSelected, instance->context);
updated = true;
}
return updated;
}
bool subbrute_main_view_input_ordinary_protocol(
InputEvent* event,
SubBruteMainView* instance,
bool is_short) {
const uint8_t min_value = 0;
const uint8_t correct_total = SubBruteAttackTotalCount - 1;
uint8_t index = instance->index;
uint8_t min_repeats = subbrute_protocol_repeats_count(index);
uint8_t max_repeats = min_repeats * 3;
uint8_t current_repeats = instance->repeat_values[index];
bool updated = false;
if(event->key == InputKeyUp && is_short) {
if(index == min_value) {
instance->index = correct_total;
} else {
instance->index = CLAMP(index - 1, correct_total, min_value);
}
updated = true;
} else if(event->key == InputKeyDown && is_short) {
if(index == correct_total) {
instance->index = min_value;
} else {
instance->index = CLAMP(index + 1, correct_total, min_value);
}
updated = true;
} else if(event->key == InputKeyLeft && is_short) {
instance->repeat_values[index] = CLAMP(current_repeats - 1, max_repeats, min_repeats);
updated = true;
} else if(event->key == InputKeyRight && is_short) {
instance->repeat_values[index] = CLAMP(current_repeats + 1, max_repeats, min_repeats);
updated = true;
} else if(event->key == InputKeyOk && is_short) {
if(index == SubBruteAttackLoadFile) {
instance->callback(SubBruteCustomEventTypeLoadFile, instance->context);
} else {
instance->callback(SubBruteCustomEventTypeMenuSelected, instance->context);
}
updated = true;
}
if(updated) {
instance->window_position = instance->index;
if(instance->window_position > 0) {
instance->window_position -= 1;
}
if(SubBruteAttackTotalCount <= ITEMS_ON_SCREEN) {
instance->window_position = 0;
} else {
if(instance->window_position >= (SubBruteAttackTotalCount - ITEMS_ON_SCREEN)) {
instance->window_position = (SubBruteAttackTotalCount - ITEMS_ON_SCREEN);
}
}
}
return updated;
}
bool subbrute_main_view_input(InputEvent* event, void* context) {
furi_assert(event);
furi_assert(context);
SubBruteMainView* instance = (SubBruteMainView*)context;
if(event->key == InputKeyBack && event->type == InputTypeShort) {
#ifdef FURI_DEBUG
FURI_LOG_I(TAG, "InputKey: BACK");
#endif
instance->callback(SubBruteCustomEventTypeBackPressed, instance->context);
return false;
}
#ifdef FURI_DEBUG
FURI_LOG_D(
TAG,
"InputKey: %d, extra_repeats: %d",
event->key,
instance->repeat_values[instance->index]);
#endif
bool updated = false;
bool is_short = (event->type == InputTypeShort) || (event->type == InputTypeRepeat);
if(instance->is_select_byte) {
if(is_short) {
updated = subbrute_main_view_input_file_protocol(event, instance);
}
} else {
updated = subbrute_main_view_input_ordinary_protocol(event, instance, is_short);
}
if(updated) {
with_view_model(
instance->view,
SubBruteMainViewModel * model,
{
model->index = instance->index;
model->window_position = instance->window_position;
model->key_from_file = instance->key_from_file;
model->is_select_byte = instance->is_select_byte;
model->two_bytes = instance->two_bytes;
model->repeat_values[model->index] = instance->repeat_values[instance->index];
},
true);
}
return updated;
}
void subbrute_main_view_enter(void* context) {
furi_assert(context);
}
void subbrute_main_view_exit(void* context) {
furi_assert(context);
}
SubBruteMainView* subbrute_main_view_alloc() {
SubBruteMainView* instance = malloc(sizeof(SubBruteMainView));
instance->view = view_alloc();
view_allocate_model(instance->view, ViewModelTypeLocking, sizeof(SubBruteMainViewModel));
view_set_context(instance->view, instance);
view_set_draw_callback(instance->view, (ViewDrawCallback)subbrute_main_view_draw);
view_set_input_callback(instance->view, subbrute_main_view_input);
view_set_enter_callback(instance->view, subbrute_main_view_enter);
view_set_exit_callback(instance->view, subbrute_main_view_exit);
instance->index = 0;
instance->window_position = 0;
instance->key_from_file = 0;
instance->is_select_byte = false;
instance->two_bytes = false;
with_view_model(
instance->view,
SubBruteMainViewModel * model,
{
model->index = instance->index;
model->window_position = instance->window_position;
model->key_from_file = instance->key_from_file;
model->is_select_byte = instance->is_select_byte;
model->two_bytes = instance->two_bytes;
},
true);
return instance;
}
void subbrute_main_view_free(SubBruteMainView* instance) {
furi_assert(instance);
view_free(instance->view);
free(instance);
}
View* subbrute_main_view_get_view(SubBruteMainView* instance) {
furi_assert(instance);
return instance->view;
}
void subbrute_main_view_set_index(
SubBruteMainView* instance,
uint8_t idx,
const uint8_t* repeats,
bool is_select_byte,
bool two_bytes,
uint64_t key_from_file) {
furi_assert(instance);
furi_assert(idx < SubBruteAttackTotalCount);
#ifdef FURI_DEBUG
FURI_LOG_I(TAG, "Set index: %d, is_select_byte: %d", idx, is_select_byte);
#endif
for(size_t i = 0; i < SubBruteAttackTotalCount; i++) {
instance->repeat_values[i] = repeats[i];
}
instance->is_select_byte = is_select_byte;
instance->two_bytes = two_bytes;
instance->key_from_file = key_from_file;
instance->index = idx;
instance->window_position = idx;
if(!is_select_byte) {
if(instance->window_position > 0) {
instance->window_position -= 1;
}
if(instance->window_position >= (SubBruteAttackTotalCount - ITEMS_ON_SCREEN)) {
instance->window_position = (SubBruteAttackTotalCount - ITEMS_ON_SCREEN);
}
}
with_view_model(
instance->view,
SubBruteMainViewModel * model,
{
model->index = instance->index;
model->window_position = instance->window_position;
model->key_from_file = instance->key_from_file;
model->is_select_byte = instance->is_select_byte;
model->two_bytes = instance->two_bytes;
for(size_t i = 0; i < SubBruteAttackTotalCount; i++) {
model->repeat_values[i] = repeats[i];
}
},
true);
}
SubBruteAttacks subbrute_main_view_get_index(SubBruteMainView* instance) {
furi_assert(instance);
return instance->index;
}
const uint8_t* subbrute_main_view_get_repeats(SubBruteMainView* instance) {
furi_assert(instance);
return instance->repeat_values;
}
bool subbrute_main_view_get_two_bytes(SubBruteMainView* instance) {
furi_assert(instance);
return instance->two_bytes;
}