mirror of
https://github.com/eried/portapack-mayhem.git
synced 2024-10-01 01:26:06 -04:00
Added back scanning in BHT TX
Added file creation date display in File Manager
This commit is contained in:
parent
aebd1757da
commit
441a266dc4
@ -37,119 +37,111 @@ void BHTView::focus() {
|
||||
void BHTView::start_tx() {
|
||||
baseband::shutdown();
|
||||
|
||||
if (tx_type == XYLOS) {
|
||||
transmitter_model.set_rf_amp(true);
|
||||
transmitter_model.set_baseband_bandwidth(1750000);
|
||||
|
||||
if (target_system == XYLOS) {
|
||||
|
||||
baseband::run_image(portapack::spi_flash::image_tag_tones);
|
||||
|
||||
view_xylos.generate_message();
|
||||
|
||||
//if (view_xylos.tx_mode == XylosView::tx_modes::SINGLE)
|
||||
progressbar.set_max(20);
|
||||
//else if (view_xylos.tx_mode == XylosView::tx_modes::SEQUENCE)
|
||||
// progressbar.set_max(20 * XY_SEQ_COUNT);
|
||||
//if (tx_mode == SINGLE) {
|
||||
progressbar.set_max(XY_TONE_COUNT);
|
||||
/*} else if (tx_mode == SCAN) {
|
||||
progressbar.set_max(XY_TONE_COUNT * view_xylos.get_scan_remaining());
|
||||
}*/
|
||||
|
||||
transmitter_model.set_sampling_rate(TONES_SAMPLERATE);
|
||||
transmitter_model.set_rf_amp(true);
|
||||
transmitter_model.set_baseband_bandwidth(1750000);
|
||||
transmitter_model.enable();
|
||||
|
||||
// Setup tones
|
||||
for (size_t c = 0; c < ccir_deltas.size(); c++)
|
||||
baseband::set_tone(c, ccir_deltas[c], XY_TONE_LENGTH);
|
||||
baseband::set_tone(c, ccir_deltas[c], XY_TONE_DURATION);
|
||||
|
||||
baseband::set_tones_config(transmitter_model.channel_bandwidth(), XY_SILENCE, XY_TONE_COUNT, false, false);
|
||||
|
||||
} else if (tx_type == EPAR) {
|
||||
} else if (target_system == EPAR) {
|
||||
|
||||
baseband::run_image(portapack::spi_flash::image_tag_ook);
|
||||
|
||||
size_t bitstream_length = view_EPAR.generate_message();
|
||||
auto bitstream_length = view_EPAR.generate_message();
|
||||
|
||||
progressbar.set_max(2 * 26);
|
||||
//if (tx_mode == SINGLE) {
|
||||
progressbar.set_max(2 * EPAR_REPEAT_COUNT);
|
||||
/*} else if (tx_mode == SCAN) {
|
||||
progressbar.set_max(2 * EPAR_REPEAT_COUNT * view_EPAR.get_scan_remaining());
|
||||
}*/
|
||||
|
||||
transmitter_model.set_sampling_rate(OOK_SAMPLERATE);
|
||||
transmitter_model.set_rf_amp(true);
|
||||
transmitter_model.set_baseband_bandwidth(1750000);
|
||||
transmitter_model.enable();
|
||||
|
||||
baseband::set_ook_data(
|
||||
bitstream_length,
|
||||
OOK_SAMPLERATE / 580,
|
||||
26,
|
||||
EPAR_BIT_DURATION,
|
||||
EPAR_REPEAT_COUNT,
|
||||
encoder_defs[ENCODER_UM3750].pause_symbols
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
void BHTView::stop_tx() {
|
||||
transmitter_model.disable();
|
||||
baseband::shutdown();
|
||||
tx_mode = IDLE;
|
||||
tx_view.set_transmitting(false);
|
||||
progressbar.set_value(0);
|
||||
}
|
||||
|
||||
void BHTView::on_tx_progress(const uint32_t progress, const bool done) {
|
||||
//if (view_xylos.tx_mode == XylosView::tx_modes::SINGLE) {
|
||||
|
||||
if (done) {
|
||||
transmitter_model.disable();
|
||||
view_xylos.tx_mode = XylosView::tx_modes::IDLE;
|
||||
tx_view.set_transmitting(false);
|
||||
}
|
||||
|
||||
if (tx_type == XYLOS) {
|
||||
if (target_system == XYLOS) {
|
||||
if (done) {
|
||||
if (!checkbox_cligno.value()) {
|
||||
view_xylos.tx_mode = XylosView::tx_modes::IDLE;
|
||||
tx_view.set_transmitting(false);
|
||||
progressbar.set_value(0);
|
||||
} else {
|
||||
chThdSleepMilliseconds(field_tempo.value() * 1000); // Dirty :(
|
||||
|
||||
view_xylos.flip_relays();
|
||||
|
||||
start_tx();
|
||||
if (tx_mode == SINGLE) {
|
||||
if (checkbox_cligno.value()) {
|
||||
// TODO: Thread !
|
||||
chThdSleepMilliseconds(field_tempo.value() * 1000); // Dirty :(
|
||||
view_xylos.flip_relays();
|
||||
start_tx();
|
||||
} else
|
||||
stop_tx();
|
||||
} else if (tx_mode == SCAN) {
|
||||
if (view_xylos.increment_address())
|
||||
start_tx();
|
||||
else
|
||||
stop_tx(); // Reached end of scan range
|
||||
}
|
||||
} else
|
||||
progressbar.set_value(progress);
|
||||
} else if (tx_type == EPAR) {
|
||||
} else if (target_system == EPAR) {
|
||||
if (done) {
|
||||
|
||||
if (!view_EPAR.half) {
|
||||
view_EPAR.half = 1;
|
||||
view_EPAR.half = true;
|
||||
start_tx(); // Start second half of transmission
|
||||
} else {
|
||||
view_EPAR.half = 0;
|
||||
progressbar.set_value(0);
|
||||
if (checkbox_cligno.value()) {
|
||||
chThdSleepMilliseconds(field_tempo.value() * 1000); // Dirty :(
|
||||
|
||||
view_EPAR.flip_relays();
|
||||
|
||||
start_tx();
|
||||
view_EPAR.half = false;
|
||||
if (tx_mode == SINGLE) {
|
||||
if (checkbox_cligno.value()) {
|
||||
// TODO: Thread !
|
||||
chThdSleepMilliseconds(field_tempo.value() * 1000); // Dirty :(
|
||||
view_EPAR.flip_relays();
|
||||
start_tx();
|
||||
} else
|
||||
stop_tx();
|
||||
} else if (tx_mode == SCAN) {
|
||||
if (view_EPAR.increment_address())
|
||||
start_tx();
|
||||
else
|
||||
stop_tx(); // Reached end of scan range
|
||||
}
|
||||
}
|
||||
} else
|
||||
progressbar.set_value((26 * view_EPAR.half) + progress);
|
||||
progressbar.set_value(progress);
|
||||
}
|
||||
|
||||
/*} else if (view_xylos.tx_mode == XylosView::tx_modes::SEQUENCE) {
|
||||
if (done) {
|
||||
transmitter_model.disable();
|
||||
|
||||
if (view_xylos.seq_index < (XY_SEQ_COUNT - 1)) {
|
||||
for (c = 0; c < view_xylos.sequence_matin[view_xylos.seq_index].delay; c++)
|
||||
chThdSleepMilliseconds(1000);
|
||||
|
||||
view_xylos.seq_index++;
|
||||
|
||||
start_tx();
|
||||
} else {
|
||||
progressbar.set_value(0);
|
||||
view_xylos.tx_mode = XylosView::tx_modes::IDLE;
|
||||
tx_view.set_transmitting(false);
|
||||
}
|
||||
} else {
|
||||
progressbar.set_value((view_xylos.seq_index * 20) + progress);
|
||||
}
|
||||
}*/
|
||||
}
|
||||
|
||||
BHTView::~BHTView() {
|
||||
transmitter_model.disable();
|
||||
baseband::shutdown();
|
||||
}
|
||||
|
||||
BHTView::BHTView(NavigationView& nav) {
|
||||
@ -158,10 +150,10 @@ BHTView::BHTView(NavigationView& nav) {
|
||||
&labels,
|
||||
&view_xylos,
|
||||
&view_EPAR,
|
||||
&checkbox_scan,
|
||||
&checkbox_cligno,
|
||||
&field_tempo,
|
||||
&progressbar,
|
||||
&text_message,
|
||||
&tx_view
|
||||
});
|
||||
|
||||
@ -174,31 +166,42 @@ BHTView::BHTView(NavigationView& nav) {
|
||||
};
|
||||
};
|
||||
|
||||
/*button_seq.on_select = [this, &nav](Button&) {
|
||||
if (tx_mode == IDLE) {
|
||||
seq_index = 0;
|
||||
tx_mode = SEQUENCE;
|
||||
tx_view.set_transmitting(true);
|
||||
start_tx();
|
||||
}
|
||||
};*/
|
||||
|
||||
tx_view.on_start = [this]() {
|
||||
if (view_xylos.tx_mode == XylosView::tx_modes::IDLE) {
|
||||
view_xylos.tx_mode = XylosView::tx_modes::SINGLE;
|
||||
if (tx_mode == IDLE) {
|
||||
if (checkbox_scan.value()) {
|
||||
tx_mode = SCAN;
|
||||
} else {
|
||||
tx_mode = SINGLE;
|
||||
}
|
||||
|
||||
progressbar.set_value(0);
|
||||
tx_view.set_transmitting(true);
|
||||
tx_type = (tx_type_t)tab_view.selected();
|
||||
target_system = (target_system_t)tab_view.selected();
|
||||
view_EPAR.half = false;
|
||||
|
||||
start_tx();
|
||||
}
|
||||
};
|
||||
|
||||
tx_view.on_stop = [this]() {
|
||||
transmitter_model.disable();
|
||||
tx_view.set_transmitting(false);
|
||||
view_xylos.tx_mode = XylosView::tx_modes::IDLE;
|
||||
stop_tx();
|
||||
};
|
||||
}
|
||||
|
||||
bool EPARView::increment_address() {
|
||||
auto city_code = field_city.value();
|
||||
|
||||
if (city_code < EPAR_MAX_CITY) {
|
||||
field_city.set_value(city_code + 1);
|
||||
return true;
|
||||
} else
|
||||
return false;
|
||||
}
|
||||
|
||||
uint32_t EPARView::get_scan_remaining() {
|
||||
return EPAR_MAX_CITY - field_city.value();
|
||||
}
|
||||
|
||||
void EPARView::flip_relays() {
|
||||
// Invert first relay's state
|
||||
relay_states[0].set_selected_index(relay_states[0].selected_index() ^ 1);
|
||||
@ -207,7 +210,7 @@ void EPARView::flip_relays() {
|
||||
size_t EPARView::generate_message() {
|
||||
// R2, then R1
|
||||
return gen_message_ep(field_city.value(), field_group.selected_index_value(),
|
||||
1 - half, relay_states[half].selected_index());
|
||||
half ? 0 : 1, relay_states[half].selected_index());
|
||||
}
|
||||
|
||||
EPARView::EPARView(
|
||||
@ -219,18 +222,17 @@ EPARView::EPARView(
|
||||
add_children({
|
||||
&labels,
|
||||
&field_city,
|
||||
&field_group,
|
||||
//&button_scan
|
||||
&field_group
|
||||
});
|
||||
|
||||
field_city.set_value(220);
|
||||
field_city.set_value(0);
|
||||
field_group.set_selected_index(2);
|
||||
|
||||
field_city.on_change = [this](int32_t) { generate_message(); };
|
||||
field_group.on_change = [this](size_t, int32_t) { generate_message(); };
|
||||
|
||||
const auto relay_state_fn = [this](size_t, OptionsField::value_t) {
|
||||
this->generate_message();
|
||||
generate_message();
|
||||
};
|
||||
|
||||
size_t n = 0;
|
||||
@ -259,19 +261,25 @@ void XylosView::flip_relays() {
|
||||
relay_states[0].set_selected_index(rs ^ 3);
|
||||
}
|
||||
|
||||
bool XylosView::increment_address() {
|
||||
auto city_code = field_city.value();
|
||||
|
||||
if (city_code < XY_MAX_CITY) {
|
||||
field_city.set_value(city_code + 1);
|
||||
return true;
|
||||
} else
|
||||
return false;
|
||||
}
|
||||
|
||||
uint32_t XylosView::get_scan_remaining() {
|
||||
return XY_MAX_CITY - field_city.value();
|
||||
}
|
||||
|
||||
void XylosView::generate_message() {
|
||||
//if (tx_mode == SINGLE) {
|
||||
//text_message.set(
|
||||
gen_message_xy(field_header_a.value(), field_header_b.value(), field_city.value(), field_family.value(),
|
||||
checkbox_wcsubfamily.value(), field_subfamily.value(), checkbox_wcid.value(), field_receiver.value(),
|
||||
relay_states[0].selected_index(), relay_states[1].selected_index(),
|
||||
relay_states[2].selected_index(), relay_states[3].selected_index());
|
||||
//);
|
||||
/*} else if (tx_mode == SEQUENCE) {
|
||||
//text_message.set(
|
||||
gen_message_xy(sequence_matin[seq_index].code);
|
||||
//);
|
||||
}*/
|
||||
gen_message_xy(field_header_a.value(), field_header_b.value(), field_city.value(), field_family.value(),
|
||||
checkbox_wcsubfamily.value(), field_subfamily.value(), checkbox_wcid.value(), field_receiver.value(),
|
||||
relay_states[0].selected_index(), relay_states[1].selected_index(),
|
||||
relay_states[2].selected_index(), relay_states[3].selected_index());
|
||||
}
|
||||
|
||||
XylosView::XylosView(
|
||||
@ -300,7 +308,11 @@ XylosView::XylosView(
|
||||
field_subfamily.set_value(1);
|
||||
field_receiver.set_value(1);
|
||||
|
||||
field_header_a.on_change = [this](int32_t) { generate_message(); };
|
||||
const auto field_fn = [this](int32_t) {
|
||||
generate_message();
|
||||
};
|
||||
|
||||
field_header_a.on_change = field_fn;
|
||||
field_header_b.on_change = [this](int32_t) { generate_message(); };
|
||||
field_city.on_change = [this](int32_t) { generate_message(); };
|
||||
field_family.on_change = [this](int32_t) { generate_message(); };
|
||||
@ -321,7 +333,7 @@ XylosView::XylosView(
|
||||
checkbox_wcid.set_value(true);
|
||||
|
||||
const auto relay_state_fn = [this](size_t, OptionsField::value_t) {
|
||||
this->generate_message();
|
||||
generate_message();
|
||||
};
|
||||
|
||||
size_t n = 0;
|
||||
|
@ -34,8 +34,6 @@
|
||||
#include "encoders.hpp"
|
||||
#include "portapack.hpp"
|
||||
|
||||
#define XY_SEQ_COUNT 14
|
||||
|
||||
namespace ui {
|
||||
|
||||
class XylosView : public View {
|
||||
@ -46,38 +44,8 @@ public:
|
||||
|
||||
void flip_relays();
|
||||
void generate_message();
|
||||
|
||||
enum tx_modes {
|
||||
IDLE = 0,
|
||||
SINGLE,
|
||||
SEQUENCE
|
||||
};
|
||||
|
||||
tx_modes tx_mode = IDLE;
|
||||
|
||||
uint32_t seq_index { 0 };
|
||||
|
||||
struct sequence_t {
|
||||
const std::string code;
|
||||
const uint32_t delay;
|
||||
};
|
||||
|
||||
const sequence_t sequence_matin[XY_SEQ_COUNT] = {
|
||||
{ "0000189000B1002B0000", 19 }, // 18:9:0:00 R1=OFF (1)
|
||||
{ "0000189200B2110B0000", 16 }, // 18:9:2:00 R1=ON (4)
|
||||
{ "0000189200B1110B0000", 52 }, // 18:9:2:00 R1=OFF (4)
|
||||
{ "0000189200B2110B0000", 17 }, // 18:9:2:00 R1=ON (4)
|
||||
{ "0000189200B1110B0000", 16 }, // 18:9:2:00 R1=OFF (4)
|
||||
{ "0000189000B0012B0000", 22 }, // 18:9:0:00 R3=OFF (2)
|
||||
{ "0000189200B1120B0000", 17 }, // 18:9:2:00 R3=ON (6)
|
||||
{ "0000189200B1110B0000", 17 }, // 18:9:2:00 R3=OFF (6)
|
||||
{ "0000181AAAB1000B0000", 17 }, // 18:1:A:AA R1=OFF (10)
|
||||
{ "0000189400B1000B0000", 17 }, // 18:9:4:00 R1=OFF (7)
|
||||
{ "0000189200B1120B0000", 14 }, // 18:9:2:00 R3=ON (6)
|
||||
{ "0000189200B1110B0000", 17 }, // 18:9:2:00 R3=OFF (6)
|
||||
{ "0000181AAAB1000B0000", 17 }, // 18:1:A:AA
|
||||
{ "0000189400B0100B0000", 17 } // 18:9:4:00 R2=OFF (8)
|
||||
};
|
||||
bool increment_address();
|
||||
uint32_t get_scan_remaining();
|
||||
|
||||
private:
|
||||
Labels labels {
|
||||
@ -107,7 +75,7 @@ private:
|
||||
NumberField field_city {
|
||||
{ 16 * 8, 3 * 8 },
|
||||
2,
|
||||
{ 0, 99 },
|
||||
{ 0, XY_MAX_CITY },
|
||||
1,
|
||||
' '
|
||||
};
|
||||
@ -154,11 +122,6 @@ private:
|
||||
{ &bitmap_bulb_off, 1 },
|
||||
{ &bitmap_bulb_on, 2 }
|
||||
};
|
||||
|
||||
Button button_seq {
|
||||
{ 24 * 8, 1 * 8, 40, 32 },
|
||||
"Seq"
|
||||
};
|
||||
};
|
||||
|
||||
class EPARView : public View {
|
||||
@ -169,8 +132,10 @@ public:
|
||||
|
||||
void flip_relays();
|
||||
size_t generate_message();
|
||||
bool increment_address();
|
||||
uint32_t get_scan_remaining();
|
||||
|
||||
size_t half { 0 };
|
||||
bool half { false };
|
||||
|
||||
private:
|
||||
Labels labels {
|
||||
@ -182,9 +147,9 @@ private:
|
||||
NumberField field_city {
|
||||
{ 16 * 8, 1 * 8 },
|
||||
3,
|
||||
{ 0, 255 },
|
||||
{ 0, EPAR_MAX_CITY },
|
||||
1,
|
||||
' '
|
||||
'0'
|
||||
};
|
||||
|
||||
OptionsField field_group {
|
||||
@ -204,11 +169,6 @@ private:
|
||||
{ &bitmap_bulb_off, 0 },
|
||||
{ &bitmap_bulb_on, 1 }
|
||||
};
|
||||
|
||||
Button button_scan {
|
||||
{ 22 * 8, 1 * 8, 56, 32 },
|
||||
"Scan"
|
||||
};
|
||||
};
|
||||
|
||||
class BHTView : public View {
|
||||
@ -223,15 +183,24 @@ public:
|
||||
private:
|
||||
void on_tx_progress(const uint32_t progress, const bool done);
|
||||
void start_tx();
|
||||
void stop_tx();
|
||||
|
||||
enum tx_type_t {
|
||||
enum target_system_t {
|
||||
XYLOS = 0,
|
||||
EPAR = 1
|
||||
};
|
||||
|
||||
tx_type_t tx_type = { };
|
||||
target_system_t target_system = { };
|
||||
|
||||
Rect view_rect = { 0, 3 * 8, 240, 192 };
|
||||
enum tx_modes {
|
||||
IDLE = 0,
|
||||
SINGLE,
|
||||
SCAN
|
||||
};
|
||||
|
||||
tx_modes tx_mode = IDLE;
|
||||
|
||||
Rect view_rect = { 0, 3 * 8, 240, 176 };
|
||||
|
||||
XylosView view_xylos { view_rect };
|
||||
EPARView view_EPAR { view_rect };
|
||||
@ -245,27 +214,29 @@ private:
|
||||
{ { 29 * 8, 14 * 16 + 4 }, "s", Color::light_grey() }
|
||||
};
|
||||
|
||||
ProgressBar progressbar {
|
||||
{ 1 * 8, 14 * 16, 20 * 8, 16 },
|
||||
};
|
||||
Text text_message {
|
||||
{ 1 * 8, 15 * 16, 20 * 8, 16 },
|
||||
""
|
||||
Checkbox checkbox_scan {
|
||||
{ 1 * 8, 25 * 8 },
|
||||
4,
|
||||
"Scan"
|
||||
};
|
||||
|
||||
Checkbox checkbox_cligno {
|
||||
{ 22 * 8, 14 * 16 },
|
||||
1,
|
||||
"~"
|
||||
{ 16 * 8, 25 * 8 },
|
||||
6,
|
||||
"Cligno"
|
||||
};
|
||||
NumberField field_tempo {
|
||||
{ 27 * 8, 14 * 16 + 4 },
|
||||
{ 26 * 8, 25 * 8 + 4 },
|
||||
2,
|
||||
{ 1, 99 },
|
||||
1,
|
||||
' '
|
||||
};
|
||||
|
||||
ProgressBar progressbar {
|
||||
{ 0 * 8, 29 * 8, 30 * 8, 16 },
|
||||
};
|
||||
|
||||
TransmitterView tx_view {
|
||||
16 * 16,
|
||||
10000,
|
||||
|
@ -274,11 +274,17 @@ FileManagerView::FileManagerView(
|
||||
add_children({
|
||||
&menu_view,
|
||||
&text_empty,
|
||||
&labels,
|
||||
&text_date,
|
||||
&button_rename,
|
||||
&button_new_dir,
|
||||
&button_delete
|
||||
});
|
||||
|
||||
menu_view.on_highlight = [this]() {
|
||||
text_date.set(to_string_FAT_timestamp(file_created_date(get_selected_path())));
|
||||
};
|
||||
|
||||
refresh_list();
|
||||
|
||||
on_select_entry = [this]() {
|
||||
|
@ -150,20 +150,29 @@ private:
|
||||
void on_rename(NavigationView& nav);
|
||||
void on_delete();
|
||||
|
||||
Labels labels {
|
||||
{ { 0, 26 * 8 }, "Created ", Color::light_grey() }
|
||||
};
|
||||
|
||||
Text text_date {
|
||||
{ 8 * 8, 26 * 8 , 19 * 8, 16 },
|
||||
""
|
||||
};
|
||||
|
||||
Button button_rename {
|
||||
{ 0 * 8, 28 * 8, 14 * 8, 32 },
|
||||
{ 0 * 8, 29 * 8, 12 * 8, 32 },
|
||||
"Rename"
|
||||
};
|
||||
Button button_delete {
|
||||
{ 18 * 8, 29 * 8, 12 * 8, 32 },
|
||||
"Delete"
|
||||
};
|
||||
|
||||
Button button_new_dir {
|
||||
{ 0 * 8, 33 * 8, 14 * 8, 32 },
|
||||
{ 0 * 8, 34 * 8, 14 * 8, 32 },
|
||||
"New dir"
|
||||
};
|
||||
|
||||
Button button_delete {
|
||||
{ 18 * 8, 28 * 8, 12 * 8, 32 },
|
||||
"Delete"
|
||||
};
|
||||
};
|
||||
|
||||
} /* namespace ui */
|
||||
|
@ -72,7 +72,7 @@ void FreqManBaseView::populate_categories() {
|
||||
categories.clear();
|
||||
|
||||
for (size_t n = 0; n < file_list.size(); n++)
|
||||
categories.emplace_back(std::make_pair(file_list[n], n));
|
||||
categories.emplace_back(std::make_pair(file_list[n].substr(0, 14), n));
|
||||
|
||||
// Alphabetical sort
|
||||
std::sort(categories.begin(), categories.end(), [](auto &left, auto &right) {
|
||||
|
@ -149,8 +149,8 @@ private:
|
||||
};
|
||||
|
||||
Button button_new_category {
|
||||
{ 18 * 8, 2, 12 * 8, 20 },
|
||||
"Create new"
|
||||
{ 23 * 8, 2, 7 * 8, 20 },
|
||||
"New"
|
||||
};
|
||||
|
||||
Button button_edit_freq {
|
||||
|
@ -260,6 +260,7 @@ RDSView::RDSView(
|
||||
};
|
||||
|
||||
tx_view.on_stop = [this]() {
|
||||
// Kill tx_thread here ?
|
||||
tx_view.set_transmitting(false);
|
||||
transmitter_model.disable();
|
||||
txing = false;
|
||||
|
@ -205,6 +205,14 @@ void rename_file(const std::filesystem::path& file_path, const std::filesystem::
|
||||
f_rename(reinterpret_cast<const TCHAR*>(file_path.c_str()), reinterpret_cast<const TCHAR*>(new_name.c_str()));
|
||||
}
|
||||
|
||||
FATTimestamp file_created_date(const std::filesystem::path& file_path) {
|
||||
FILINFO filinfo;
|
||||
|
||||
f_stat(reinterpret_cast<const TCHAR*>(file_path.c_str()), &filinfo);
|
||||
|
||||
return { filinfo.fdate, filinfo.ftime };
|
||||
}
|
||||
|
||||
uint32_t make_new_directory(const std::filesystem::path& dir_path) {
|
||||
return f_mkdir(reinterpret_cast<const TCHAR*>(dir_path.c_str()));
|
||||
}
|
||||
|
@ -233,8 +233,14 @@ space_info space(const path& p);
|
||||
} /* namespace filesystem */
|
||||
} /* namespace std */
|
||||
|
||||
struct FATTimestamp {
|
||||
uint16_t FAT_date;
|
||||
uint16_t FAT_time;
|
||||
};
|
||||
|
||||
void delete_file(const std::filesystem::path& file_path);
|
||||
void rename_file(const std::filesystem::path& file_path, const std::filesystem::path& new_name);
|
||||
FATTimestamp file_created_date(const std::filesystem::path& file_path);
|
||||
uint32_t make_new_directory(const std::filesystem::path& dir_path);
|
||||
|
||||
std::vector<std::filesystem::path> scan_root_files(const std::filesystem::path& directory, const std::filesystem::path& extension);
|
||||
@ -255,6 +261,7 @@ class File {
|
||||
public:
|
||||
using Size = uint64_t;
|
||||
using Offset = uint64_t;
|
||||
using Timestamp = uint32_t;
|
||||
using Error = std::filesystem::filesystem_error;
|
||||
|
||||
template<typename T>
|
||||
@ -323,6 +330,7 @@ public:
|
||||
Result<Size> write(const void* const data, const Size bytes_to_write);
|
||||
|
||||
Result<Offset> seek(const uint64_t Offset);
|
||||
Timestamp created_date();
|
||||
Size size();
|
||||
|
||||
template<size_t N>
|
||||
|
@ -27,15 +27,14 @@
|
||||
//TEST: Check AFSK transmit end, skips last bits ?
|
||||
//TEST: Imperial in whipcalc
|
||||
|
||||
//BUG: Auto backlight off doesn't work anymore
|
||||
//BUG: (Workaround ok) CPLD-related rx ok, tx bad, see portapack.cpp lines 214+ to disable CPLD overlay
|
||||
//BUG: SCANNER Lock on frequency, if frequency jump, still locked on first one
|
||||
//BUG: SCANNER Multiple slices
|
||||
|
||||
//TODO: Open files in File Manager
|
||||
//TODO: Ask for filename after record
|
||||
//TODO: Make entries disappear from RecentEntries list in ADS-B RX (after 2 minutes with no update ?)
|
||||
//TODO: Display file creation/modification date in FileLoadView
|
||||
//TODO: Super simple text file viewer
|
||||
//TODO: Display recording frequency in Replay (from associated .txt file, if present)
|
||||
//TODO: Clean up ReplayThread
|
||||
//TODO: Cap Wav viewer position
|
||||
//TODO: Adapt wav viewer position step
|
||||
|
@ -1,168 +0,0 @@
|
||||
/*
|
||||
* Copyright (C) 2014 Jared Boone, ShareBrained Technology, Inc.
|
||||
* Copyright (C) 2016 Furrtek
|
||||
*
|
||||
* This file is part of PortaPack.
|
||||
*
|
||||
* This program is free software; you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation; either version 2, or (at your option)
|
||||
* any later version.
|
||||
*
|
||||
* 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.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program; see the file COPYING. If not, write to
|
||||
* the Free Software Foundation, Inc., 51 Franklin Street,
|
||||
* Boston, MA 02110-1301, USA.
|
||||
*/
|
||||
|
||||
// Color bitmaps generated with:
|
||||
// Gimp image > indexed colors (16), then "xxd -i *.bmp"
|
||||
|
||||
//TEST: Menuview refresh, seems to blink a lot
|
||||
//TEST: Check AFSK transmit end, skips last bits ?
|
||||
//TEST: Imperial in whipcalc
|
||||
|
||||
//BUG: Auto backlight off doesn't work anymore
|
||||
//BUG: CPLD-related rx ok, tx bad, see portapack.cpp lines 214+ to disable CPLD overlay
|
||||
//BUG: REPLAY See what's wrong with quality (format, or need for interpolation filter ?)
|
||||
//BUG: SCANNER Lock on frequency, if frequency jump, still locked on first one
|
||||
//BUG: SCANNER Multiple slices
|
||||
|
||||
//TODO: Cap Wav viewer position
|
||||
//TODO: Adapt wav viewer position step
|
||||
//TODO: Use unit_auto_scale
|
||||
//TODO: Remove make_bistream from encoders.cpp, too complex, stinks. bitstream_append should be enough.
|
||||
//TODO: Continue work on proc_afskrx_corr, see python script (it works !)
|
||||
//TODO: Super simple text file viewer
|
||||
//TODO: De bruijn sequence scanner for encoders
|
||||
//TODO: FILEMAN Rename folders
|
||||
//TODO: FILEMAN Move files
|
||||
//TODO: Frequency and bw settings were removed from modemsetup, put those back in LCR TX
|
||||
//TODO: Use separate thread for scanning in EPAR TX
|
||||
//TODO: Use separate thread for scanning in LCR TX
|
||||
//TODO: REPLAY Convert C16 to C8 on M0 core
|
||||
//TODO: Make freqman refresh simpler (use previous black rectangle method)
|
||||
//TODO: Merge AFSK and TONES procs ?
|
||||
//TODO: NFM RX mode: nav.pop on squelch
|
||||
//TODO: MORSE use prosigns
|
||||
//TODO: MORSE live keying mode
|
||||
//TODO: Use to_string_short_freq wherever possible
|
||||
//TODO: SCANNER Waveform widget as FFT view ?
|
||||
//TODO: Optimize (and group ?) CTCSS tone gen code
|
||||
/*
|
||||
Continuous (Fox-oring)
|
||||
12s transmit, 48s space (Sprint 1/5th)
|
||||
60s transmit, 240s space (Classic 1/5 min)
|
||||
60s transmit, 360s space (Classic 1/7 min)
|
||||
*/
|
||||
<<<<<<< Updated upstream
|
||||
=======
|
||||
//TODO: Use TransmitterView in TEDI/LCR, Numbers, ...
|
||||
>>>>>>> Stashed changes
|
||||
//TODO: FreqMan: Remove and rename categories
|
||||
//TODO: Mousejack ?
|
||||
//TODO: Move frequencykeypad from ui_receiver to ui_widget (used everywhere)
|
||||
//TODO: ADS-B draw trajectory + GPS coordinates + scale, and playback
|
||||
//TODO: RDS multiple groups (sequence)
|
||||
//TODO: Use ModalMessageView confirmation for TX ?
|
||||
//TODO: Use msgpack for settings, lists... on sd card
|
||||
|
||||
// Multimon-style stuff:
|
||||
//TODO: CTCSS detector
|
||||
//TODO: DMR detector
|
||||
//TODO: GSM channel detector
|
||||
//TODO: SIGFOX RX/TX
|
||||
//TODO: Playdead amnesia and login
|
||||
//TODO: Setup: Play dead by default ? Enable/disable ?
|
||||
|
||||
// Old or low-priority stuff:
|
||||
//TODO: Bodet :)
|
||||
//TODO: Analog TV tx with camcorder font character generator
|
||||
//TODO: Show address/data bit fields in OOK TX
|
||||
//TODO: Scan for OOK TX
|
||||
//TODO: Script engine ?
|
||||
//TODO: AFSK receiver
|
||||
//TODO: Check more OOK encoders
|
||||
//BUG (fixed ?): No audio in about when shown second time
|
||||
//TODO: Show MD5 mismatches for modules not found, etc...
|
||||
//TODO: Module name/filename in modules.hpp to indicate requirement in case it's not found ui_loadmodule
|
||||
//BUG: Description doesn't show up first time going to system>module info (UI drawn on top)
|
||||
//TODO: Two players tic-tac-toe
|
||||
//TODO: Analog TV pong game
|
||||
|
||||
#include "ch.h"
|
||||
|
||||
#include "portapack.hpp"
|
||||
#include "portapack_shared_memory.hpp"
|
||||
|
||||
#include "message_queue.hpp"
|
||||
|
||||
#include "ui.hpp"
|
||||
#include "ui_widget.hpp"
|
||||
#include "ui_painter.hpp"
|
||||
#include "ui_navigation.hpp"
|
||||
|
||||
#include "irq_lcd_frame.hpp"
|
||||
#include "irq_controls.hpp"
|
||||
#include "irq_rtc.hpp"
|
||||
|
||||
#include "event_m0.hpp"
|
||||
|
||||
#include "core_control.hpp"
|
||||
#include "spi_image.hpp"
|
||||
|
||||
#include "debug.hpp"
|
||||
#include "led.hpp"
|
||||
|
||||
#include "gcc.hpp"
|
||||
|
||||
#include "sd_card.hpp"
|
||||
|
||||
#include <string.h>
|
||||
|
||||
static void event_loop() {
|
||||
ui::Context context;
|
||||
ui::SystemView system_view {
|
||||
context,
|
||||
portapack::display.screen_rect()
|
||||
};
|
||||
|
||||
EventDispatcher event_dispatcher { &system_view, context };
|
||||
MessageHandlerRegistration message_handler_display_sleep {
|
||||
Message::ID::DisplaySleep,
|
||||
[&event_dispatcher](const Message* const) {
|
||||
event_dispatcher.set_display_sleep(true);
|
||||
}
|
||||
};
|
||||
|
||||
event_dispatcher.run();
|
||||
}
|
||||
|
||||
int main(void) {
|
||||
if( portapack::init() ) {
|
||||
portapack::display.init();
|
||||
|
||||
sdcStart(&SDCD1, nullptr);
|
||||
|
||||
controls_init();
|
||||
lcd_frame_sync_configure();
|
||||
rtc_interrupt_enable();
|
||||
|
||||
event_loop();
|
||||
|
||||
sdcDisconnect(&SDCD1);
|
||||
sdcStop(&SDCD1);
|
||||
|
||||
portapack::shutdown();
|
||||
}
|
||||
|
||||
m4_init(portapack::spi_flash::image_tag_hackrf, portapack::memory::map::m4_code_hackrf);
|
||||
m0_halt();
|
||||
|
||||
return 0;
|
||||
}
|
@ -28,9 +28,14 @@
|
||||
|
||||
using namespace encoders;
|
||||
|
||||
#define XY_TONE_LENGTH ((TONES_SAMPLERATE * 0.1) - 1) // 100ms
|
||||
#define XY_SILENCE (TONES_SAMPLERATE * 0.4) // 400ms
|
||||
#define XY_TONE_COUNT 20
|
||||
#define XY_TONE_DURATION ((TONES_SAMPLERATE * 0.1) - 1) // 100ms
|
||||
#define XY_SILENCE (TONES_SAMPLERATE * 0.4) // 400ms
|
||||
#define XY_TONE_COUNT 20
|
||||
#define XY_MAX_CITY 99
|
||||
|
||||
#define EPAR_BIT_DURATION (OOK_SAMPLERATE / 580)
|
||||
#define EPAR_REPEAT_COUNT 26
|
||||
#define EPAR_MAX_CITY 255
|
||||
|
||||
struct bht_city {
|
||||
std::string name;
|
||||
|
@ -165,27 +165,35 @@ std::string to_string_datetime(const rtc::RTC& value, const TimeFormat format) {
|
||||
std::string string { "" };
|
||||
|
||||
if (format == YMDHMS) {
|
||||
string += to_string_dec_uint(value.year(), 4, '0') + "/" +
|
||||
to_string_dec_uint(value.month(), 2, '0') + "/" +
|
||||
to_string_dec_uint(value.day(), 2, '0') + " ";
|
||||
string += to_string_dec_uint(value.year(), 4) + "/" +
|
||||
to_string_dec_uint(value.month(), 2) + "/" +
|
||||
to_string_dec_uint(value.day(), 2) + " ";
|
||||
}
|
||||
|
||||
string += to_string_dec_uint(value.hour(), 2, '0') + ":" +
|
||||
string += to_string_dec_uint(value.minute(), 2, '0');
|
||||
string += to_string_dec_uint(value.hour(), 2) + ":" +
|
||||
string += to_string_dec_uint(value.minute(), 2);
|
||||
|
||||
if ((format == YMDHMS) || (format == HMS))
|
||||
string += ":" + to_string_dec_uint(value.second(), 2, '0');
|
||||
string += ":" + to_string_dec_uint(value.second(), 2);
|
||||
|
||||
return string;
|
||||
}
|
||||
|
||||
std::string to_string_timestamp(const rtc::RTC& value) {
|
||||
return to_string_dec_uint(value.year(), 4, '0') +
|
||||
to_string_dec_uint(value.month(), 2, '0') +
|
||||
to_string_dec_uint(value.day(), 2, '0') +
|
||||
to_string_dec_uint(value.hour(), 2, '0') +
|
||||
to_string_dec_uint(value.minute(), 2, '0') +
|
||||
to_string_dec_uint(value.second(), 2, '0');
|
||||
return to_string_dec_uint(value.year(), 4) +
|
||||
to_string_dec_uint(value.month(), 2) +
|
||||
to_string_dec_uint(value.day(), 2) +
|
||||
to_string_dec_uint(value.hour(), 2) +
|
||||
to_string_dec_uint(value.minute(), 2) +
|
||||
to_string_dec_uint(value.second(), 2);
|
||||
}
|
||||
|
||||
std::string to_string_FAT_timestamp(const FATTimestamp& timestamp) {
|
||||
return to_string_dec_uint((timestamp.FAT_date >> 9) + 1980) + "/" +
|
||||
to_string_dec_uint((timestamp.FAT_date >> 5) & 0xF, 2) + "/" +
|
||||
to_string_dec_uint((timestamp.FAT_date & 0x1F), 2) + " " +
|
||||
to_string_dec_uint((timestamp.FAT_time >> 11), 2) + ":" +
|
||||
to_string_dec_uint((timestamp.FAT_time >> 5) & 0x3F, 2);
|
||||
}
|
||||
|
||||
std::string unit_auto_scale(double n, const uint32_t base_nano, uint32_t precision) {
|
||||
@ -208,7 +216,7 @@ std::string unit_auto_scale(double n, const uint32_t base_nano, uint32_t precisi
|
||||
|
||||
string = to_string_dec_int(integer_part);
|
||||
if (precision)
|
||||
string += '.' + to_string_dec_uint(fractional_part, precision, '0');
|
||||
string += '.' + to_string_dec_uint(fractional_part, precision);
|
||||
|
||||
if (prefix_index != 3)
|
||||
string += unit_prefix[prefix_index];
|
||||
|
@ -25,6 +25,8 @@
|
||||
#include <cstdint>
|
||||
#include <string>
|
||||
|
||||
#include "file.hpp"
|
||||
|
||||
// BARF! rtc::RTC is leaking everywhere.
|
||||
#include "lpc43xx_cpp.hpp"
|
||||
using namespace lpc43xx;
|
||||
@ -39,7 +41,7 @@ const char unit_prefix[7] { 'n', 'u', 'm', 0, 'k', 'M', 'G' };
|
||||
|
||||
// TODO: Allow l=0 to not fill/justify? Already using this way in ui_spectrum.hpp...
|
||||
std::string to_string_bin(const uint32_t n, const uint8_t l = 0);
|
||||
std::string to_string_dec_uint(const uint32_t n, const int32_t l = 0, const char fill = 0);
|
||||
std::string to_string_dec_uint(const uint32_t n, const int32_t l = 0, const char fill = '0');
|
||||
std::string to_string_dec_int(const int32_t n, const int32_t l = 0, const char fill = 0);
|
||||
std::string to_string_hex(const uint64_t n, const int32_t l = 0);
|
||||
std::string to_string_hex_array(uint8_t * const array, const int32_t l = 0);
|
||||
@ -49,6 +51,7 @@ std::string to_string_time_ms(const uint32_t ms);
|
||||
|
||||
std::string to_string_datetime(const rtc::RTC& value, const TimeFormat format = YMDHMS);
|
||||
std::string to_string_timestamp(const rtc::RTC& value);
|
||||
std::string to_string_FAT_timestamp(const FATTimestamp& timestamp);
|
||||
|
||||
std::string unit_auto_scale(double n, const uint32_t base_nano, uint32_t precision);
|
||||
|
||||
|
@ -232,6 +232,9 @@ bool MenuView::set_highlighted(int32_t new_value) {
|
||||
item_view(highlighted_item - offset)->highlight();
|
||||
}
|
||||
|
||||
if (on_highlight)
|
||||
on_highlight();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
@ -74,6 +74,7 @@ private:
|
||||
class MenuView : public View {
|
||||
public:
|
||||
std::function<void(void)> on_left { };
|
||||
std::function<void(void)> on_highlight { nullptr };
|
||||
|
||||
MenuView(Rect new_parent_rect = { 0, 0, 240, 304 }, bool keep_highlight = false);
|
||||
|
||||
|
@ -531,7 +531,9 @@ ProgressBar::ProgressBar(
|
||||
void ProgressBar::set_max(const uint32_t max) {
|
||||
if (max == _max) return;
|
||||
|
||||
_value = 0;
|
||||
if (_value > _max)
|
||||
_value = _max;
|
||||
|
||||
_max = max;
|
||||
set_dirty();
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user