2017-05-18 11:06:11 +01:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2015 Jared Boone, ShareBrained Technology, Inc.
|
2018-04-19 20:50:32 +01:00
|
|
|
* Copyright (C) 2018 Furrtek
|
2017-05-18 11:06:11 +01:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "ui_scanner.hpp"
|
2020-09-11 21:11:42 +02:00
|
|
|
#include "ui_fileman.hpp"
|
2023-07-17 11:43:37 -07:00
|
|
|
#include "ui_freqman.hpp"
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
|
2017-05-18 11:06:11 +01:00
|
|
|
using namespace portapack;
|
|
|
|
|
|
|
|
namespace ui {
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
ScannerThread::ScannerThread(std::vector<rf::Frequency> frequency_list)
|
|
|
|
: frequency_list_{std::move(frequency_list)} {
|
|
|
|
_manual_search = false;
|
|
|
|
create_thread();
|
2022-12-26 17:18:30 -05:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
ScannerThread::ScannerThread(const jammer::jammer_range_t& frequency_range, size_t def_step_hz)
|
|
|
|
: frequency_range_(frequency_range), def_step_hz_(def_step_hz) {
|
|
|
|
_manual_search = true;
|
|
|
|
create_thread();
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ScannerThread::~ScannerThread() {
|
2023-05-19 08:16:05 +12:00
|
|
|
stop();
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
void ScannerThread::create_thread() {
|
|
|
|
thread = chThdCreateFromHeap(NULL, 1024, NORMALPRIO + 10, ScannerThread::static_fn, this);
|
2022-12-26 17:18:30 -05:00
|
|
|
}
|
|
|
|
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
void ScannerThread::stop() {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (thread) {
|
|
|
|
chThdTerminate(thread);
|
|
|
|
chThdWait(thread);
|
|
|
|
thread = nullptr;
|
|
|
|
}
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
// Set by "userpause"
|
2018-04-19 20:50:32 +01:00
|
|
|
void ScannerThread::set_scanning(const bool v) {
|
2023-05-19 08:16:05 +12:00
|
|
|
_scanning = v;
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
bool ScannerThread::is_scanning() {
|
2023-05-19 08:16:05 +12:00
|
|
|
return _scanning;
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
}
|
|
|
|
|
2020-07-25 14:07:03 -03:00
|
|
|
void ScannerThread::set_freq_lock(const uint32_t v) {
|
2023-05-19 08:16:05 +12:00
|
|
|
_freq_lock = v;
|
2020-07-25 14:07:03 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t ScannerThread::is_freq_lock() {
|
2023-05-19 08:16:05 +12:00
|
|
|
return _freq_lock;
|
2020-07-25 14:07:03 -03:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
// Delete an entry from frequency list
|
|
|
|
// Caller must pause scan_thread AND can't delete a second one until this field is cleared
|
2023-05-18 10:01:39 -05:00
|
|
|
void ScannerThread::set_freq_del(const rf::Frequency v) {
|
2023-05-19 08:16:05 +12:00
|
|
|
_freq_del = v;
|
2020-07-28 01:21:52 -03:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
// Force a one-time forward or reverse frequency index change; OK to do this without pausing scan thread
|
2023-05-18 10:01:39 -05:00
|
|
|
//(used when rotary encoder is turned)
|
|
|
|
void ScannerThread::set_index_stepper(const int32_t v) {
|
2023-05-19 08:16:05 +12:00
|
|
|
_index_stepper = v;
|
2023-05-18 10:01:39 -05:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
// Set scanning direction; OK to do this without pausing scan_thread
|
2023-05-18 10:01:39 -05:00
|
|
|
void ScannerThread::set_scanning_direction(bool fwd) {
|
2023-05-19 08:16:05 +12:00
|
|
|
int32_t new_stepper = fwd ? 1 : -1;
|
2020-07-28 21:05:10 -03:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
if (_stepper != new_stepper) {
|
|
|
|
_stepper = new_stepper;
|
|
|
|
chThdSleepMilliseconds(300); // Give some pause after reversing scanning direction
|
|
|
|
}
|
2020-07-28 21:05:10 -03:00
|
|
|
}
|
|
|
|
|
2018-04-19 20:50:32 +01:00
|
|
|
msg_t ScannerThread::static_fn(void* arg) {
|
2023-05-19 08:16:05 +12:00
|
|
|
auto obj = static_cast<ScannerThread*>(arg);
|
|
|
|
obj->run();
|
|
|
|
return 0;
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScannerThread::run() {
|
2023-05-19 08:16:05 +12:00
|
|
|
RetuneMessage message{};
|
|
|
|
|
|
|
|
if (!_manual_search && frequency_list_.size()) { // IF NOT MANUAL MODE AND THERE IS A FREQUENCY LIST ...
|
|
|
|
int32_t size = frequency_list_.size();
|
|
|
|
int32_t frequency_index = (_stepper > 0) ? size : 0; // Forcing wraparound to starting frequency on 1st pass
|
|
|
|
|
|
|
|
while (!chThdShouldTerminate()) {
|
|
|
|
bool force_one_step = (_index_stepper != 0);
|
|
|
|
int32_t step = force_one_step ? _index_stepper : _stepper; //_index_stepper direction takes priority
|
|
|
|
|
|
|
|
if (_scanning || force_one_step) { // Scanning, or paused and using rotary encoder
|
|
|
|
if ((_freq_lock == 0) || force_one_step) { // normal scanning (not performing freq_lock)
|
|
|
|
frequency_index += step;
|
|
|
|
if (frequency_index >= size) // Wrap
|
|
|
|
frequency_index = 0;
|
|
|
|
else if (frequency_index < 0)
|
|
|
|
frequency_index = size - 1;
|
|
|
|
|
|
|
|
if (force_one_step)
|
|
|
|
_index_stepper = 0;
|
|
|
|
|
2023-06-11 11:47:13 -07:00
|
|
|
receiver_model.set_target_frequency(frequency_list_[frequency_index]); // Retune
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
message.freq = frequency_list_[frequency_index];
|
|
|
|
message.range = frequency_index; // Inform freq (for coloring purposes also!)
|
|
|
|
EventDispatcher::send_message(message);
|
|
|
|
} else if (_freq_del != 0) { // There is a frequency to delete
|
|
|
|
for (int32_t i = 0; i < size; i++) { // Search for the freq to delete
|
|
|
|
if (frequency_list_[i] == _freq_del) { // found: Erase it
|
|
|
|
frequency_list_.erase(frequency_list_.begin() + i);
|
|
|
|
size = frequency_list_.size();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_freq_del = 0; // deleted.
|
|
|
|
}
|
|
|
|
|
|
|
|
chThdSleepMilliseconds(SCANNER_SLEEP_MS); // Needed to (eventually) stabilize the receiver into new freq
|
|
|
|
}
|
|
|
|
} else if (_manual_search && (def_step_hz_ > 0)) // manual search range mode
|
|
|
|
{
|
|
|
|
int64_t size = (frequency_range_.max - frequency_range_.min) / def_step_hz_;
|
|
|
|
int64_t frequency_index = (_stepper > 0) ? size : 0; // Forcing wraparound to starting frequency on 1st pass
|
|
|
|
|
|
|
|
while (!chThdShouldTerminate()) {
|
|
|
|
bool force_one_step = (_index_stepper != 0);
|
|
|
|
int32_t step = force_one_step ? _index_stepper : _stepper; //_index_stepper direction takes priority
|
|
|
|
|
|
|
|
if (_scanning || force_one_step) { // Scanning, or paused and using rotary encoder
|
|
|
|
if ((_freq_lock == 0) || force_one_step) { // normal scanning (not performing freq_lock)
|
|
|
|
frequency_index += step;
|
|
|
|
if (frequency_index >= size) // Wrap
|
|
|
|
frequency_index = 0;
|
|
|
|
else if (frequency_index < 0)
|
|
|
|
frequency_index = size - 1;
|
|
|
|
|
|
|
|
if (force_one_step)
|
|
|
|
_index_stepper = 0;
|
|
|
|
|
2023-06-11 11:47:13 -07:00
|
|
|
receiver_model.set_target_frequency(frequency_range_.min + frequency_index * def_step_hz_); // Retune
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
message.freq = frequency_range_.min + frequency_index * def_step_hz_;
|
|
|
|
message.range = 0; // Inform freq (for coloring purposes also!)
|
|
|
|
EventDispatcher::send_message(message);
|
|
|
|
}
|
|
|
|
|
|
|
|
chThdSleepMilliseconds(SCANNER_SLEEP_MS); // Needed to (eventually) stabilize the receiver into new freq
|
|
|
|
}
|
|
|
|
}
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
2023-05-18 10:01:39 -05:00
|
|
|
void ScannerView::bigdisplay_update(int32_t v) {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (v != bigdisplay_current_color) {
|
|
|
|
if (v != -1)
|
|
|
|
bigdisplay_current_color = v; // -1 means refresh display but keep current color
|
|
|
|
|
|
|
|
switch (bigdisplay_current_color) {
|
|
|
|
case BDC_GREY:
|
2023-06-07 08:33:32 -07:00
|
|
|
big_display.set_style(&Styles::grey);
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
|
|
|
case BDC_YELLOW:
|
2023-06-07 08:33:32 -07:00
|
|
|
big_display.set_style(&Styles::yellow);
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
|
|
|
case BDC_GREEN:
|
2023-06-07 08:33:32 -07:00
|
|
|
big_display.set_style(&Styles::green);
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
|
|
|
case BDC_RED:
|
2023-06-07 08:33:32 -07:00
|
|
|
big_display.set_style(&Styles::red);
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// update frequency display
|
|
|
|
bigdisplay_current_frequency = current_frequency;
|
|
|
|
big_display.set(bigdisplay_current_frequency);
|
|
|
|
} else {
|
|
|
|
// no style change, but update frequency display if it's changed
|
|
|
|
if (current_frequency != bigdisplay_current_frequency) {
|
|
|
|
bigdisplay_current_frequency = current_frequency;
|
|
|
|
big_display.set(bigdisplay_current_frequency);
|
|
|
|
}
|
|
|
|
}
|
2023-05-18 10:01:39 -05:00
|
|
|
}
|
2022-12-26 17:18:30 -05:00
|
|
|
|
2023-05-18 10:01:39 -05:00
|
|
|
void ScannerView::handle_retune(int64_t freq, uint32_t freq_idx) {
|
2023-05-19 08:16:05 +12:00
|
|
|
current_index = freq_idx; // since it is an ongoing scan, this is a new index
|
|
|
|
current_frequency = freq;
|
|
|
|
|
|
|
|
if (scan_thread) {
|
|
|
|
switch (scan_thread->is_freq_lock()) {
|
|
|
|
case 0: // NO FREQ LOCK, ONGOING STANDARD SCANNING
|
|
|
|
bigdisplay_update(BDC_GREY);
|
|
|
|
break;
|
|
|
|
case 1: // STARTING LOCK FREQ
|
|
|
|
bigdisplay_update(BDC_YELLOW);
|
|
|
|
break;
|
|
|
|
case MAX_FREQ_LOCK: // FREQ IS STRONG: GREEN and scanner will pause when on_statistics_update()
|
|
|
|
bigdisplay_update(BDC_GREEN);
|
|
|
|
break;
|
|
|
|
default: // freq lock is checking the signal, do not update display
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!manual_search) {
|
|
|
|
if (frequency_list.size() > 0) {
|
|
|
|
text_current_index.set(to_string_dec_uint(freq_idx + 1, 3));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (freq_idx < description_list.size() && description_list[freq_idx].size() > 1)
|
|
|
|
desc_current_index.set(description_list[freq_idx]); // Show description from file
|
|
|
|
else
|
|
|
|
desc_current_index.set(desc_freq_list_scan); // Show Scan file name (no description in file)
|
|
|
|
}
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
2017-05-18 11:06:11 +01:00
|
|
|
void ScannerView::focus() {
|
2023-06-18 01:14:29 -05:00
|
|
|
button_load.focus();
|
2017-05-18 11:06:11 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ScannerView::~ScannerView() {
|
2023-05-19 08:16:05 +12:00
|
|
|
// make sure to stop the thread before shutting down the receiver
|
|
|
|
scan_thread.reset();
|
|
|
|
audio::output::stop();
|
|
|
|
receiver_model.disable();
|
|
|
|
baseband::shutdown();
|
2017-05-18 11:06:11 +01:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
void ScannerView::show_max_index() { // show total number of freqs to scan
|
|
|
|
text_current_index.set("---");
|
|
|
|
|
|
|
|
if (frequency_list.size() == FREQMAN_MAX_PER_FILE) {
|
2023-06-07 08:33:32 -07:00
|
|
|
text_max_index.set_style(&Styles::red);
|
2023-05-19 08:16:05 +12:00
|
|
|
text_max_index.set("/ " + to_string_dec_uint(FREQMAN_MAX_PER_FILE) + " (DB MAX!)");
|
|
|
|
} else {
|
2023-06-07 08:33:32 -07:00
|
|
|
text_max_index.set_style(&Styles::grey);
|
2023-05-19 08:16:05 +12:00
|
|
|
text_max_index.set("/ " + to_string_dec_uint(frequency_list.size()));
|
|
|
|
}
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
}
|
|
|
|
|
2017-05-18 11:06:11 +01:00
|
|
|
ScannerView::ScannerView(
|
2023-05-19 08:16:05 +12:00
|
|
|
NavigationView& nav)
|
|
|
|
: nav_{nav}, loaded_file_name{"SCANNER"} {
|
|
|
|
add_children({&labels,
|
|
|
|
&field_lna,
|
|
|
|
&field_vga,
|
|
|
|
&field_rf_amp,
|
|
|
|
&field_volume,
|
|
|
|
&field_bw,
|
|
|
|
&field_squelch,
|
|
|
|
&field_browse_wait,
|
|
|
|
&field_lock_wait,
|
|
|
|
&button_load,
|
|
|
|
&button_clear,
|
|
|
|
&rssi,
|
|
|
|
&text_current_index,
|
|
|
|
&text_max_index,
|
|
|
|
&desc_current_index,
|
|
|
|
&big_display,
|
|
|
|
&button_manual_start,
|
|
|
|
&button_manual_end,
|
|
|
|
&field_mode,
|
|
|
|
&field_step,
|
|
|
|
&button_manual_search,
|
|
|
|
&button_pause,
|
|
|
|
&button_dir,
|
|
|
|
&button_audio_app,
|
|
|
|
&button_mic_app,
|
|
|
|
&button_add,
|
|
|
|
&button_remove
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
// Populate option text for these fields
|
|
|
|
freqman_set_modulation_option(field_mode);
|
|
|
|
freqman_set_step_option(field_step);
|
|
|
|
|
2023-06-14 09:54:19 -05:00
|
|
|
// Default starting modulation (from saved App Settings if enabled, and may be overridden in SCANNER.TXT)
|
|
|
|
field_mode.set_by_value((OptionsField::value_t)receiver_model.modulation()); // Reflect the mode into the manual selector
|
|
|
|
field_step.set_by_value(receiver_model.frequency_step()); // Default step interval (Hz)
|
|
|
|
change_mode((freqman_index_t)field_mode.selected_index_value());
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
// FUTURE: perhaps additional settings should be stored in persistent memory vs using defaults
|
2023-06-11 11:47:13 -07:00
|
|
|
rf::Frequency stored_freq = receiver_model.target_frequency();
|
2023-05-19 08:16:05 +12:00
|
|
|
frequency_range.min = stored_freq - 1000000;
|
|
|
|
button_manual_start.set_text(to_string_short_freq(frequency_range.min));
|
|
|
|
frequency_range.max = stored_freq + 1000000;
|
|
|
|
button_manual_end.set_text(to_string_short_freq(frequency_range.max));
|
|
|
|
|
|
|
|
// Button to load txt files from the FREQMAN folder
|
|
|
|
button_load.on_select = [this, &nav](Button&) {
|
|
|
|
auto open_view = nav.push<FileLoadView>(".TXT");
|
2023-07-11 13:48:36 -07:00
|
|
|
open_view->push_dir(freqman_dir);
|
|
|
|
open_view->on_changed = [this, &nav](std::filesystem::path new_file_path) {
|
|
|
|
if (new_file_path.native().find(freqman_dir.native()) == 0) {
|
2023-05-19 08:16:05 +12:00
|
|
|
scan_pause();
|
2023-07-11 13:48:36 -07:00
|
|
|
frequency_file_load(new_file_path.stem().string(), true);
|
2023-05-19 08:16:05 +12:00
|
|
|
} else {
|
2023-07-11 13:48:36 -07:00
|
|
|
nav.display_modal("LOAD ERROR", "A valid file from\nFREQMAN directory is\nrequired.");
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to clear in-memory frequency list
|
|
|
|
button_clear.on_select = [this, &nav](Button&) {
|
|
|
|
if (scan_thread && frequency_list.size()) {
|
|
|
|
scan_thread->stop(); // STOP SCANNER THREAD
|
|
|
|
frequency_list.clear();
|
|
|
|
description_list.clear();
|
|
|
|
|
|
|
|
show_max_index(); // UPDATE new list size on screen
|
|
|
|
text_current_index.set("");
|
|
|
|
desc_current_index.set(desc_freq_list_scan);
|
|
|
|
scan_thread->set_freq_lock(0); // Reset the scanner lock
|
|
|
|
|
|
|
|
// FUTURE: Consider switching to manual search mode automatically after clear (but would need to validate freq range)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to configure starting frequency for a manual range search
|
|
|
|
button_manual_start.on_select = [this, &nav](Button& button) {
|
|
|
|
auto new_view = nav_.push<FrequencyKeypadView>(frequency_range.min);
|
|
|
|
new_view->on_changed = [this, &button](rf::Frequency f) {
|
|
|
|
frequency_range.min = f;
|
|
|
|
button_manual_start.set_text(to_string_short_freq(f));
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to configure ending frequency for a manual range search
|
|
|
|
button_manual_end.on_select = [this, &nav](Button& button) {
|
|
|
|
auto new_view = nav.push<FrequencyKeypadView>(frequency_range.max);
|
|
|
|
new_view->on_changed = [this, &button](rf::Frequency f) {
|
|
|
|
frequency_range.max = f;
|
|
|
|
button_manual_end.set_text(to_string_short_freq(f));
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to pause/resume scan (note that some other buttons will trigger resume also)
|
|
|
|
button_pause.on_select = [this](ButtonWithEncoder&) {
|
|
|
|
if (userpause)
|
|
|
|
user_resume();
|
|
|
|
else {
|
|
|
|
scan_pause();
|
|
|
|
button_pause.set_text("<RESUME>"); // PAUSED, show resume
|
|
|
|
userpause = true;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Encoder dial causes frequency change when focus is on pause button
|
|
|
|
button_pause.on_change = [this]() {
|
|
|
|
int32_t encoder_delta{(button_pause.get_encoder_delta() > 0) ? 1 : -1};
|
|
|
|
|
|
|
|
if (scan_thread)
|
|
|
|
scan_thread->set_index_stepper(encoder_delta);
|
|
|
|
|
|
|
|
// Restart browse timer when frequency changes
|
|
|
|
if (browse_timer != 0)
|
|
|
|
browse_timer = 1;
|
|
|
|
|
|
|
|
button_pause.set_encoder_delta(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to switch to Audio app
|
|
|
|
button_audio_app.on_select = [this](Button&) {
|
|
|
|
if (scan_thread)
|
|
|
|
scan_thread->stop();
|
2023-07-04 16:26:26 -07:00
|
|
|
auto settings = receiver_model.settings();
|
|
|
|
settings.frequency_step = field_step.selected_index_value();
|
|
|
|
nav_.replace<AnalogAudioView>(settings);
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
|
|
|
|
|
|
|
// Button to switch to Mic app
|
|
|
|
button_mic_app.on_select = [this](Button&) {
|
|
|
|
if (scan_thread)
|
|
|
|
scan_thread->stop();
|
2023-07-04 16:26:26 -07:00
|
|
|
// MicTX wants Modulation and Bandwidth overrides, but that's only stored on the RX model.
|
|
|
|
nav_.replace<MicTXView>(receiver_model.settings());
|
2023-05-19 08:16:05 +12:00
|
|
|
};
|
|
|
|
|
|
|
|
// Button to delete current frequency from scan Freq List
|
|
|
|
button_remove.on_select = [this](Button&) {
|
|
|
|
if (scan_thread && (frequency_list.size() > current_index)) {
|
|
|
|
scan_thread->set_scanning(false); // PAUSE Scanning if necessary
|
|
|
|
|
|
|
|
// Remove frequency from the Freq List in memory (it is not removed from the file)
|
|
|
|
scan_thread->set_freq_del(frequency_list[current_index]);
|
|
|
|
description_list.erase(description_list.begin() + current_index);
|
|
|
|
frequency_list.erase(frequency_list.begin() + current_index);
|
|
|
|
|
|
|
|
show_max_index(); // UPDATE new list size on screen
|
|
|
|
desc_current_index.set(""); // Clean up description (cosmetic detail)
|
|
|
|
scan_thread->set_freq_lock(0); // Reset the scanner lock
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to toggle between Manual Search and Freq List Scan modes
|
|
|
|
button_manual_search.on_select = [this](Button&) {
|
|
|
|
if (!manual_search) {
|
|
|
|
if (!frequency_range.min || !frequency_range.max) {
|
|
|
|
nav_.display_modal("Error", "Both START and END freqs\nneed a value");
|
|
|
|
} else if (frequency_range.min > frequency_range.max) {
|
|
|
|
nav_.display_modal("Error", "END freq\nis lower than START");
|
|
|
|
} else {
|
|
|
|
manual_search = true; // Switch to Manual Search mode
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
manual_search = false; // Switch to List Scan mode
|
|
|
|
}
|
|
|
|
|
|
|
|
audio::output::stop();
|
|
|
|
|
|
|
|
if (scan_thread)
|
|
|
|
scan_thread->stop(); // STOP SCANNER THREAD
|
|
|
|
|
|
|
|
if (userpause) // If user-paused, resume
|
|
|
|
user_resume();
|
|
|
|
|
|
|
|
start_scan_thread(); // RESTART SCANNER THREAD in selected mode
|
|
|
|
};
|
|
|
|
|
|
|
|
// Mode field was changed (AM/NFM/WFM)
|
|
|
|
field_mode.on_change = [this](size_t, OptionsField::value_t v) {
|
2023-06-25 08:16:49 +02:00
|
|
|
static freqman_index_t last_mode = AM_MODULATION;
|
|
|
|
// unsupported SPEC mode fix
|
|
|
|
if (v == SPEC_MODULATION) {
|
|
|
|
if (last_mode == AM_MODULATION)
|
|
|
|
v = WFM_MODULATION;
|
|
|
|
else
|
|
|
|
v = AM_MODULATION;
|
|
|
|
field_mode.set_selected_index(v);
|
|
|
|
}
|
|
|
|
last_mode = v;
|
2023-05-19 08:16:05 +12:00
|
|
|
receiver_model.disable();
|
|
|
|
baseband::shutdown();
|
|
|
|
change_mode((freqman_index_t)v);
|
|
|
|
if (scan_thread && !scan_thread->is_scanning()) // for some motive, audio output gets stopped.
|
|
|
|
audio::output::start(); // So if scan was stopped we resume audio
|
|
|
|
receiver_model.enable();
|
|
|
|
};
|
|
|
|
|
|
|
|
// Step field was changed (Hz) -- only affects manual Search mode
|
|
|
|
field_step.on_change = [this](size_t, OptionsField::value_t v) {
|
2023-06-14 09:54:19 -05:00
|
|
|
receiver_model.set_frequency_step(v);
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
if (manual_search && scan_thread) {
|
|
|
|
// Restart scan thread with new step value
|
|
|
|
scan_thread->stop(); // STOP SCANNER THREAD
|
|
|
|
|
|
|
|
// Resuming pause automatically
|
|
|
|
// FUTURE: Consider whether we should stay in Pause mode...
|
|
|
|
if (userpause) // If user-paused, resume
|
|
|
|
user_resume();
|
|
|
|
|
|
|
|
start_scan_thread(); // RESTART SCANNER THREAD in Manual Search mode
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Button to toggle Forward/Reverse
|
|
|
|
button_dir.on_select = [this](Button&) {
|
|
|
|
fwd = !fwd;
|
|
|
|
if (scan_thread)
|
|
|
|
scan_thread->set_scanning_direction(fwd);
|
|
|
|
if (userpause) // If user-paused, resume
|
|
|
|
user_resume();
|
|
|
|
button_dir.set_text(fwd ? "REVERSE" : "FORWARD");
|
|
|
|
bigdisplay_update(BDC_GREY); // Back to grey color
|
|
|
|
};
|
|
|
|
|
2023-07-11 13:48:36 -07:00
|
|
|
// TODO: remove this parsing?
|
2023-05-19 08:16:05 +12:00
|
|
|
// Button to add current frequency (found during Search) to the Scan Frequency List
|
|
|
|
button_add.on_select = [this](Button&) {
|
|
|
|
File scanner_file;
|
|
|
|
const std::string freq_file_path = "FREQMAN/" + loaded_file_name + ".TXT";
|
|
|
|
auto result = scanner_file.open(freq_file_path); // First search if freq is already in txt
|
|
|
|
|
|
|
|
if (!result.is_valid()) {
|
|
|
|
const std::string frequency_to_add = "f=" + to_string_dec_uint(current_frequency / 1000) + to_string_dec_uint(current_frequency % 1000UL, 3, '0');
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
constexpr size_t buffer_size = 1024;
|
|
|
|
char buffer[buffer_size];
|
|
|
|
|
|
|
|
for (size_t pointer = 0, freq_str_idx = 0; pointer < scanner_file.size(); pointer += buffer_size) {
|
|
|
|
size_t adjusted_buffer_size;
|
|
|
|
if (pointer + buffer_size >= scanner_file.size()) {
|
|
|
|
memset(buffer, 0, sizeof(buffer));
|
|
|
|
adjusted_buffer_size = scanner_file.size() - pointer;
|
|
|
|
} else {
|
|
|
|
adjusted_buffer_size = buffer_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
scanner_file.seek(pointer);
|
|
|
|
scanner_file.read(buffer, adjusted_buffer_size);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < adjusted_buffer_size; ++i) {
|
|
|
|
if (buffer[i] == frequency_to_add.data()[freq_str_idx]) {
|
|
|
|
++freq_str_idx;
|
|
|
|
if (freq_str_idx >= frequency_to_add.size()) {
|
|
|
|
found = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
freq_str_idx = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (found) {
|
|
|
|
nav_.display_modal("Error", "Frequency already exists");
|
|
|
|
bigdisplay_update(-1); // After showing an error
|
|
|
|
} else {
|
|
|
|
scanner_file.append(freq_file_path); // Second: append if it is not there
|
|
|
|
scanner_file.write_line(frequency_to_add);
|
|
|
|
|
|
|
|
// Add to frequency_list in memory too, since we can now switch back from manual mode
|
|
|
|
// Note that we are allowing freqs to be added to file (code above) that exceed the max count we can load into memory
|
|
|
|
if (frequency_list.size() < FREQMAN_MAX_PER_FILE) {
|
|
|
|
frequency_list.push_back(current_frequency);
|
|
|
|
description_list.push_back("");
|
|
|
|
|
|
|
|
show_max_index(); // Display updated frequency list size
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
nav_.display_modal("Error", "Cannot open " + loaded_file_name + ".TXT\nfor appending freq.");
|
|
|
|
bigdisplay_update(-1); // After showing an error
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// PRE-CONFIGURATION:
|
|
|
|
field_browse_wait.on_change = [this](int32_t v) { browse_wait = v; };
|
|
|
|
field_browse_wait.set_value(5);
|
|
|
|
|
|
|
|
field_lock_wait.on_change = [this](int32_t v) { lock_wait = v; };
|
|
|
|
field_lock_wait.set_value(2);
|
|
|
|
|
|
|
|
field_squelch.on_change = [this](int32_t v) { squelch = v; };
|
2023-07-23 14:06:19 -05:00
|
|
|
field_squelch.set_value(-30);
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
// LEARN FREQUENCIES
|
|
|
|
std::string scanner_txt = "SCANNER";
|
|
|
|
frequency_file_load(scanner_txt);
|
2020-09-11 21:11:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScannerView::frequency_file_load(std::string file_name, bool stop_all_before) {
|
2023-05-19 08:16:05 +12:00
|
|
|
bool found_range{false};
|
|
|
|
bool found_single{false};
|
2023-07-08 13:04:12 -07:00
|
|
|
freqman_index_t def_mod_index{freqman_invalid_index};
|
|
|
|
freqman_index_t def_bw_index{freqman_invalid_index};
|
|
|
|
freqman_index_t def_step_index{freqman_invalid_index};
|
2023-05-19 08:16:05 +12:00
|
|
|
|
|
|
|
// stop everything running now if required
|
|
|
|
if (stop_all_before) {
|
|
|
|
scan_thread->stop();
|
|
|
|
frequency_list.clear(); // clear the existing frequency list (expected behavior)
|
|
|
|
description_list.clear();
|
|
|
|
}
|
|
|
|
|
2023-07-08 13:04:12 -07:00
|
|
|
if (load_freqman_file(file_name, database, {})) {
|
|
|
|
loaded_file_name = file_name;
|
|
|
|
for (auto& entry_ptr : database) {
|
|
|
|
if (frequency_list.size() >= FREQMAN_MAX_PER_FILE)
|
|
|
|
break;
|
|
|
|
|
|
|
|
auto& entry = *entry_ptr;
|
|
|
|
|
|
|
|
// Get modulation & bw & step from file if specified
|
|
|
|
// Note these values could be different for each line in the file, but we only look at the first one
|
|
|
|
//
|
|
|
|
// Note that freqman requires a very specific string for these parameters,
|
|
|
|
// so check syntax in frequency file if specified value isn't being loaded
|
|
|
|
//
|
|
|
|
if (is_invalid(def_mod_index))
|
|
|
|
def_mod_index = entry.modulation;
|
|
|
|
|
|
|
|
if (is_invalid(def_bw_index))
|
|
|
|
def_bw_index = entry.bandwidth;
|
|
|
|
|
|
|
|
if (is_invalid(def_step_index))
|
|
|
|
def_step_index = entry.step;
|
|
|
|
|
|
|
|
// Get frequency
|
|
|
|
if (entry.type == freqman_type::Range) {
|
|
|
|
if (!found_range) {
|
|
|
|
// Set Start & End Search Range instead of populating the small in-memory frequency table
|
|
|
|
// NOTE: There may be multiple single frequencies in file, but only one search range is supported.
|
|
|
|
found_range = true;
|
|
|
|
frequency_range.min = entry.frequency_a;
|
|
|
|
button_manual_start.set_text(to_string_short_freq(frequency_range.min));
|
|
|
|
frequency_range.max = entry.frequency_b;
|
|
|
|
button_manual_end.set_text(to_string_short_freq(frequency_range.max));
|
|
|
|
}
|
|
|
|
} else if (entry.type == freqman_type::Single) {
|
|
|
|
found_single = true;
|
|
|
|
frequency_list.push_back(entry.frequency_a);
|
|
|
|
description_list.push_back(entry.description);
|
|
|
|
} else if (entry.type == freqman_type::HamRadio) {
|
|
|
|
// For HAM repeaters, add both receive & transmit frequencies to scan list and modify description
|
|
|
|
// (FUTURE fw versions might handle these differently)
|
|
|
|
found_single = true;
|
|
|
|
frequency_list.push_back(entry.frequency_a);
|
|
|
|
description_list.push_back("R:" + entry.description);
|
|
|
|
|
|
|
|
if ((entry.frequency_a != entry.frequency_b) &&
|
|
|
|
(frequency_list.size() < FREQMAN_MAX_PER_FILE)) {
|
|
|
|
frequency_list.push_back(entry.frequency_b);
|
|
|
|
description_list.push_back("T:" + entry.description);
|
2023-05-19 08:16:05 +12:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
loaded_file_name = "SCANNER"; // back to the default frequency file
|
|
|
|
desc_current_index.set(" NO " + file_name + ".TXT FILE ...");
|
|
|
|
}
|
|
|
|
|
|
|
|
desc_freq_list_scan = loaded_file_name + ".TXT";
|
|
|
|
if (desc_freq_list_scan.length() > 23) { // Truncate description and add ellipses if long file name
|
|
|
|
desc_freq_list_scan.resize(20);
|
|
|
|
desc_freq_list_scan = desc_freq_list_scan + "...";
|
|
|
|
}
|
|
|
|
|
2023-07-08 13:04:12 -07:00
|
|
|
if (is_valid(def_mod_index) && def_mod_index != (freqman_index_t)field_mode.selected_index_value())
|
2023-05-19 08:16:05 +12:00
|
|
|
field_mode.set_by_value(def_mod_index); // Update mode (also triggers a change callback that disables & reenables RF background)
|
|
|
|
|
2023-07-08 13:04:12 -07:00
|
|
|
if (is_valid(def_bw_index)) // Update BW if specified in file
|
2023-05-19 08:16:05 +12:00
|
|
|
field_bw.set_selected_index(def_bw_index);
|
|
|
|
|
2023-07-08 13:04:12 -07:00
|
|
|
if (is_valid(def_step_index)) // Update step if specified in file
|
2023-05-19 08:16:05 +12:00
|
|
|
field_step.set_selected_index(def_step_index);
|
|
|
|
|
|
|
|
audio::output::stop();
|
|
|
|
|
|
|
|
if (userpause) // If user-paused, resume
|
|
|
|
user_resume();
|
|
|
|
|
|
|
|
// Scan list if we found one, otherwise do manual range search
|
|
|
|
manual_search = !found_single;
|
|
|
|
|
|
|
|
start_scan_thread();
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
2023-05-18 10:01:39 -05:00
|
|
|
void ScannerView::update_squelch_while_paused(int32_t max_db) {
|
2023-05-19 08:16:05 +12:00
|
|
|
// Update audio & color based on signal level even if paused
|
|
|
|
if (++color_timer > 2) { // Counter to reduce color toggling when weak signal
|
|
|
|
if (max_db > squelch) {
|
2023-06-04 12:56:46 -07:00
|
|
|
audio::output::start(); // Re-enable audio when signal goes above squelch
|
|
|
|
receiver_model.set_headphone_volume(receiver_model.headphone_volume()); // quick fix to make sure WM8731S chips don't stay silent after pause
|
2023-05-19 08:16:05 +12:00
|
|
|
bigdisplay_update(BDC_GREEN);
|
|
|
|
} else {
|
|
|
|
audio::output::stop(); // Silence audio when signal drops below squelch
|
|
|
|
bigdisplay_update(BDC_GREY); // Back to grey color
|
|
|
|
}
|
|
|
|
|
|
|
|
color_timer = 0;
|
|
|
|
}
|
2023-05-18 10:01:39 -05:00
|
|
|
}
|
|
|
|
|
2018-04-19 20:50:32 +01:00
|
|
|
void ScannerView::on_statistics_update(const ChannelStatistics& statistics) {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (userpause) {
|
|
|
|
update_squelch_while_paused(statistics.max_db);
|
|
|
|
} else if (scan_thread) // Scanning not user-paused
|
|
|
|
{
|
|
|
|
// Resume regardless of signal strength if browse time reached
|
|
|
|
if ((browse_wait != 0) && (browse_timer >= (browse_wait * STATISTICS_UPDATES_PER_SEC))) {
|
|
|
|
browse_timer = 0;
|
|
|
|
scan_resume(); // Resume scanning
|
|
|
|
} else {
|
|
|
|
if (statistics.max_db > squelch) { // There is something on the air...(statistics.max_db > -squelch)
|
|
|
|
if (scan_thread->is_freq_lock() >= MAX_FREQ_LOCK) { // Pause scanning when signal checking time reached
|
|
|
|
if (!browse_timer) // Don't bother pausing if already paused
|
|
|
|
scan_pause();
|
|
|
|
browse_timer++; // browse_timer!=0 is also an indication that we've paused the scan
|
|
|
|
update_squelch_while_paused(statistics.max_db);
|
|
|
|
} else {
|
|
|
|
scan_thread->set_freq_lock(scan_thread->is_freq_lock() + 1); // in lock period, still analyzing the signal
|
|
|
|
if (browse_timer) // Continue incrementing browse timer while paused
|
|
|
|
browse_timer++;
|
|
|
|
}
|
|
|
|
lock_timer = 0; // Keep resetting lock timer while signal remains
|
|
|
|
} else { // There is NOTHING on the air
|
|
|
|
if (!browse_timer) {
|
|
|
|
// Signal lost and scan was never paused
|
|
|
|
if (scan_thread->is_freq_lock() > 0) { // But are we already in freq_lock ?
|
|
|
|
bigdisplay_update(BDC_GREY); // Back to grey color
|
|
|
|
scan_thread->set_freq_lock(0); // Reset the scanner lock, since there is no sig
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Signal lost and scan is still paused
|
|
|
|
lock_timer++; // Bump paused time
|
|
|
|
if (lock_timer >= (lock_wait * STATISTICS_UPDATES_PER_SEC)) { // Stay on freq until lock_wait time elapses
|
|
|
|
browse_timer = 0;
|
|
|
|
scan_resume();
|
|
|
|
} else {
|
|
|
|
browse_timer++; // Bump browse time too (may hit that limit before lock_timer reached)
|
|
|
|
update_squelch_while_paused(statistics.max_db);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScannerView::scan_pause() {
|
2023-05-19 08:16:05 +12:00
|
|
|
if (scan_thread && scan_thread->is_scanning()) {
|
|
|
|
scan_thread->set_freq_lock(0); // Reset the scanner lock (because user paused, or MAX_FREQ_LOCK reached) for next freq scan
|
|
|
|
scan_thread->set_scanning(false); // WE STOP SCANNING
|
|
|
|
}
|
|
|
|
audio::output::start();
|
2023-06-04 12:56:46 -07:00
|
|
|
receiver_model.set_headphone_volume(receiver_model.headphone_volume()); // quick fix to make sure WM8731S chips don't stay silent after pause
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScannerView::scan_resume() {
|
2023-05-19 08:16:05 +12:00
|
|
|
audio::output::stop();
|
|
|
|
bigdisplay_update(BDC_GREY); // Back to grey color
|
2023-05-18 10:01:39 -05:00
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
if (scan_thread) {
|
|
|
|
scan_thread->set_index_stepper(fwd ? 1 : -1);
|
|
|
|
scan_thread->set_scanning(true); // RESUME!
|
|
|
|
}
|
2020-07-28 01:21:52 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScannerView::user_resume() {
|
2023-05-22 16:21:00 -05:00
|
|
|
browse_timer = browse_wait * STATISTICS_UPDATES_PER_SEC + 1; // Will trigger a scan_resume() on_statistics_update, also advancing to next freq.
|
|
|
|
button_pause.set_text("<PAUSE>"); // Show button for pause, arrows indicate rotary encoder enabled for freq change
|
|
|
|
userpause = false; // Resume scanning
|
2018-04-19 20:50:32 +01:00
|
|
|
}
|
|
|
|
|
2023-05-19 08:16:05 +12:00
|
|
|
void ScannerView::change_mode(freqman_index_t new_mod) { // Before this, do a scan_thread->stop(); After this do a start_scan_thread()
|
|
|
|
using option_t = std::pair<std::string, int32_t>;
|
|
|
|
using options_t = std::vector<option_t>;
|
|
|
|
options_t bw;
|
|
|
|
field_bw.on_change = [this](size_t n, OptionsField::value_t) {
|
|
|
|
(void)n; // avoid unused warning
|
|
|
|
};
|
|
|
|
|
|
|
|
switch (new_mod) {
|
|
|
|
case AM_MODULATION:
|
|
|
|
freqman_set_bandwidth_option(new_mod, field_bw);
|
|
|
|
baseband::run_image(portapack::spi_flash::image_tag_am_audio);
|
|
|
|
receiver_model.set_modulation(ReceiverModel::Mode::AMAudio);
|
2023-06-14 09:54:19 -05:00
|
|
|
field_bw.set_by_value(receiver_model.am_configuration());
|
2023-06-06 11:46:08 -05:00
|
|
|
field_bw.on_change = [this](size_t, OptionsField::value_t n) { receiver_model.set_am_configuration(n); };
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
2023-06-14 09:54:19 -05:00
|
|
|
case NFM_MODULATION:
|
2023-05-19 08:16:05 +12:00
|
|
|
freqman_set_bandwidth_option(new_mod, field_bw);
|
|
|
|
baseband::run_image(portapack::spi_flash::image_tag_nfm_audio);
|
|
|
|
receiver_model.set_modulation(ReceiverModel::Mode::NarrowbandFMAudio);
|
2023-06-14 09:54:19 -05:00
|
|
|
field_bw.set_by_value(receiver_model.nbfm_configuration());
|
2023-06-06 11:46:08 -05:00
|
|
|
field_bw.on_change = [this](size_t, OptionsField::value_t n) { receiver_model.set_nbfm_configuration(n); };
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
|
|
|
case WFM_MODULATION:
|
|
|
|
freqman_set_bandwidth_option(new_mod, field_bw);
|
|
|
|
baseband::run_image(portapack::spi_flash::image_tag_wfm_audio);
|
|
|
|
receiver_model.set_modulation(ReceiverModel::Mode::WidebandFMAudio);
|
2023-06-14 09:54:19 -05:00
|
|
|
field_bw.set_by_value(receiver_model.wfm_configuration());
|
2023-06-06 11:46:08 -05:00
|
|
|
field_bw.on_change = [this](size_t, OptionsField::value_t n) { receiver_model.set_wfm_configuration(n); };
|
2023-05-19 08:16:05 +12:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
scanner-enhanced-version
New ui_scanner, inspired on AlainD's (alain00091) PR: https://github.com/eried/portapack-mayhem/pull/80
It includes the following:
1) A big frequency numbers display.
2) A Manual scan section (you can input a frequency range (START / END), choose a STEP value from an available of standard frequency intervals, and press SCAN button.
3) An AM / WFM / NFM scan mode selector, changing "on the fly".
4) A PAUSE / RESUME button, which will make the scanner to stop upon you listening something of interest
5) AUDIO APP button, a quick shortcut into the analog audio visualizing / recording app, with the mode, frequency, amp, LNA, VGA settings already in tune with the scanner.
6) Two enums are added to freqman.hpp, reserved for compatibility with AlainD's proposed freqman's app and / or further enhancement. More on this topic:
ORIGINAL scanner just used one frequency step, when creating scanning frequency ranges, which was unacceptable. AlainD enhanced freqman in order to pass different steppings along with ranges. This seems an excellent idea, and I preserved that aspect on my current implementation of thisscanner, while adding those enums into the freqman just to keep the door open for AlainD's freqman in the future.
7) I did eliminate the extra blank spaces added by function to_string_short_freq() which created unnecessary spacing in every app where there is need for a SHORT string, from a frequency number. (SHORT!, no extra spaces!!)
8) I also maintained AlainD idea of capping the number of frequencies which are dynamically created for each range and stored inside a memory based db. While AlainD capped the number into 400 frequencies, I was able to up that value a bit more, into 500.
Cheers!
2020-07-20 16:43:24 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void ScannerView::start_scan_thread() {
|
2023-05-19 08:16:05 +12:00
|
|
|
receiver_model.enable();
|
|
|
|
receiver_model.set_squelch_level(0);
|
|
|
|
show_max_index();
|
|
|
|
|
|
|
|
// Start Scanner Thread
|
|
|
|
// FUTURE: Consider passing additional control flags (fwd,userpause,etc) to scanner thread at start (perhaps in a data structure)
|
|
|
|
if (manual_search) {
|
|
|
|
button_manual_search.set_text("SCAN"); // Update meaning of Manual Scan button
|
|
|
|
desc_current_index.set(desc_freq_range_search);
|
|
|
|
scan_thread = std::make_unique<ScannerThread>(frequency_range, field_step.selected_index_value());
|
|
|
|
} else {
|
|
|
|
button_manual_search.set_text("SRCH"); // Update meaning of Manual Scan button
|
|
|
|
desc_current_index.set(desc_freq_list_scan);
|
|
|
|
scan_thread = std::make_unique<ScannerThread>(frequency_list);
|
|
|
|
}
|
|
|
|
|
|
|
|
scan_thread->set_scanning_direction(fwd);
|
2022-12-26 17:18:30 -05:00
|
|
|
}
|
|
|
|
|
2021-01-27 17:03:20 +01:00
|
|
|
} /* namespace ui */
|