mirror of
https://github.com/f4exb/sdrangel.git
synced 2024-09-12 08:06:34 -04:00
Allow sampling devices to be refreshed while SDRangel is running.
(Possibly need to add a mutex for MainCore::getPluginManager, so only can be used by one thread) Add option to automatically update My Position based on GPS. This is started in MainCore, so we get position as soon as possible. Don't set QFileDialog::DontUseNativeDialog on Android, as Qt's file dialog can't access user storage. Set globally for Linux/Windows, rather than for each dialog. MainWindow: - Add welcome dialog for Android. - Don't show menu bar or status bar on Android to save screen space. - On Android, change tab position when screen orientation changes. - Load default configurations and presets first time SDRangel is run. - Change loadConfiguration to use a QProgressDialog rather than QMessageBox, as the latter was crashing on Android. - Use DialogPositioner to ensure dialogs are on screen.
This commit is contained in:
parent
cedd7c20d1
commit
68603fec43
30
app/main.cpp
30
app/main.cpp
@ -26,6 +26,9 @@
|
||||
#include <QGLFormat>
|
||||
#include <QSurfaceFormat>
|
||||
#endif
|
||||
#ifdef ANDROID
|
||||
#include "util/android.h"
|
||||
#endif
|
||||
|
||||
#include "loggerwithfile.h"
|
||||
#include "mainwindow.h"
|
||||
@ -49,6 +52,9 @@ static int runQtApplication(int argc, char* argv[], qtwebapp::LoggerWithFile *lo
|
||||
#if (QT_VERSION >= QT_VERSION_CHECK(5, 10, 0)) && (QT_VERSION <= QT_VERSION_CHECK(6, 0, 0))
|
||||
QApplication::setAttribute(Qt::AA_DisableWindowContextHelpButton);
|
||||
#endif
|
||||
#ifndef ANDROID
|
||||
QApplication::setAttribute(Qt::AA_DontUseNativeDialogs); // Don't use on Android, otherwise we can't access files on internal storage
|
||||
#endif
|
||||
|
||||
QApplication a(argc, argv);
|
||||
|
||||
@ -111,6 +117,20 @@ static int runQtApplication(int argc, char* argv[], qtwebapp::LoggerWithFile *lo
|
||||
#endif
|
||||
|
||||
#endif
|
||||
|
||||
#ifdef ANDROID
|
||||
// Default sized sliders can be hard to move using touch GUIs, so increase szie
|
||||
// FIXME: How can we do a double border around the handle, as Fusion style seems to use?
|
||||
// Dialog borders are hard to see as is (perhaps as Android doesn't have a title bar), so use same color as for MDI
|
||||
qApp->setStyleSheet("QSlider {min-height: 20px; } "
|
||||
"QSlider::groove:horizontal { border: 1px solid #2e2e2e; height: 1px; background: #444444; margin: 1px 0;}"
|
||||
"QSlider::handle:horizontal { background: #585858; border: 1px double #676767; width: 16px; margin: -8px 0px; border-radius: 3px;}"
|
||||
"QSlider::sub-page {background: #ff8c00; border: 1px solid #2e2e2e;border-top-right-radius: 0px;border-bottom-right-radius: 0px;border-top-left-radius: 5px;border-bottom-left-radius: 5px;}"
|
||||
"QSlider::add-page {background: #444444; border: 1px solid #2e2e2e;border-top-right-radius: 5px;border-bottom-right-radius: 5px;border-top-left-radius: 0px;border-bottom-left-radius: 0px;}"
|
||||
"QDialog { border: 1px solid #ff8c00;}"
|
||||
);
|
||||
#endif
|
||||
|
||||
MainParser parser;
|
||||
parser.parse(*qApp);
|
||||
|
||||
@ -160,10 +180,18 @@ int main(int argc, char* argv[])
|
||||
QSurfaceFormat::setDefaultFormat(sfc);
|
||||
#endif
|
||||
|
||||
qtwebapp::LoggerWithFile *logger = new qtwebapp::LoggerWithFile(qApp);
|
||||
#ifdef ANDROID
|
||||
qtwebapp::LoggerWithFile *logger = nullptr;
|
||||
qInstallMessageHandler(Android::messageHandler);
|
||||
#else
|
||||
qtwebapp::LoggerWithFile *logger = new qtwebapp::LoggerWithFile(qApp);
|
||||
logger->installMsgHandler();
|
||||
#endif
|
||||
|
||||
int res = runQtApplication(argc, argv, logger);
|
||||
|
||||
delete logger;
|
||||
|
||||
qWarning("SDRangel quit.");
|
||||
return res;
|
||||
}
|
||||
|
@ -1,6 +1,7 @@
|
||||
project (sdrbase)
|
||||
|
||||
if(WIN32)
|
||||
# This should probably be in ../CMakeLists.txt with the other variables like it
|
||||
set(OPUS_INCLUDE_DIRS "${CMAKE_SOURCE_DIR}/external/windows/libopus/include")
|
||||
set(OPUS_LIBRARIES "${CMAKE_SOURCE_DIR}/external/windows/libopus/lib/x64/libopus.lib")
|
||||
endif()
|
||||
@ -45,10 +46,11 @@ if (LIBSIGMF_FOUND AND CMAKE_CXX_COMPILER_ID STREQUAL "GNU")
|
||||
set(sdrbase_LIBSIGMF_LIB ${LIBSIGMF_LIBRARIES})
|
||||
endif()
|
||||
|
||||
# serialdv now required
|
||||
add_definitions(-DDSD_USE_SERIALDV)
|
||||
include_directories(${LIBSERIALDV_INCLUDE_DIR})
|
||||
set(sdrbase_SERIALDV_LIB ${LIBSERIALDV_LIBRARY})
|
||||
if (LIBSERIALDV_FOUND)
|
||||
add_definitions(-DDSD_USE_SERIALDV)
|
||||
include_directories(${LIBSERIALDV_INCLUDE_DIR})
|
||||
set(sdrbase_SERIALDV_LIB ${LIBSERIALDV_LIBRARY})
|
||||
endif()
|
||||
|
||||
set(sdrbase_SOURCES
|
||||
${sdrbase_SOURCES}
|
||||
@ -168,6 +170,7 @@ set(sdrbase_SOURCES
|
||||
settings/rollupstate.cpp
|
||||
|
||||
util/ais.cpp
|
||||
util/android.cpp
|
||||
util/aviationweather.cpp
|
||||
util/ax25.cpp
|
||||
util/aprs.cpp
|
||||
@ -388,6 +391,7 @@ set(sdrbase_HEADERS
|
||||
settings/rollupstate.h
|
||||
|
||||
util/ais.h
|
||||
util/android.h
|
||||
util/aviationweather.h
|
||||
util/ax25.h
|
||||
util/aprs.h
|
||||
@ -491,6 +495,7 @@ target_link_libraries(sdrbase
|
||||
Qt::Core
|
||||
Qt::Multimedia
|
||||
Qt::WebSockets
|
||||
Qt::Positioning
|
||||
httpserver
|
||||
logging
|
||||
qrtplib
|
||||
@ -499,8 +504,15 @@ target_link_libraries(sdrbase
|
||||
if (Qt6_FOUND)
|
||||
target_link_libraries(sdrbase
|
||||
Qt::Core5Compat
|
||||
Qt::CorePrivate
|
||||
)
|
||||
endif()
|
||||
if(ANDROID AND NOT ENABLE_QT6)
|
||||
target_link_libraries(sdrbase Qt::AndroidExtras)
|
||||
endif()
|
||||
if(ANDROID)
|
||||
target_link_libraries(sdrbase log)
|
||||
endif()
|
||||
|
||||
install(TARGETS sdrbase DESTINATION ${INSTALL_LIB_DIR})
|
||||
|
||||
|
@ -17,8 +17,6 @@
|
||||
|
||||
#include <QGlobalStatic>
|
||||
|
||||
#include "plugin/pluginmanager.h"
|
||||
|
||||
#include "deviceenumerator.h"
|
||||
|
||||
Q_GLOBAL_STATIC(DeviceEnumerator, deviceEnumerator)
|
||||
@ -246,89 +244,101 @@ bool DeviceEnumerator::isMIMOEnumerated(const QString& deviceHwId, int deviceSeq
|
||||
return false;
|
||||
}
|
||||
|
||||
void DeviceEnumerator::enumerateRxDevices(PluginManager *pluginManager)
|
||||
void DeviceEnumerator::enumerateDevices(PluginAPI::SamplingDeviceRegistrations& deviceRegistrations, DevicesEnumeration& enumeration, PluginInterface::SamplingDevice::StreamType type)
|
||||
{
|
||||
m_rxEnumeration.clear();
|
||||
PluginAPI::SamplingDeviceRegistrations& rxDeviceRegistrations = pluginManager->getSourceDeviceRegistrations();
|
||||
int index = 0;
|
||||
DevicesEnumeration tempEnumeration;
|
||||
PluginInterface::OriginDevices originDevices;
|
||||
QStringList originDevicesHwIds;
|
||||
|
||||
for (int i = 0; i < rxDeviceRegistrations.count(); i++)
|
||||
for (int i = 0; i < deviceRegistrations.count(); i++)
|
||||
{
|
||||
qDebug("DeviceEnumerator::enumerateRxDevices: %s", qPrintable(rxDeviceRegistrations[i].m_deviceId));
|
||||
rxDeviceRegistrations[i].m_plugin->enumOriginDevices(m_originDevicesHwIds, m_originDevices);
|
||||
PluginInterface::SamplingDevices samplingDevices = rxDeviceRegistrations[i].m_plugin->enumSampleSources(m_originDevices);
|
||||
qDebug("DeviceEnumerator::enumerateDevices: %s", qPrintable(deviceRegistrations[i].m_deviceId));
|
||||
deviceRegistrations[i].m_plugin->enumOriginDevices(originDevicesHwIds, originDevices);
|
||||
PluginInterface::SamplingDevices samplingDevices;
|
||||
if (type == PluginInterface::SamplingDevice::StreamSingleRx) {
|
||||
samplingDevices = deviceRegistrations[i].m_plugin->enumSampleSources(originDevices);
|
||||
} else if (type == PluginInterface::SamplingDevice::StreamSingleTx) {
|
||||
samplingDevices = deviceRegistrations[i].m_plugin->enumSampleSinks(originDevices);
|
||||
} else {
|
||||
samplingDevices = deviceRegistrations[i].m_plugin->enumSampleMIMO(originDevices);
|
||||
}
|
||||
|
||||
for (int j = 0; j < samplingDevices.count(); j++)
|
||||
{
|
||||
m_rxEnumeration.push_back(
|
||||
tempEnumeration.push_back(
|
||||
DeviceEnumeration(
|
||||
samplingDevices[j],
|
||||
rxDeviceRegistrations[i].m_plugin,
|
||||
index
|
||||
deviceRegistrations[i].m_plugin,
|
||||
0 // Index will be set when adding to enumeration below
|
||||
)
|
||||
);
|
||||
index++;
|
||||
}
|
||||
}
|
||||
|
||||
// We don't remove devices that are no-longer found from the enumeration list, in case their
|
||||
// index is referenced in some other object (E.g. DeviceGUI)
|
||||
// Instead we mark as removed. If later re-added, then we re-use the same index
|
||||
for (DevicesEnumeration::iterator it = enumeration.begin(); it != enumeration.end(); ++it)
|
||||
{
|
||||
bool found = false;
|
||||
for (DevicesEnumeration::iterator it2 = tempEnumeration.begin(); it2 != tempEnumeration.end(); ++it2)
|
||||
{
|
||||
if (it->m_samplingDevice == it2->m_samplingDevice)
|
||||
{
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (!found) {
|
||||
it->m_samplingDevice.removed = true;
|
||||
}
|
||||
}
|
||||
|
||||
// Add new entries and update existing (in case re-added or sequence number has changed)
|
||||
int index = enumeration.size();
|
||||
for (DevicesEnumeration::iterator it = tempEnumeration.begin(); it != tempEnumeration.end(); ++it)
|
||||
{
|
||||
|
||||
bool found = false;
|
||||
for (DevicesEnumeration::iterator it2 = enumeration.begin(); it2 != enumeration.end(); ++it2)
|
||||
{
|
||||
if (it->m_samplingDevice == it2->m_samplingDevice)
|
||||
{
|
||||
it2->m_samplingDevice.removed = false;
|
||||
it2->m_samplingDevice.displayedName = it->m_samplingDevice.displayedName;
|
||||
found = true;
|
||||
break;
|
||||
}
|
||||
|
||||
}
|
||||
if (!found)
|
||||
{
|
||||
it->m_index = index++;
|
||||
enumeration.push_back(*it);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void DeviceEnumerator::enumerateRxDevices(PluginManager *pluginManager)
|
||||
{
|
||||
enumerateDevices(pluginManager->getSourceDeviceRegistrations(), m_rxEnumeration, PluginInterface::SamplingDevice::StreamSingleRx);
|
||||
}
|
||||
|
||||
void DeviceEnumerator::enumerateTxDevices(PluginManager *pluginManager)
|
||||
{
|
||||
m_txEnumeration.clear();
|
||||
PluginAPI::SamplingDeviceRegistrations& txDeviceRegistrations = pluginManager->getSinkDeviceRegistrations();
|
||||
int index = 0;
|
||||
|
||||
for (int i = 0; i < txDeviceRegistrations.count(); i++)
|
||||
{
|
||||
qDebug("DeviceEnumerator::enumerateTxDevices: %s", qPrintable(txDeviceRegistrations[i].m_deviceId));
|
||||
txDeviceRegistrations[i].m_plugin->enumOriginDevices(m_originDevicesHwIds, m_originDevices);
|
||||
PluginInterface::SamplingDevices samplingDevices = txDeviceRegistrations[i].m_plugin->enumSampleSinks(m_originDevices);
|
||||
|
||||
for (int j = 0; j < samplingDevices.count(); j++)
|
||||
{
|
||||
m_txEnumeration.push_back(
|
||||
DeviceEnumeration(
|
||||
samplingDevices[j],
|
||||
txDeviceRegistrations[i].m_plugin,
|
||||
index
|
||||
)
|
||||
);
|
||||
index++;
|
||||
}
|
||||
}
|
||||
enumerateDevices(pluginManager->getSinkDeviceRegistrations(), m_txEnumeration, PluginInterface::SamplingDevice::StreamSingleTx);
|
||||
}
|
||||
|
||||
void DeviceEnumerator::enumerateMIMODevices(PluginManager *pluginManager)
|
||||
{
|
||||
m_mimoEnumeration.clear();
|
||||
PluginAPI::SamplingDeviceRegistrations& mimoDeviceRegistrations = pluginManager->getMIMODeviceRegistrations();
|
||||
int index = 0;
|
||||
|
||||
for (int i = 0; i < mimoDeviceRegistrations.count(); i++)
|
||||
{
|
||||
qDebug("DeviceEnumerator::enumerateMIMODevices: %s", qPrintable(mimoDeviceRegistrations[i].m_deviceId));
|
||||
mimoDeviceRegistrations[i].m_plugin->enumOriginDevices(m_originDevicesHwIds, m_originDevices);
|
||||
PluginInterface::SamplingDevices samplingDevices = mimoDeviceRegistrations[i].m_plugin->enumSampleMIMO(m_originDevices);
|
||||
|
||||
for (int j = 0; j < samplingDevices.count(); j++)
|
||||
{
|
||||
m_mimoEnumeration.push_back(
|
||||
DeviceEnumeration(
|
||||
samplingDevices[j],
|
||||
mimoDeviceRegistrations[i].m_plugin,
|
||||
index
|
||||
)
|
||||
);
|
||||
index++;
|
||||
}
|
||||
}
|
||||
enumerateDevices(pluginManager->getMIMODeviceRegistrations(), m_mimoEnumeration, PluginInterface::SamplingDevice::StreamMIMO);
|
||||
}
|
||||
|
||||
void DeviceEnumerator::listRxDeviceNames(QList<QString>& list, std::vector<int>& indexes) const
|
||||
{
|
||||
for (DevicesEnumeration::const_iterator it = m_rxEnumeration.begin(); it != m_rxEnumeration.end(); ++it)
|
||||
{
|
||||
if ((it->m_samplingDevice.claimed < 0) || (it->m_samplingDevice.type == PluginInterface::SamplingDevice::BuiltInDevice))
|
||||
if (((it->m_samplingDevice.claimed < 0) && (!it->m_samplingDevice.removed)) || (it->m_samplingDevice.type == PluginInterface::SamplingDevice::BuiltInDevice))
|
||||
{
|
||||
list.append(it->m_samplingDevice.displayedName);
|
||||
indexes.push_back(it->m_index);
|
||||
@ -340,7 +350,7 @@ void DeviceEnumerator::listTxDeviceNames(QList<QString>& list, std::vector<int>&
|
||||
{
|
||||
for (DevicesEnumeration::const_iterator it = m_txEnumeration.begin(); it != m_txEnumeration.end(); ++it)
|
||||
{
|
||||
if ((it->m_samplingDevice.claimed < 0) || (it->m_samplingDevice.type == PluginInterface::SamplingDevice::BuiltInDevice))
|
||||
if (((it->m_samplingDevice.claimed < 0) && (!it->m_samplingDevice.removed)) || (it->m_samplingDevice.type == PluginInterface::SamplingDevice::BuiltInDevice))
|
||||
{
|
||||
list.append(it->m_samplingDevice.displayedName);
|
||||
indexes.push_back(it->m_index);
|
||||
@ -352,7 +362,7 @@ void DeviceEnumerator::listMIMODeviceNames(QList<QString>& list, std::vector<int
|
||||
{
|
||||
for (DevicesEnumeration::const_iterator it = m_mimoEnumeration.begin(); it != m_mimoEnumeration.end(); ++it)
|
||||
{
|
||||
if ((it->m_samplingDevice.claimed < 0) || (it->m_samplingDevice.type == PluginInterface::SamplingDevice::BuiltInDevice))
|
||||
if (((it->m_samplingDevice.claimed < 0) && (!it->m_samplingDevice.removed)) || (it->m_samplingDevice.type == PluginInterface::SamplingDevice::BuiltInDevice))
|
||||
{
|
||||
list.append(it->m_samplingDevice.displayedName);
|
||||
indexes.push_back(it->m_index);
|
||||
|
@ -21,6 +21,7 @@
|
||||
#include <vector>
|
||||
|
||||
#include "plugin/plugininterface.h"
|
||||
#include "plugin/pluginmanager.h"
|
||||
#include "device/deviceuserargs.h"
|
||||
#include "export.h"
|
||||
|
||||
@ -85,9 +86,8 @@ private:
|
||||
DevicesEnumeration m_rxEnumeration;
|
||||
DevicesEnumeration m_txEnumeration;
|
||||
DevicesEnumeration m_mimoEnumeration;
|
||||
PluginInterface::OriginDevices m_originDevices;
|
||||
QStringList m_originDevicesHwIds;
|
||||
|
||||
void enumerateDevices(PluginAPI::SamplingDeviceRegistrations& deviceRegistrations, DevicesEnumeration& enumeration, PluginInterface::SamplingDevice::StreamType type);
|
||||
PluginInterface *getRxRegisteredPlugin(PluginManager *pluginManager, const QString& deviceHwId);
|
||||
PluginInterface *getTxRegisteredPlugin(PluginManager *pluginManager, const QString& deviceHwId);
|
||||
PluginInterface *getMIMORegisteredPlugin(PluginManager *pluginManager, const QString& deviceHwId);
|
||||
|
@ -18,6 +18,8 @@
|
||||
#include <QGlobalStatic>
|
||||
#include <QCoreApplication>
|
||||
#include <QString>
|
||||
#include <QDebug>
|
||||
#include <QGeoPositionInfoSource>
|
||||
|
||||
#include "loggerwithfile.h"
|
||||
#include "dsp/dsptypes.h"
|
||||
@ -71,6 +73,8 @@ MainCore::MainCore()
|
||||
m_masterTimer.start(50);
|
||||
m_startMsecsSinceEpoch = QDateTime::currentMSecsSinceEpoch();
|
||||
m_masterElapsedTimer.start();
|
||||
// Position can take a while to determine, so we start updates at program startup
|
||||
initPosition();
|
||||
}
|
||||
|
||||
MainCore::~MainCore()
|
||||
@ -335,3 +339,36 @@ void MainCore::debugMaps()
|
||||
feIt.value()->getIndex(), feIt.key()->getIndexInFeatureSet(), qPrintable(feIt.key()->getURI()), qPrintable(feIt.key()->getName()));
|
||||
}
|
||||
}
|
||||
|
||||
void MainCore::initPosition()
|
||||
{
|
||||
m_positionSource = QGeoPositionInfoSource::createDefaultSource(this);
|
||||
if (m_positionSource)
|
||||
{
|
||||
connect(m_positionSource, &QGeoPositionInfoSource::positionUpdated, this, &MainCore::positionUpdated);
|
||||
m_positionSource->startUpdates();
|
||||
}
|
||||
else
|
||||
{
|
||||
qDebug() << "MainCore::initPosition: No position source.";
|
||||
}
|
||||
}
|
||||
|
||||
const QGeoPositionInfo& MainCore::getPosition() const
|
||||
{
|
||||
return m_position;
|
||||
}
|
||||
|
||||
void MainCore::positionUpdated(const QGeoPositionInfo &info)
|
||||
{
|
||||
if (info.isValid())
|
||||
{
|
||||
m_position = info;
|
||||
if (m_settings.getAutoUpdatePosition())
|
||||
{
|
||||
m_settings.setLatitude(m_position.coordinate().latitude());
|
||||
m_settings.setLongitude(m_position.coordinate().longitude());
|
||||
m_settings.setAltitude(m_position.coordinate().altitude());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -25,6 +25,7 @@
|
||||
#include <QElapsedTimer>
|
||||
#include <QDateTime>
|
||||
#include <QObject>
|
||||
#include <QGeoPositionInfo>
|
||||
|
||||
#include "export.h"
|
||||
#include "settings/mainsettings.h"
|
||||
@ -38,6 +39,7 @@ class FeatureSet;
|
||||
class Feature;
|
||||
class PluginManager;
|
||||
class MessageQueue;
|
||||
class QGeoPositionInfoSource;
|
||||
|
||||
namespace qtwebapp {
|
||||
class LoggerWithFile;
|
||||
@ -846,7 +848,7 @@ public:
|
||||
qint64 getStartMsecsSinceEpoch() const { return m_startMsecsSinceEpoch; } //!< Epoch timestamp in millisecodns close to elapsed timer start
|
||||
const MainSettings& getSettings() const { return m_settings; }
|
||||
MessageQueue *getMainMessageQueue() { return m_mainMessageQueue; }
|
||||
const PluginManager *getPluginManager() const { return m_pluginManager; }
|
||||
PluginManager *getPluginManager() const { return m_pluginManager; }
|
||||
std::vector<DeviceSet*>& getDeviceSets() { return m_deviceSets; }
|
||||
std::vector<FeatureSet*>& getFeatureeSets() { return m_featureSets; }
|
||||
void setLoggingOptions();
|
||||
@ -875,6 +877,8 @@ public:
|
||||
// pipes
|
||||
MessagePipes& getMessagePipes() { return m_messagePipes; }
|
||||
DataPipes& getDataPipes() { return m_dataPipes; }
|
||||
// Position
|
||||
const QGeoPositionInfo& getPosition() const;
|
||||
|
||||
friend class MainServer;
|
||||
friend class MainWindow;
|
||||
@ -883,6 +887,9 @@ public:
|
||||
friend class DeviceSetPresetsDialog;
|
||||
friend class ConfigurationsDialog;
|
||||
|
||||
public slots:
|
||||
void positionUpdated(const QGeoPositionInfo &info);
|
||||
|
||||
signals:
|
||||
void deviceSetAdded(int index, DeviceAPI *device);
|
||||
void deviceChanged(int index);
|
||||
@ -912,7 +919,10 @@ private:
|
||||
PluginManager* m_pluginManager;
|
||||
MessagePipes m_messagePipes;
|
||||
DataPipes m_dataPipes;
|
||||
QGeoPositionInfoSource *m_positionSource;
|
||||
QGeoPositionInfo m_position;
|
||||
|
||||
void initPosition();
|
||||
void debugMaps();
|
||||
};
|
||||
|
||||
|
@ -66,6 +66,7 @@ public:
|
||||
int deviceNbItems; //!< Number of items (or streams) in the device. >1 for composite devices.
|
||||
int deviceItemIndex; //!< For composite devices this is the Rx or Tx stream index. -1 if not initialized
|
||||
int claimed; //!< This is the device set index if claimed else -1
|
||||
bool removed; //!< Set if device has been removed
|
||||
|
||||
SamplingDevice(const QString& _displayedName,
|
||||
const QString& _hardwareId,
|
||||
@ -85,8 +86,17 @@ public:
|
||||
streamType(_streamType),
|
||||
deviceNbItems(_deviceNbItems),
|
||||
deviceItemIndex(_deviceItemIndex),
|
||||
claimed(-1)
|
||||
claimed(-1),
|
||||
removed(false)
|
||||
{ }
|
||||
|
||||
bool operator==(const SamplingDevice& rhs) const
|
||||
{
|
||||
return displayedName == rhs.displayedName
|
||||
&& hardwareId == rhs.hardwareId
|
||||
&& id == rhs.id
|
||||
&& serial == rhs.serial;
|
||||
}
|
||||
};
|
||||
typedef QList<SamplingDevice> SamplingDevices;
|
||||
|
||||
|
@ -151,6 +151,12 @@ public:
|
||||
emit preferenceChanged(Preferences::Altitude);
|
||||
}
|
||||
|
||||
bool getAutoUpdatePosition() const { return m_preferences.getAutoUpdatePosition(); }
|
||||
void setAutoUpdatePosition(bool autoUpdatePosition)
|
||||
{
|
||||
m_preferences.setAutoUpdatePosition(autoUpdatePosition);
|
||||
}
|
||||
|
||||
QtMsgType getConsoleMinLogLevel() const { return m_preferences.getConsoleMinLogLevel(); }
|
||||
void setConsoleMinLogLevel(const QtMsgType& minLogLevel)
|
||||
{
|
||||
|
@ -17,6 +17,7 @@ void Preferences::resetToDefaults()
|
||||
m_latitude = 49.012423; // Set an interesting location so map doesn't open up in the middle of the ocean
|
||||
m_longitude = 8.418125;
|
||||
m_altitude = 0.0f;
|
||||
m_autoUpdatePosition = true;
|
||||
m_useLogFile = false;
|
||||
m_logFileName = "sdrangel.log";
|
||||
m_consoleMinLogLevel = QtDebugMsg;
|
||||
@ -41,6 +42,7 @@ QByteArray Preferences::serialize() const
|
||||
s.writeFloat((int) Altitude, m_altitude);
|
||||
s.writeS32((int) SourceItemIndex, m_sourceItemIndex);
|
||||
s.writeS32((int) Multisampling, m_multisampling);
|
||||
s.writeBool((int) AutoUpdatePosition, m_autoUpdatePosition);
|
||||
return s.final();
|
||||
}
|
||||
|
||||
@ -95,9 +97,11 @@ bool Preferences::deserialize(const QByteArray& data)
|
||||
}
|
||||
|
||||
d.readS32((int) Multisampling, &m_multisampling, 0);
|
||||
d.readBool((int) AutoUpdatePosition, &m_autoUpdatePosition, true);
|
||||
|
||||
return true;
|
||||
} else
|
||||
}
|
||||
else
|
||||
{
|
||||
resetToDefaults();
|
||||
return false;
|
||||
|
@ -22,7 +22,8 @@ public:
|
||||
StationName,
|
||||
Altitude,
|
||||
SourceItemIndex,
|
||||
Multisampling
|
||||
Multisampling,
|
||||
AutoUpdatePosition
|
||||
};
|
||||
|
||||
Preferences();
|
||||
@ -60,6 +61,9 @@ public:
|
||||
float getAltitude() const { return m_altitude; }
|
||||
void setAltitude(float altitude) { m_altitude = altitude; }
|
||||
|
||||
bool getAutoUpdatePosition() const { return m_autoUpdatePosition; }
|
||||
void setAutoUpdatePosition(bool autoUpdatePosition) { m_autoUpdatePosition = autoUpdatePosition; }
|
||||
|
||||
QtMsgType getConsoleMinLogLevel() const { return m_consoleMinLogLevel; }
|
||||
void setConsoleMinLogLevel(const QtMsgType& minLogLevel) { m_consoleMinLogLevel = minLogLevel; }
|
||||
|
||||
@ -87,6 +91,7 @@ protected:
|
||||
float m_latitude; //!< Position of the station
|
||||
float m_longitude;
|
||||
float m_altitude; //!< Altitude in metres
|
||||
bool m_autoUpdatePosition; //!< Automatically update position from GPS
|
||||
|
||||
QtMsgType m_consoleMinLogLevel;
|
||||
QtMsgType m_fileMinLogLevel;
|
||||
|
213
sdrbase/util/android.cpp
Normal file
213
sdrbase/util/android.cpp
Normal file
@ -0,0 +1,213 @@
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// Copyright (C) 2022 Jon Beniston, M7RCE //
|
||||
// //
|
||||
// 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 as version 3 of the License, 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 V3 for more details. //
|
||||
// //
|
||||
// You should have received a copy of the GNU General Public License //
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#ifdef ANDROID
|
||||
|
||||
#include <QDebug>
|
||||
|
||||
#include <android/log.h>
|
||||
|
||||
#include "android.h"
|
||||
|
||||
#if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
|
||||
|
||||
#include <QtCore/private/qandroidextras_p.h>
|
||||
#include <QJniObject>
|
||||
#include <QJniEnvironment>
|
||||
|
||||
void Android::sendIntent()
|
||||
{
|
||||
QJniObject url = QJniObject::fromString("iqsrc://-f 1090000000 -p 1234 -s 2000000 -a 127.0.0.1 -g 100");
|
||||
QJniObject intent = QJniObject::callStaticObjectMethod("android/content/Intent", "parseUri", "(Ljava/lang/String;I)Landroid/content/Intent;", url.object<jstring>(), 0x00000001); // Creates Intent(ACTION_VIEW, url)
|
||||
QtAndroidPrivate::startActivity(intent, 0, [](int requestCode, int resultCode, const QJniObject &data) {
|
||||
(void) data;
|
||||
|
||||
qDebug() << "MainCore::sendIntent " << requestCode << resultCode;
|
||||
});
|
||||
}
|
||||
|
||||
QStringList Android::listUSBDeviceSerials(int vid, int pid)
|
||||
{
|
||||
QStringList serials;
|
||||
QJniEnvironment jniEnv;
|
||||
|
||||
QJniObject activity = QJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid())
|
||||
{
|
||||
QJniObject serialsObj = activity.callObjectMethod("listUSBDeviceSerials", "(II)[Ljava/lang/String;", vid, pid);
|
||||
int serialsLen = jniEnv->GetArrayLength(serialsObj.object<jarray>());
|
||||
for (int i = 0; i < serialsLen; i++)
|
||||
{
|
||||
QJniObject arrayElement = jniEnv->GetObjectArrayElement(serialsObj.object<jobjectArray>(), i);
|
||||
QString serial = arrayElement.toString();
|
||||
serials.append(serial);
|
||||
}
|
||||
}
|
||||
|
||||
return serials;
|
||||
}
|
||||
|
||||
int Android::openUSBDevice(const QString &serial)
|
||||
{
|
||||
int fd = -1;
|
||||
QJniObject activity = QJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid())
|
||||
{
|
||||
QJniObject serialsObj = QJniObject::fromString(serial);
|
||||
fd = activity.callMethod<jint>("openUSBDevice", "(Ljava/lang/String;)I", serialsObj.object<jstring>());
|
||||
}
|
||||
|
||||
return fd;
|
||||
}
|
||||
|
||||
void Android::closeUSBDevice(int fd)
|
||||
{
|
||||
if (fd >= 0)
|
||||
{
|
||||
QJniObject activity = QJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid()) {
|
||||
activity.callMethod<void>("closeUSBDevice", "(I)V", fd);
|
||||
} else {
|
||||
qCritical() << "MainCore::closeUSBDevice: activity is not valid.";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Android::moveTaskToBack()
|
||||
{
|
||||
QJniObject activity = QJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid()) {
|
||||
activity.callMethod<jboolean>("moveTaskToBack", "(Z)Z", true);
|
||||
}
|
||||
}
|
||||
|
||||
#else // QT_VERSION
|
||||
|
||||
#include <QtAndroid>
|
||||
#include <QAndroidIntent>
|
||||
#include <QAndroidJniObject>
|
||||
#include <QAndroidJniEnvironment>
|
||||
#include <QAndroidActivityResultReceiver>
|
||||
|
||||
void Android::sendIntent()
|
||||
{
|
||||
QAndroidJniObject url = QAndroidJniObject::fromString("iqsrc://-f 1090000000 -p 1234 -s 2000000 -a 127.0.0.1 -g 100");
|
||||
QAndroidJniObject intent = QAndroidJniObject::callStaticObjectMethod("android/content/Intent", "parseUri", "(Ljava/lang/String;I)Landroid/content/Intent;", url.object<jstring>(), 0x00000001); // Creates Intent(ACTION_VIEW, url)
|
||||
QtAndroid::startActivity(intent, 0, [](int requestCode, int resultCode, const QAndroidJniObject &data) {
|
||||
(void) data;
|
||||
|
||||
qDebug() << "MainCore::sendIntent " << requestCode << resultCode;
|
||||
});
|
||||
}
|
||||
|
||||
QStringList Android::listUSBDeviceSerials(int vid, int pid)
|
||||
{
|
||||
QStringList serials;
|
||||
QAndroidJniEnvironment jniEnv;
|
||||
|
||||
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid())
|
||||
{
|
||||
QAndroidJniObject serialsObj = activity.callObjectMethod("listUSBDeviceSerials", "(II)[Ljava/lang/String;", vid, pid);
|
||||
int serialsLen = jniEnv->GetArrayLength(serialsObj.object<jarray>());
|
||||
for (int i = 0; i < serialsLen; i++)
|
||||
{
|
||||
QAndroidJniObject arrayElement = jniEnv->GetObjectArrayElement(serialsObj.object<jobjectArray>(), i);
|
||||
QString serial = arrayElement.toString();
|
||||
serials.append(serial);
|
||||
}
|
||||
}
|
||||
|
||||
return serials;
|
||||
}
|
||||
|
||||
int Android::openUSBDevice(const QString &serial)
|
||||
{
|
||||
int fd = -1;
|
||||
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid())
|
||||
{
|
||||
QAndroidJniObject serialsObj = QAndroidJniObject::fromString(serial);
|
||||
fd = activity.callMethod<jint>("openUSBDevice", "(Ljava/lang/String;)I", serialsObj.object<jstring>());
|
||||
}
|
||||
|
||||
return fd;
|
||||
}
|
||||
|
||||
void Android::closeUSBDevice(int fd)
|
||||
{
|
||||
if (fd >= 0)
|
||||
{
|
||||
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid()) {
|
||||
activity.callMethod<void>("closeUSBDevice", "(I)V", fd);
|
||||
} else {
|
||||
qCritical() << "MainCore::closeUSBDevice: activity is not valid.";
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void Android::moveTaskToBack()
|
||||
{
|
||||
QAndroidJniObject activity = QAndroidJniObject::callStaticObjectMethod("org/qtproject/qt5/android/QtNative", "activity", "()Landroid/app/Activity;");
|
||||
if (activity.isValid()) {
|
||||
activity.callMethod<jboolean>("moveTaskToBack", "(Z)Z", true);
|
||||
}
|
||||
}
|
||||
|
||||
#endif // QT6
|
||||
|
||||
// Redirect qDebug/qWarning to Android log, so we can view remotely with adb
|
||||
void Android::messageHandler(QtMsgType type, const QMessageLogContext& context, const QString& msg)
|
||||
{
|
||||
QString report = msg;
|
||||
if (context.file && !QString(context.file).isEmpty())
|
||||
{
|
||||
report += " in file ";
|
||||
report += QString(context.file);
|
||||
report += " line ";
|
||||
report += QString::number(context.line);
|
||||
}
|
||||
if (context.function && !QString(context.function).isEmpty())
|
||||
{
|
||||
report += +" function ";
|
||||
report += QString(context.function);
|
||||
}
|
||||
const char * const local = report.toLocal8Bit().constData();
|
||||
const char * const applicationName = "sdrangel";
|
||||
switch (type)
|
||||
{
|
||||
case QtDebugMsg:
|
||||
__android_log_write(ANDROID_LOG_DEBUG, applicationName, local);
|
||||
break;
|
||||
case QtInfoMsg:
|
||||
__android_log_write(ANDROID_LOG_INFO, applicationName, local);
|
||||
break;
|
||||
case QtWarningMsg:
|
||||
__android_log_write(ANDROID_LOG_WARN, applicationName, local);
|
||||
break;
|
||||
case QtCriticalMsg:
|
||||
__android_log_write(ANDROID_LOG_ERROR, applicationName, local);
|
||||
break;
|
||||
case QtFatalMsg:
|
||||
default:
|
||||
__android_log_write(ANDROID_LOG_FATAL, applicationName, local);
|
||||
abort();
|
||||
}
|
||||
}
|
||||
|
||||
#endif // ANDROID
|
44
sdrbase/util/android.h
Normal file
44
sdrbase/util/android.h
Normal file
@ -0,0 +1,44 @@
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
// Copyright (C) 2022 Jon Beniston, M7RCE //
|
||||
// //
|
||||
// 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 as version 3 of the License, 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 V3 for more details. //
|
||||
// //
|
||||
// You should have received a copy of the GNU General Public License //
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>. //
|
||||
///////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#ifndef SDRBASE_ANDROID_H_
|
||||
#define SDRBASE_ANDROID_H_
|
||||
|
||||
#ifdef ANDROID
|
||||
|
||||
#include <QtGlobal>
|
||||
#include <QString>
|
||||
|
||||
#include "export.h"
|
||||
|
||||
// Android specific functions
|
||||
class SDRBASE_API Android
|
||||
{
|
||||
public:
|
||||
|
||||
static void sendIntent();
|
||||
static QStringList listUSBDeviceSerials(int vid, int pid);
|
||||
static int openUSBDevice(const QString &serial);
|
||||
static void closeUSBDevice(int fd);
|
||||
static void moveTaskToBack();
|
||||
static void messageHandler(QtMsgType type, const QMessageLogContext& context, const QString& msg);
|
||||
|
||||
};
|
||||
|
||||
#endif // ANDROID
|
||||
|
||||
#endif // SDRBASE_ANDROID_H_
|
@ -27,11 +27,26 @@
|
||||
|
||||
#include "settings/configuration.h"
|
||||
|
||||
ConfigurationsDialog::ConfigurationsDialog(QWidget* parent) :
|
||||
ConfigurationsDialog::ConfigurationsDialog(bool openOnly, QWidget* parent) :
|
||||
QDialog(parent),
|
||||
ui(new Ui::ConfigurationsDialog)
|
||||
{
|
||||
ui->setupUi(this);
|
||||
if (openOnly)
|
||||
{
|
||||
ui->buttonBox->setStandardButtons(QDialogButtonBox::Close | QDialogButtonBox::Open);
|
||||
ui->configurationDelete->setVisible(false);
|
||||
ui->configurationEdit->setVisible(false);
|
||||
ui->configurationExport->setVisible(false);
|
||||
ui->configurationImport->setVisible(false);
|
||||
ui->configurationLoad->setVisible(false);
|
||||
ui->configurationSave->setVisible(false);
|
||||
ui->configurationUpdate->setVisible(false);
|
||||
}
|
||||
else
|
||||
{
|
||||
ui->description->setVisible(false);
|
||||
}
|
||||
}
|
||||
|
||||
ConfigurationsDialog::~ConfigurationsDialog()
|
||||
@ -354,9 +369,7 @@ void ConfigurationsDialog::on_configurationExport_clicked()
|
||||
tr("Open preset export file"),
|
||||
".",
|
||||
tr("Configuration export files (*.cfgx)"),
|
||||
0,
|
||||
QFileDialog::DontUseNativeDialog
|
||||
);
|
||||
0);
|
||||
|
||||
if (fileName != "")
|
||||
{
|
||||
@ -404,8 +417,7 @@ void ConfigurationsDialog::on_configurationImport_clicked()
|
||||
tr("Open preset export file"),
|
||||
".",
|
||||
tr("Preset export files (*.cfgx)"),
|
||||
0,
|
||||
QFileDialog::DontUseNativeDialog
|
||||
0
|
||||
);
|
||||
|
||||
if (fileName != "")
|
||||
@ -497,3 +509,9 @@ void ConfigurationsDialog::deleteConfigurationGroup(const QString& groupName)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void ConfigurationsDialog::accept()
|
||||
{
|
||||
on_configurationLoad_clicked();
|
||||
QDialog::accept();
|
||||
}
|
||||
|
@ -38,7 +38,7 @@ namespace Ui {
|
||||
class SDRGUI_API ConfigurationsDialog : public QDialog {
|
||||
Q_OBJECT
|
||||
public:
|
||||
explicit ConfigurationsDialog(QWidget* parent = nullptr);
|
||||
explicit ConfigurationsDialog(bool openOnly, QWidget* parent = nullptr);
|
||||
~ConfigurationsDialog();
|
||||
void setConfigurations(QList<Configuration*>* configurations) { m_configurations = configurations; }
|
||||
void populateTree();
|
||||
@ -71,6 +71,7 @@ private slots:
|
||||
void on_configurationLoad_clicked();
|
||||
void on_configurationTree_currentItemChanged(QTreeWidgetItem *current, QTreeWidgetItem *previous);
|
||||
void on_configurationTree_itemActivated(QTreeWidgetItem *item, int column);
|
||||
void accept() override;
|
||||
|
||||
signals:
|
||||
void saveConfiguration(Configuration*);
|
||||
|
@ -6,8 +6,8 @@
|
||||
<rect>
|
||||
<x>0</x>
|
||||
<y>0</y>
|
||||
<width>392</width>
|
||||
<height>414</height>
|
||||
<width>410</width>
|
||||
<height>424</height>
|
||||
</rect>
|
||||
</property>
|
||||
<property name="font">
|
||||
@ -19,218 +19,207 @@
|
||||
<property name="windowTitle">
|
||||
<string>Configurations</string>
|
||||
</property>
|
||||
<widget class="QDialogButtonBox" name="buttonBox">
|
||||
<property name="geometry">
|
||||
<rect>
|
||||
<x>40</x>
|
||||
<y>380</y>
|
||||
<width>341</width>
|
||||
<height>32</height>
|
||||
</rect>
|
||||
</property>
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="standardButtons">
|
||||
<set>QDialogButtonBox::Close</set>
|
||||
</property>
|
||||
</widget>
|
||||
<widget class="QWidget" name="widget" native="true">
|
||||
<property name="geometry">
|
||||
<rect>
|
||||
<x>0</x>
|
||||
<y>10</y>
|
||||
<width>392</width>
|
||||
<height>310</height>
|
||||
</rect>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout">
|
||||
<item>
|
||||
<widget class="QTreeWidget" name="configurationsTree">
|
||||
<property name="toolTip">
|
||||
<string>List of configurations</string>
|
||||
</property>
|
||||
<property name="indentation">
|
||||
<number>5</number>
|
||||
</property>
|
||||
<property name="allColumnsShowFocus">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<property name="columnCount">
|
||||
<number>1</number>
|
||||
</property>
|
||||
<attribute name="headerMinimumSectionSize">
|
||||
<number>5</number>
|
||||
</attribute>
|
||||
<column>
|
||||
<property name="text">
|
||||
<string>Description</string>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_2">
|
||||
<item>
|
||||
<widget class="QWidget" name="widget" native="true">
|
||||
<layout class="QVBoxLayout" name="verticalLayout">
|
||||
<item>
|
||||
<widget class="QLabel" name="description">
|
||||
<property name="text">
|
||||
<string>Select configuration to load:</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QTreeWidget" name="configurationsTree">
|
||||
<property name="toolTip">
|
||||
<string>List of configurations</string>
|
||||
</property>
|
||||
<property name="indentation">
|
||||
<number>5</number>
|
||||
</property>
|
||||
<property name="allColumnsShowFocus">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<property name="columnCount">
|
||||
<number>1</number>
|
||||
</property>
|
||||
<attribute name="headerMinimumSectionSize">
|
||||
<number>5</number>
|
||||
</attribute>
|
||||
<column>
|
||||
<property name="text">
|
||||
<string>Description</string>
|
||||
</property>
|
||||
</column>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<layout class="QHBoxLayout" name="configurationsControl">
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationSave">
|
||||
<property name="toolTip">
|
||||
<string>Save current workspaces configuration as new configuration</string>
|
||||
</property>
|
||||
</column>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<widget class="QWidget" name="layoutWidget">
|
||||
<property name="geometry">
|
||||
<rect>
|
||||
<x>0</x>
|
||||
<y>330</y>
|
||||
<width>392</width>
|
||||
<height>34</height>
|
||||
</rect>
|
||||
</property>
|
||||
<layout class="QHBoxLayout" name="configurationsControl">
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationSave">
|
||||
<property name="toolTip">
|
||||
<string>Save current workspaces configuration as new configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/create.png</normaloff>:/create.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationUpdate">
|
||||
<property name="toolTip">
|
||||
<string>Update selected configuration with current workspaces configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/recycle.png</normaloff>:/recycle.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationEdit">
|
||||
<property name="toolTip">
|
||||
<string>Edit configuration details</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string/>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/edit.png</normaloff>:/edit.png</iconset>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationExport">
|
||||
<property name="toolTip">
|
||||
<string>Export current configuration to file</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string/>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/export.png</normaloff>:/export.png</iconset>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationImport">
|
||||
<property name="toolTip">
|
||||
<string>Import configuration from file into current group</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string/>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/import.png</normaloff>:/import.png</iconset>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="horizontalSpacer">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationDelete">
|
||||
<property name="toolTip">
|
||||
<string>Delete selected configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/bin.png</normaloff>:/bin.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="horizontalSpacer_2">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationLoad">
|
||||
<property name="toolTip">
|
||||
<string>Load selected configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/load.png</normaloff>:/load.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/create.png</normaloff>:/create.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationUpdate">
|
||||
<property name="toolTip">
|
||||
<string>Update selected configuration with current workspaces configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/recycle.png</normaloff>:/recycle.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationEdit">
|
||||
<property name="toolTip">
|
||||
<string>Edit configuration details</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string/>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/edit.png</normaloff>:/edit.png</iconset>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationExport">
|
||||
<property name="toolTip">
|
||||
<string>Export current configuration to file</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string/>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/export.png</normaloff>:/export.png</iconset>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationImport">
|
||||
<property name="toolTip">
|
||||
<string>Import configuration from file into current group</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string/>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/import.png</normaloff>:/import.png</iconset>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="horizontalSpacer">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationDelete">
|
||||
<property name="toolTip">
|
||||
<string>Delete selected configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/bin.png</normaloff>:/bin.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="horizontalSpacer_2">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>40</width>
|
||||
<height>20</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QToolButton" name="configurationLoad">
|
||||
<property name="toolTip">
|
||||
<string>Load selected configuration</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>...</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="../resources/res.qrc">
|
||||
<normaloff>:/load.png</normaloff>:/load.png</iconset>
|
||||
</property>
|
||||
<property name="iconSize">
|
||||
<size>
|
||||
<width>16</width>
|
||||
<height>16</height>
|
||||
</size>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QDialogButtonBox" name="buttonBox">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="standardButtons">
|
||||
<set>QDialogButtonBox::Close</set>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<resources>
|
||||
<include location="../resources/res.qrc"/>
|
||||
|
@ -292,8 +292,7 @@ void DeviceSetPresetsDialog::on_presetExport_clicked()
|
||||
tr("Open preset export file"),
|
||||
".",
|
||||
tr("Preset export files (*.prex)"),
|
||||
0,
|
||||
QFileDialog::DontUseNativeDialog
|
||||
0
|
||||
);
|
||||
|
||||
if (fileName != "")
|
||||
@ -342,8 +341,7 @@ void DeviceSetPresetsDialog::on_presetImport_clicked()
|
||||
tr("Open preset export file"),
|
||||
".",
|
||||
tr("Preset export files (*.prex)"),
|
||||
0,
|
||||
QFileDialog::DontUseNativeDialog
|
||||
0
|
||||
);
|
||||
|
||||
if (fileName != "")
|
||||
|
@ -7,7 +7,7 @@
|
||||
<x>0</x>
|
||||
<y>0</y>
|
||||
<width>324</width>
|
||||
<height>229</height>
|
||||
<height>201</height>
|
||||
</rect>
|
||||
</property>
|
||||
<property name="font">
|
||||
@ -19,44 +19,24 @@
|
||||
<property name="windowTitle">
|
||||
<string>My Position</string>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout">
|
||||
<item>
|
||||
<layout class="QGridLayout" name="gridLayout">
|
||||
<item row="1" column="0">
|
||||
<widget class="QDialogButtonBox" name="buttonBox">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
<property name="standardButtons">
|
||||
<set>QDialogButtonBox::Cancel|QDialogButtonBox::Ok</set>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item row="0" column="0">
|
||||
<widget class="QGroupBox" name="groupBox">
|
||||
<property name="title">
|
||||
<string>My Station Position</string>
|
||||
</property>
|
||||
<layout class="QFormLayout" name="formLayout">
|
||||
<item row="2" column="0">
|
||||
<widget class="QLabel" name="latitudeLabel">
|
||||
<property name="text">
|
||||
<string>Latitude</string>
|
||||