mirror of
https://github.com/eried/portapack-mayhem.git
synced 2025-01-20 19:52:06 -05:00
New Antenna length Calculator
It reads the antennas definition from a txt file: WHIPCALC/ANTENNAS.TXT Inside the textfile you place each antenna you own with the following sintaxis: <antenna label> <elements length in mm, separated by a space> For example: ANT500 185 315 450 586 724 862 Input the required frequency, adjust the wave type (full / half / quarter, etc.) and the calculator will return the antenna length (metric and imperial) while also calculating how much you need to expand the fitting antennas you got defined on the txt. It may return up to 8 matching antennas, which is more than enough (normally you will have 2, perhaps 3 telescopic antennas around for your portapack) If by any chance your antennas txt got more than 8 antennas, and more than 8 matches the length of the freq / wave you want, it will only show the first 8 matching antennas and will warn you at the bottom that there are even more results (hidden). All calculations now are rounded into the best integer, considering first decimal, so precision is double than the original antenna calculator app.
This commit is contained in:
parent
85664c27f8
commit
8443008dfa
@ -64,22 +64,48 @@ void WhipCalcView::update_result() {
|
|||||||
|
|
||||||
text_result_metric.set(m + "m " + cm + "." + mm + "cm");
|
text_result_metric.set(m + "m " + cm + "." + mm + "cm");
|
||||||
|
|
||||||
// ANT500 elements for crude adjustment
|
uint8_t ant_count = 8; //Shown antennas counter
|
||||||
length /= 0.14;
|
console.write("\f"); //Equivalent to clear console and string buffer.
|
||||||
if (int(length) <= 4) {
|
length *= 1000; //Get length in mm needed to extend the antenna
|
||||||
auto elements = to_string_dec_int((int)length, 1);
|
for (antenna_entry antenna : antenna_db) { //go thru all antennas available
|
||||||
text_result_ant500.set(elements + " " + frac_str[((int(length * 10.0) % 10) + 1) / 3] + "ANT500 elements");
|
if (length >= antenna.elements.front() && length <= antenna.elements.back()) //This antenna is OK
|
||||||
} else {
|
{
|
||||||
text_result_ant500.set("-");
|
uint16_t element,refined_quarter=0;
|
||||||
}
|
for(element=0; element < antenna.elements.size();element++) {
|
||||||
|
if (length == antenna.elements[element]) //Exact element in length
|
||||||
|
{
|
||||||
|
element++; //Real element is +1 (zero based vector)
|
||||||
|
break; //Done with this ant
|
||||||
|
}
|
||||||
|
else if (length < antenna.elements[element])
|
||||||
|
{
|
||||||
|
double remain, this_element, quarter = 0;
|
||||||
|
remain = length - antenna.elements[element-1]; //mm needed from this element to reach length
|
||||||
|
this_element=antenna.elements[element] - antenna.elements[element -1]; //total mm on this element
|
||||||
|
quarter = (remain * 4) / this_element; //havoc & portack ended on this int(quarter) resolution.
|
||||||
|
if (quarter - int(quarter) > 0.5) { //rounding gave a measure closer to next quarter
|
||||||
|
refined_quarter=int(quarter) + 1;
|
||||||
|
if(refined_quarter == 4) { //rounding gave a measure closer to next element
|
||||||
|
refined_quarter = 0;
|
||||||
|
element++;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
refined_quarter=int(quarter);
|
||||||
|
}
|
||||||
|
break; //Done with this ant
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!ant_count) {
|
||||||
|
console.write(" and more ...");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
console.write(antenna.label + " " + to_string_dec_int(element,1) + frac_str[refined_quarter] + " elements\n");
|
||||||
|
ant_count--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Imperial
|
// Imperial
|
||||||
calclength = (speed_of_light_fps / (double)field_frequency.value()) * divider;
|
calclength = (speed_of_light_fps / (double)field_frequency.value()) * divider;
|
||||||
|
|
||||||
/* auto feet = to_string_dec_int((int)length, 3);
|
|
||||||
auto inch = to_string_dec_int(int(length * 10.0) % 12, 2);
|
|
||||||
auto inch_c = to_string_dec_int(int(length * 100.0) % 10, 1); */
|
|
||||||
|
|
||||||
auto feet = to_string_dec_int(int(calclength), 3);
|
auto feet = to_string_dec_int(int(calclength), 3);
|
||||||
calclength = get_decimals(calclength,12); //inches
|
calclength = get_decimals(calclength,12); //inches
|
||||||
auto inch = to_string_dec_int(int(calclength), 2);
|
auto inch = to_string_dec_int(int(calclength), 2);
|
||||||
@ -94,14 +120,39 @@ WhipCalcView::WhipCalcView(
|
|||||||
|
|
||||||
add_children({
|
add_children({
|
||||||
&labels,
|
&labels,
|
||||||
|
&antennas_on_memory,
|
||||||
&field_frequency,
|
&field_frequency,
|
||||||
&options_type,
|
&options_type,
|
||||||
&text_result_metric,
|
&text_result_metric,
|
||||||
&text_result_imperial,
|
&text_result_imperial,
|
||||||
&text_result_ant500,
|
&console,
|
||||||
&button_exit
|
&button_exit
|
||||||
});
|
});
|
||||||
|
|
||||||
|
File antennas_file; //LOAD /WHIPCALC/ANTENNAS.TXT from microSD
|
||||||
|
auto result = antennas_file.open("WHIPCALC/ANTENNAS.TXT");
|
||||||
|
antenna_db.clear(); //Start with fresh db
|
||||||
|
if (result.is_valid()) {
|
||||||
|
antenna_Default(); //There is no txt, store a default ant500
|
||||||
|
} else {
|
||||||
|
std::string line; //There is a txt file
|
||||||
|
char one_char[1]; //Read it char by char
|
||||||
|
for (size_t pointer=0; pointer < antennas_file.size();pointer++) {
|
||||||
|
antennas_file.seek(pointer);
|
||||||
|
antennas_file.read(one_char, 1);
|
||||||
|
if ((int)one_char[0] > 31) { //ascii space upwards
|
||||||
|
line += one_char[0]; //Add it to the textline
|
||||||
|
}
|
||||||
|
else if (one_char[0] == '\n') { //New Line
|
||||||
|
txtline_process(line); //make sense of this textline
|
||||||
|
line.clear(); //Ready for next textline
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (line.length() > 0) txtline_process(line); //Last line had no newline at end ?
|
||||||
|
if (!antenna_db.size()) antenna_Default(); //no antenna found on txt, use default
|
||||||
|
}
|
||||||
|
antennas_on_memory.set(to_string_dec_int(antenna_db.size(),2) + " antennas"); //tell user
|
||||||
|
|
||||||
options_type.on_change = [this](size_t, OptionsField::value_t) {
|
options_type.on_change = [this](size_t, OptionsField::value_t) {
|
||||||
this->update_result();
|
this->update_result();
|
||||||
};
|
};
|
||||||
@ -128,4 +179,32 @@ WhipCalcView::WhipCalcView(
|
|||||||
update_result();
|
update_result();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ui::WhipCalcView::txtline_process(std::string& line) {
|
||||||
|
if (line.find("#") != std::string::npos) return; //Line is just a comment
|
||||||
|
size_t previous = 0;
|
||||||
|
uint16_t value = 0;
|
||||||
|
antenna_entry new_antenna;
|
||||||
|
size_t current = line.find(" ");
|
||||||
|
while (current != std::string::npos) {
|
||||||
|
if (!previous) { //first space found
|
||||||
|
new_antenna.label.assign(line,0,current); //antenna label
|
||||||
|
} else {
|
||||||
|
value = std::stoi(line.substr(previous,current - previous));
|
||||||
|
if (!value) return; //No element length? abort antenna
|
||||||
|
new_antenna.elements.push_back(value); //Store this new element
|
||||||
|
}
|
||||||
|
previous = current + 1;
|
||||||
|
current = line.find(" ",previous); //Search for next space delimiter
|
||||||
|
}
|
||||||
|
if (!previous) return; //Not even a label ? drop this antenna!
|
||||||
|
value = std::stoi(line.substr(previous,current - previous)); //Last element
|
||||||
|
if (!value) return;
|
||||||
|
new_antenna.elements.push_back(value);
|
||||||
|
antenna_db.push_back(new_antenna); //Add this antenna
|
||||||
|
}
|
||||||
|
|
||||||
|
void ui::WhipCalcView::antenna_Default() {
|
||||||
|
antenna_db.push_back({"ANT500",{ 185, 315, 450, 586, 724, 862} }); //store a default ant500
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -28,6 +28,7 @@
|
|||||||
#include "ui_receiver.hpp"
|
#include "ui_receiver.hpp"
|
||||||
#include "ui_navigation.hpp"
|
#include "ui_navigation.hpp"
|
||||||
#include "string_format.hpp"
|
#include "string_format.hpp"
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
namespace ui {
|
namespace ui {
|
||||||
|
|
||||||
@ -43,50 +44,67 @@ private:
|
|||||||
const double speed_of_light_mps = 299792458.0; // m/s
|
const double speed_of_light_mps = 299792458.0; // m/s
|
||||||
const double speed_of_light_fps = 983571087.90472; // feet/s
|
const double speed_of_light_fps = 983571087.90472; // feet/s
|
||||||
|
|
||||||
const std::string frac_str[4] = { "", "1/4 ", "1/2 ", "3/4 " };
|
const std::string frac_str[4] = { "", " 1/4", " 1/2", " 3/4" };
|
||||||
|
|
||||||
|
struct antenna_entry {
|
||||||
|
std::string label { };
|
||||||
|
std::vector <uint16_t> elements { };
|
||||||
|
};
|
||||||
|
|
||||||
|
std::vector<antenna_entry> antenna_db { };
|
||||||
|
|
||||||
double get_decimals(double num, int16_t mult, bool round = false);
|
double get_decimals(double num, int16_t mult, bool round = false);
|
||||||
void update_result();
|
void update_result();
|
||||||
|
|
||||||
|
uint16_t string_to_number(std::string);
|
||||||
|
void txtline_process(std::string&);
|
||||||
|
void antenna_Default();
|
||||||
|
|
||||||
Labels labels {
|
Labels labels {
|
||||||
|
{ { 5 * 8, 1 * 16 }, "Loaded:", Color::light_grey() },
|
||||||
{ { 2 * 8, 2 * 16 }, "Frequency:", Color::light_grey() },
|
{ { 2 * 8, 2 * 16 }, "Frequency:", Color::light_grey() },
|
||||||
{ { 2 * 8, 3 * 16 }, "Type:", Color::light_grey() }
|
{ { 7 * 8, 3 * 16 }, "Wave:", Color::light_grey() },
|
||||||
|
{ { 5 * 8, 4 * 16 }, "Metric:", Color::light_grey() },
|
||||||
|
{ { 3 * 8, 5 * 16 }, "Imperial:", Color::light_grey() }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Text antennas_on_memory {
|
||||||
|
{ 13 * 8, 1 * 16, 2 * 16, 16 },
|
||||||
|
};
|
||||||
|
|
||||||
FrequencyField field_frequency {
|
FrequencyField field_frequency {
|
||||||
{ 13 * 8, 2 * 16 },
|
{ 13 * 8, 2 * 16 },
|
||||||
};
|
};
|
||||||
|
|
||||||
OptionsField options_type {
|
OptionsField options_type {
|
||||||
{ 8 * 8, 3 * 16 },
|
{ 13 * 8, 3 * 16 },
|
||||||
12,
|
7,
|
||||||
{
|
{
|
||||||
{ "Full wave", 8 },
|
{ "Full", 8 },
|
||||||
{ "Half wave", 4 },
|
{ "Half", 4 },
|
||||||
{ "Quarter wave", 2 },
|
{ "Quarter", 2 },
|
||||||
{ "3/4 wave", 6 },
|
{ "3/4", 6 },
|
||||||
{ "1/8 wave", 1 },
|
{ "1/8", 1 },
|
||||||
{ "3/8 wave", 3 },
|
{ "3/8", 3 },
|
||||||
{ "5/8 wave", 5 },
|
{ "5/8", 5 },
|
||||||
{ "7/8 wave", 7 }
|
{ "7/8", 7 }
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
Text text_result_metric {
|
Text text_result_metric {
|
||||||
{ 3 * 8, 5 * 16, 10 * 16, 16 },
|
{ 13 * 8, 4 * 16, 10 * 16, 16 },
|
||||||
"-"
|
"-"
|
||||||
};
|
};
|
||||||
Text text_result_imperial {
|
Text text_result_imperial {
|
||||||
{ 2 * 8, 6 * 16, 10 * 16, 16 },
|
{ 13 * 8, 5 * 16, 10 * 16, 16 },
|
||||||
"-"
|
|
||||||
};
|
|
||||||
Text text_result_ant500 {
|
|
||||||
{ 2 * 8, 8 * 16, 26 * 16, 16 },
|
|
||||||
"-"
|
"-"
|
||||||
};
|
};
|
||||||
|
Console console {
|
||||||
|
{ 0, 7 * 16, 240, 144 } //Allows to show up to 8 antennas
|
||||||
|
};
|
||||||
|
|
||||||
Button button_exit {
|
Button button_exit {
|
||||||
{ 72, 264, 96, 32 },
|
{ 72, 17 * 16, 96, 32 },
|
||||||
"Exit"
|
"Exit"
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -595,7 +595,10 @@ void Console::write(std::string message) {
|
|||||||
pen_color = s.foreground;
|
pen_color = s.foreground;
|
||||||
escape = false;
|
escape = false;
|
||||||
} else {
|
} else {
|
||||||
if (c == '\n') {
|
if (c=='\f') { //Add FORM FEED (clear screen)
|
||||||
|
clear();
|
||||||
|
buffer.clear();
|
||||||
|
} else if (c == '\n') {
|
||||||
crlf();
|
crlf();
|
||||||
} else if (c == '\x1B') {
|
} else if (c == '\x1B') {
|
||||||
escape = true;
|
escape = true;
|
||||||
|
4
sdcard/WHIPCALC/ANTENNAS.TXT
Normal file
4
sdcard/WHIPCALC/ANTENNAS.TXT
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
#<antenna label> <elements length in mm, separated by a space>
|
||||||
|
ANT700 95 134 175 206 230 245
|
||||||
|
ANT500 185 315 450 586 724 862
|
||||||
|
CHEAPO 118 183 253 326 399 476
|
Loading…
Reference in New Issue
Block a user