mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-05 02:30:44 +00:00
Merge branch 'master' into translations
This commit is contained in:
commit
fffa7a1383
28
Building/Linux/copyright
Normal file
28
Building/Linux/copyright
Normal file
@ -0,0 +1,28 @@
|
||||
Format: https://www.debian.org/doc/packaging-manuals/copyright-format/1.0/
|
||||
Upstream-Name: OSCAR
|
||||
|
||||
Files: *
|
||||
Copyright: 2011-2018 Mark Watkins
|
||||
2019-2020 The OSCAR Team
|
||||
License: GPL-3+
|
||||
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 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 for more
|
||||
details.
|
||||
.
|
||||
You should have received a copy of the GNU General Public
|
||||
License along with this package; if not, write to the Free
|
||||
Software Foundation, Inc., 51 Franklin St, Fifth Floor,
|
||||
Boston, MA 02110-1301 USA
|
||||
.
|
||||
On Debian systems, the full text of the GNU General Public
|
||||
License version 3 can be found in the file
|
||||
`/usr/share/common-licenses/GPL-3'.
|
||||
|
@ -36,11 +36,13 @@ echo Version: ${VERSION}
|
||||
|
||||
# application name
|
||||
appli_name="OSCAR"
|
||||
package_name="oscar"
|
||||
pre_inst="tst_user.sh"
|
||||
# build folder (absolute path is better)
|
||||
build_folder="/home/$USER/OSCAR/build"
|
||||
if [[ -n ${PRERELEASE} && -z ${RC} ]] ; then
|
||||
appli_name=${appli_name}-test
|
||||
package_name=${package_name}-test
|
||||
post_inst="ln_usrbin-test.sh"
|
||||
pre_rem="rm_usrbin-test.sh"
|
||||
post_rem="clean_rm-test.sh"
|
||||
@ -98,7 +100,7 @@ mkdir ${temp_folder}/bin
|
||||
mkdir ${temp_folder}/share
|
||||
mkdir ${temp_folder}/share/${appli_name}
|
||||
mkdir ${temp_folder}/share/doc
|
||||
share_doc_folder="${temp_folder}/share/doc/${appli_name}"
|
||||
share_doc_folder="${temp_folder}/share/doc/${package_name}"
|
||||
mkdir ${share_doc_folder}
|
||||
mkdir ${temp_folder}/share/icons
|
||||
mkdir ${temp_folder}/share/icons/hicolor
|
||||
@ -121,22 +123,29 @@ cp ./${appli_name}.png ${temp_folder}/share/icons/hicolor/48x48/apps/${appli_nam
|
||||
cp ./${appli_name}.svg ${temp_folder}/share/icons/hicolor/scalable/apps/${appli_name}.svg
|
||||
cp ./${appli_name}.desktop ${temp_folder}/share/applications/${appli_name}.desktop
|
||||
|
||||
echo "Copyright 2019-2020 oscar-team.org <oscar@oscar-team.org>" > $share_doc_folder/copyright
|
||||
#echo "Copyright 2019-2020 oscar-team.org <oscar@oscar-team.org>" > $share_doc_folder/copyright
|
||||
#echo "Licensed under /usr/share/common-licenses/GPL-3" >> $share_doc_folder/copyright
|
||||
cp ./copyright $share_doc_folder/copyright
|
||||
|
||||
changelog_file="$share_doc_folder/changelog"
|
||||
changelog_file="./changelog"
|
||||
|
||||
#automatic changelog as a bad name
|
||||
# need to generate one and say fpm to use it instead of create one
|
||||
# it seems that it needs both of them...
|
||||
|
||||
# creation of the changelog.Debian.gz
|
||||
# creation of the Debian changelog
|
||||
echo "$appli_name (${VERSION}-${ITERATION}) whatever; urgency=medium" > $changelog_file
|
||||
echo "" >> $changelog_file
|
||||
echo " * Package created with FPM." >> $changelog_file
|
||||
echo "" >> $changelog_file
|
||||
echo " -- oscar-team.org <oscar@oscar-team.org>" >> $changelog_file
|
||||
gzip --best $changelog_file
|
||||
description='Open Source CPAP Analysis Reporter\n<extended description needed to be filled with the right value>'
|
||||
echo " * See the Release Notes under Help/About menu" >> $changelog_file
|
||||
echo "" >> $changelog_file
|
||||
echo -n " -- oscar-team.org <oscar@oscar-team.org> " >> $changelog_file
|
||||
date -Iminutes >> $changelog_file
|
||||
cp $changelog_file $share_doc_folder/changelog
|
||||
gzip --best $share_doc_folder/changelog
|
||||
|
||||
description='Open Source CPAP Analysis Reporter\nProvides graphical and statistical display of the CPAP stored data'
|
||||
# trick for dummies : need to use echo -e to take care of \n (cariage return to slip description and extra one
|
||||
description=$(echo -e $description)
|
||||
|
||||
@ -154,7 +163,6 @@ fpm --input-type dir --output-type deb \
|
||||
--category misc \
|
||||
--deb-priority optional \
|
||||
--maintainer " -- oscar-team.org <oscar@oscar-team.org>" \
|
||||
--license GPL-v3 \
|
||||
--vendor oscar-team.org \
|
||||
--description "$description" \
|
||||
--url https://sleepfiles.com/OSCAR \
|
||||
@ -162,7 +170,7 @@ fpm --input-type dir --output-type deb \
|
||||
--depends $dblPkg \
|
||||
--depends libpcre16-3 \
|
||||
--depends qttranslations5-l10n \
|
||||
--depends "libqt5core5a > $qtver" \
|
||||
--depends "libqt5core5a > 5.9" \
|
||||
--depends libqt5serialport5 \
|
||||
--depends libqt5xml5 \
|
||||
--depends libqt5network5 \
|
||||
|
@ -37,6 +37,4 @@ if [ -f "$file" ]; then
|
||||
rm $file
|
||||
fi
|
||||
|
||||
if [ -x /usr/bin/update-menus ]; then update-menus; fi
|
||||
|
||||
|
||||
|
@ -90,6 +90,5 @@ if [ -f "$file" ]; then
|
||||
rm $file
|
||||
fi
|
||||
|
||||
if [ -x /usr/bin/update-menus ]; then update-menus; fi
|
||||
|
||||
|
||||
|
@ -1,10 +1,10 @@
|
||||
#! /bin/bash
|
||||
#set -e
|
||||
set -e
|
||||
#
|
||||
#test USER (must be root) vs SUDO_USER (must be other than root but not empty)
|
||||
#if [ "$USER" != "root" ] || [ "$SUDO_USER" = "root" ] || [ -z "$SUDO_USER" ]; then
|
||||
# echo "dpkg -i must be launched as normal user with sudo command. fatal error"
|
||||
# exit
|
||||
#fi
|
||||
if [ "$USER" != "root" ] || [ "$SUDO_USER" = "root" ] || [ -z "$SUDO_USER" ]; then
|
||||
echo "Installation must be done as normal user with sudo command. fatal error"
|
||||
exit
|
||||
fi
|
||||
# do nothing
|
||||
|
||||
|
@ -8,10 +8,27 @@
|
||||
</head>
|
||||
<body>
|
||||
<p>
|
||||
<<<<<<< HEAD
|
||||
<b>For other languages, go to:</b>
|
||||
<br>http://www.apneaboard.com/wiki/index.php/OSCAR_Release_Notes</p>
|
||||
<p>
|
||||
<b>Changes and fixes in OSCAR v1.2.0</b>
|
||||
=======
|
||||
<b>Changes and fixes in OSCAR v1.2.0-beta-2</b>
|
||||
<br>Portions of OSCAR are © 2019-2020 by
|
||||
<i>The OSCAR Team</i></p>
|
||||
<ul>
|
||||
<li>[fix] Support migration from both SleepyHead and OSCAR.</li>
|
||||
<li>[fix] Correct 95th percentile computations on Statistics page when some days were summary-only.</li>
|
||||
<li>[fix] Improve some prompts and tooltip messages.</li>
|
||||
<li>[fix] Improve support of rare Philips Respironics 950P events and update warnings.</li>
|
||||
<li>[fix] Empty directories in Profiles directory are not shown in Profiles list.</li>
|
||||
<li>[fix] Show correct copyright when installing Debian version.</li>
|
||||
<li>[fix] Improvements to Chromebook identification.</li>
|
||||
</ul></p>
|
||||
<p>
|
||||
<b>Changes and fixes in OSCAR v1.2.0-beta-1</b>
|
||||
>>>>>>> e39adbd41ec60e2787e34a840afdfa1114baba19
|
||||
<br>Portions of OSCAR are © 2019-2020 by
|
||||
<i>The OSCAR Team</i></p> <ul>
|
||||
<li>[new] Support for Chromebooks using Linux beta - (Intel/AMD only)</li>
|
||||
|
@ -1147,6 +1147,9 @@ void Scan()
|
||||
// Iterate through subdirectories and load profiles..
|
||||
for (auto & fi : list) {
|
||||
QString npath = fi.canonicalFilePath();
|
||||
QDir profilePath(npath);
|
||||
if (profilePath.isEmpty()) // skip any empty folders
|
||||
continue;
|
||||
Profile *prof = new Profile(npath);
|
||||
//prof->Open();
|
||||
|
||||
@ -1724,69 +1727,70 @@ EventDataType Profile::calcPercentile(ChannelID code, EventDataType percent, Mac
|
||||
|
||||
qint64 SN = 0;
|
||||
bool timeweight;
|
||||
bool summaryOnly = false;
|
||||
|
||||
bool summaryOnly = true;
|
||||
do {
|
||||
Day *day = GetGoodDay(date, mt);
|
||||
|
||||
if (day) {
|
||||
if (day->summaryOnly()) {
|
||||
summaryOnly = true;
|
||||
break;
|
||||
date = date.addDays(1);
|
||||
continue;
|
||||
}
|
||||
|
||||
summaryOnly = false;
|
||||
|
||||
// why was this nested like this???
|
||||
//for (int i = 0; i < day->size(); i++) {
|
||||
for (auto & sess : day->sessions) {
|
||||
if (!sess->enabled()) {
|
||||
continue;
|
||||
}
|
||||
for (auto & sess : day->sessions) {
|
||||
if (!sess->enabled()) {
|
||||
continue;
|
||||
}
|
||||
|
||||
gain = sess->m_gain[code];
|
||||
gain = sess->m_gain[code];
|
||||
|
||||
if (!gain) { gain = 1; }
|
||||
if (!gain) { gain = 1; }
|
||||
|
||||
vsi = sess->m_valuesummary.find(code);
|
||||
vsi = sess->m_valuesummary.find(code);
|
||||
|
||||
if (vsi == sess->m_valuesummary.end()) { continue; }
|
||||
if (vsi == sess->m_valuesummary.end()) { continue; }
|
||||
|
||||
tsi = sess->m_timesummary.find(code);
|
||||
timeweight = (tsi != sess->m_timesummary.end());
|
||||
tsi = sess->m_timesummary.find(code);
|
||||
timeweight = (tsi != sess->m_timesummary.end());
|
||||
|
||||
QHash<EventStoreType, EventStoreType> &vsum = vsi.value();
|
||||
QHash<EventStoreType, quint32> &tsum = tsi.value();
|
||||
QHash<EventStoreType, EventStoreType> &vsum = vsi.value();
|
||||
QHash<EventStoreType, quint32> &tsum = tsi.value();
|
||||
|
||||
if (timeweight) {
|
||||
for (auto k=tsum.begin(), tsumend=tsum.end(); k != tsumend; k++) {
|
||||
weight = k.value();
|
||||
value = EventDataType(k.key()) * gain;
|
||||
if (timeweight) {
|
||||
for (auto k=tsum.begin(), tsumend=tsum.end(); k != tsumend; k++) {
|
||||
weight = k.value();
|
||||
value = EventDataType(k.key()) * gain;
|
||||
|
||||
SN += weight;
|
||||
wmi = wmap.find(value);
|
||||
SN += weight;
|
||||
wmi = wmap.find(value);
|
||||
|
||||
if (wmi == wmap.end()) {
|
||||
wmap[value] = weight;
|
||||
} else {
|
||||
wmi.value() += weight;
|
||||
}
|
||||
if (wmi == wmap.end()) {
|
||||
wmap[value] = weight;
|
||||
} else {
|
||||
wmi.value() += weight;
|
||||
}
|
||||
} else {
|
||||
for (auto k=vsum.begin(), vsumend=vsum.end(); k!=vsumend; k++) {
|
||||
weight = k.value();
|
||||
value = EventDataType(k.key()) * gain;
|
||||
}
|
||||
} else {
|
||||
for (auto k=vsum.begin(), vsumend=vsum.end(); k!=vsumend; k++) {
|
||||
weight = k.value();
|
||||
value = EventDataType(k.key()) * gain;
|
||||
|
||||
SN += weight;
|
||||
wmi = wmap.find(value);
|
||||
SN += weight;
|
||||
wmi = wmap.find(value);
|
||||
|
||||
if (wmi == wmap.end()) {
|
||||
wmap[value] = weight;
|
||||
} else {
|
||||
wmi.value() += weight;
|
||||
}
|
||||
if (wmi == wmap.end()) {
|
||||
wmap[value] = weight;
|
||||
} else {
|
||||
wmi.value() += weight;
|
||||
}
|
||||
}
|
||||
}
|
||||
// }
|
||||
}
|
||||
// }
|
||||
}
|
||||
|
||||
date = date.addDays(1);
|
||||
|
@ -340,8 +340,8 @@ void init()
|
||||
// <channel id="0x111e" class="data" name="TestChan1" details="Debugging Channel #1" label="Test #1" unit="" color="pink"/>
|
||||
// <channel id="0x111f" class="data" name="TestChan2" details="Debugging Channel #2" label="Test #2" unit="" color="blue"/>
|
||||
|
||||
schema::channel.add(GRP_CPAP, ch=new Channel(CPAP_Test1 = 0x111e, DATA, MT_CPAP, SESSION, STR_GRAPH_TestChan1, QObject::tr("Debugging channel #1"), QObject::tr("Top secret internal stuff you're not supposed to see ;)"), QObject::tr("Test #1"), QString(), INTEGER, QColor("pink")));
|
||||
schema::channel.add(GRP_CPAP, ch=new Channel(CPAP_Test2 = 0x111f, DATA, MT_CPAP, SESSION, STR_GRAPH_TestChan2, QObject::tr("Debugging channel #2"), QObject::tr("Top secret internal stuff you're not supposed to see ;)"), QObject::tr("Test #2"), QString(), INTEGER, Qt::blue));
|
||||
schema::channel.add(GRP_CPAP, ch=new Channel(CPAP_Test1 = 0x111e, DATA, MT_CPAP, SESSION, STR_GRAPH_TestChan1, QObject::tr("Debugging channel #1"), QObject::tr("For internal use only"), QObject::tr("Test #1"), QString(), INTEGER, QColor("pink")));
|
||||
schema::channel.add(GRP_CPAP, ch=new Channel(CPAP_Test2 = 0x111f, DATA, MT_CPAP, SESSION, STR_GRAPH_TestChan2, QObject::tr("Debugging channel #2"), QObject::tr("For internal use only"), QObject::tr("Test #2"), QString(), INTEGER, Qt::blue));
|
||||
|
||||
RMS9_E01 = schema::channel["RMS9_E01"].id();
|
||||
RMS9_E02 = schema::channel["RMS9_E02"].id();
|
||||
|
@ -120,9 +120,12 @@ bool processPreferenceFile( QString path ) {
|
||||
tmp.open(QIODevice::WriteOnly);
|
||||
QTextStream instr(&fl);
|
||||
QTextStream outstr(&tmp);
|
||||
bool isSleepyHead = false;
|
||||
while (instr.readLineInto(&line)) {
|
||||
if (line.contains("<SleepyHead>")) // Is this SleepyHead or OSCAR preferences file?
|
||||
isSleepyHead = true;
|
||||
line.replace("SleepyHead","OSCAR");
|
||||
if (line.contains("VersionString")) {
|
||||
if (isSleepyHead && line.contains("VersionString")) {
|
||||
int rtAngle = line.indexOf(">", 0);
|
||||
int lfAngle = line.indexOf("<", rtAngle);
|
||||
line.replace(rtAngle+1, lfAngle-rtAngle-1, "1.0.0-beta");
|
||||
@ -185,7 +188,7 @@ bool migrateFromSH(QString destDir) {
|
||||
|
||||
while (selectingFolder) {
|
||||
datadir = QFileDialog::getExistingDirectory(nullptr,
|
||||
QObject::tr("Choose the SleepyHead data folder to migrate")+" "+
|
||||
QObject::tr("Choose the SleepyHead or OSCAR data folder to migrate")+" "+
|
||||
QObject::tr("or CANCEL to skip migration."),
|
||||
homeDocs, QFileDialog::ShowDirsOnly);
|
||||
qDebug() << "Migration folder selected: " + datadir;
|
||||
@ -200,7 +203,7 @@ bool migrateFromSH(QString destDir) {
|
||||
if (!file.exists() || !dirP.exists()) { // It doesn't have a Preferences.xml file or a Profiles directory in it
|
||||
// Not a new directory.. nag the user.
|
||||
QMessageBox::warning(nullptr, STR_MessageBox_Error,
|
||||
QObject::tr("The folder you chose does not contain valid SleepyHead data.") +
|
||||
QObject::tr("The folder you chose does not contain valid SleepyHead or OSCAR data.") +
|
||||
"\n\n"+QObject::tr("You cannot use this folder:")+" " + datadir,
|
||||
QMessageBox::Ok);
|
||||
continue; // Nope, don't use it, go around the loop again
|
||||
@ -486,7 +489,8 @@ int main(int argc, char *argv[]) {
|
||||
if ( ! force_data_dir ) { // unless they explicitly selected it by --datadir param
|
||||
if (QMessageBox::question(nullptr, STR_MessageBox_Question,
|
||||
QObject::tr("OSCAR will set up a folder for your data.")+"\n"+
|
||||
QObject::tr("If you have been using SleepyHead, OSCAR can copy your old data to this folder later.")+"\n"+
|
||||
QObject::tr("If you have been using SleepyHead or an older version of OSCAR,") + "\n" +
|
||||
QObject::tr("OSCAR can copy your old data to this folder later.")+"\n"+
|
||||
QObject::tr("We suggest you use this folder: ")+QDir::toNativeSeparators(GetAppData())+"\n"+
|
||||
QObject::tr("Click Ok to accept this, or No if you want to use a different folder."),
|
||||
QMessageBox::Ok | QMessageBox::No, QMessageBox::Ok) == QMessageBox::No) {
|
||||
@ -539,9 +543,9 @@ int main(int argc, char *argv[]) {
|
||||
#else
|
||||
if ( ! newDir.exists() || newDir.isEmpty() ) { // directory doesn't exist yet or is empty, try to migrate old data
|
||||
#endif
|
||||
if (QMessageBox::question(nullptr, QObject::tr("Migrate SleepyHead Data?"),
|
||||
QObject::tr("On the next screen OSCAR will ask you to select a folder with SleepyHead data") +"\n" +
|
||||
QObject::tr("Click [OK] to go to the next screen or [No] if you do not wish to use any SleepyHead data."),
|
||||
if (QMessageBox::question(nullptr, QObject::tr("Migrate SleepyHead or OSCAR Data?"),
|
||||
QObject::tr("On the next screen OSCAR will ask you to select a folder with SleepyHead or OSCAR data") +"\n" +
|
||||
QObject::tr("Click [OK] to go to the next screen or [No] if you do not wish to use any SleepyHead or OSCAR data."),
|
||||
QMessageBox::Ok|QMessageBox::No, QMessageBox::Ok) == QMessageBox::Ok) {
|
||||
migrateFromSH( GetAppData() ); // doesn't matter if no migration
|
||||
}
|
||||
|
@ -811,15 +811,18 @@ void MainWindow::importCPAPBackups()
|
||||
QStringList getDriveList()
|
||||
{
|
||||
QStringList drivelist;
|
||||
bool crostini_detected = false;
|
||||
|
||||
#if QT_VERSION >= QT_VERSION_CHECK(5,4,0)
|
||||
#if defined(Q_OS_LINUX)
|
||||
#define VFAT "vfat"
|
||||
#elif defined(Q_OS_WIN)
|
||||
#define VFAT "FAT32"
|
||||
Q_UNUSED(crostini_detected)
|
||||
#elif defined(Q_OS_MAC)
|
||||
#define VFAT "msdos"
|
||||
#endif
|
||||
Q_UNUSED(crostini_detected)
|
||||
#endif
|
||||
foreach (const QStorageInfo &storage, QStorageInfo::mountedVolumes()) {
|
||||
if (storage.isValid() && storage.isReady()) {
|
||||
#ifdef DEBUG_SDCARD
|
||||
@ -833,20 +836,31 @@ QStringList getDriveList()
|
||||
if (storage.fileSystemType() == VFAT) {
|
||||
qDebug() << "Adding" << storage.name() << "on" << storage.rootPath() << "to drivelist";
|
||||
drivelist.append(storage.rootPath());
|
||||
} else if (storage.fileSystemType() == "9p") {
|
||||
qDebug() << "Crostini filesystem 9p found";
|
||||
crostini_detected = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
#endif
|
||||
#if defined(Q_OS_LINUX)
|
||||
QString mntName("/mnt/chromeos/removable");
|
||||
QDir mnt(mntName);
|
||||
qDebug() << "Checking for" << mntName;
|
||||
if ( mnt.exists() ) {
|
||||
qDebug() << "Checking Crostini removable folders";
|
||||
QFileInfoList mntPts = mnt.entryInfoList();
|
||||
foreach ( const auto dir, mntPts ) {
|
||||
qDebug() << "Adding" << dir.filePath() << "to drivelist";
|
||||
drivelist.append(dir.filePath() );
|
||||
if (crostini_detected) {
|
||||
QString mntName("/mnt/chromeos/removable");
|
||||
QDir mnt(mntName);
|
||||
qDebug() << "Checking for" << mntName;
|
||||
if ( mnt.exists() ) {
|
||||
qDebug() << "Checking Crostini removable folders";
|
||||
QFileInfoList mntPts = mnt.entryInfoList();
|
||||
foreach ( const auto dir, mntPts ) {
|
||||
qDebug() << "Adding" << dir.filePath() << "to drivelist";
|
||||
drivelist.append(dir.filePath() );
|
||||
}
|
||||
} else {
|
||||
QMessageBox::warning(nullptr, STR_MessageBox_Warning,
|
||||
QObject::tr("Chromebook file system detected, but no removable device found\n") +
|
||||
QObject::tr("You must share your SD card with Linux using the ChromeOS Files program"));
|
||||
drivelist.clear();
|
||||
drivelist.append("CROSTINI");
|
||||
}
|
||||
}
|
||||
#endif
|
||||
@ -903,6 +917,8 @@ QList<ImportPath> MainWindow::detectCPAPCards()
|
||||
do {
|
||||
// Rescan in case card inserted
|
||||
QStringList AutoScannerPaths = getDriveList();
|
||||
if (AutoScannerPaths.contains("CROSTINI")) // no Crostini removable drives found!
|
||||
break; // break out of the 20 second wait loop
|
||||
// AutoScannerPaths.push_back(lastpath);
|
||||
qDebug() << "Drive list size:" << AutoScannerPaths.size();
|
||||
|
||||
|
@ -40,7 +40,7 @@
|
||||
<item>
|
||||
<widget class="QStackedWidget" name="stackedWidget">
|
||||
<property name="currentIndex">
|
||||
<number>2</number>
|
||||
<number>1</number>
|
||||
</property>
|
||||
<widget class="QWidget" name="welcomePage">
|
||||
<layout class="QVBoxLayout" name="verticalLayout_8">
|
||||
@ -149,7 +149,7 @@
|
||||
<item>
|
||||
<widget class="QGroupBox" name="passwordGroupBox">
|
||||
<property name="toolTip">
|
||||
<string>Keep the kids out.. Nothing more.. This isn't meant to be uber security.</string>
|
||||
<string>Very weak password protection and not recommended if security is required.</string>
|
||||
</property>
|
||||
<property name="title">
|
||||
<string>Password Protect Profile</string>
|
||||
|
@ -258,7 +258,6 @@ SOURCES += \
|
||||
newprofile.cpp \
|
||||
overview.cpp \
|
||||
preferencesdialog.cpp \
|
||||
profileselect.cpp \
|
||||
reports.cpp \
|
||||
sessionbar.cpp \
|
||||
# updateparser.cpp \
|
||||
@ -337,7 +336,6 @@ HEADERS += \
|
||||
newprofile.h \
|
||||
overview.h \
|
||||
preferencesdialog.h \
|
||||
profileselect.h \
|
||||
reports.h \
|
||||
sessionbar.h \
|
||||
# updateparser.h \
|
||||
@ -415,7 +413,6 @@ FORMS += \
|
||||
mainwindow.ui \
|
||||
oximetry.ui \
|
||||
preferencesdialog.ui \
|
||||
profileselect.ui \
|
||||
newprofile.ui \
|
||||
exportcsv.ui \
|
||||
# UpdaterWindow.ui \
|
||||
|
@ -1,368 +0,0 @@
|
||||
/* Profile Select Implementation (Login Screen)
|
||||
*
|
||||
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU General Public
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#include "profileselect.h"
|
||||
#include <QDebug>
|
||||
#include <QStringListModel>
|
||||
#include <QStandardItem>
|
||||
#include <QDialog>
|
||||
#include <QLineEdit>
|
||||
#include <QVBoxLayout>
|
||||
#include <QCryptographicHash>
|
||||
#include <QMessageBox>
|
||||
#include <QHostInfo>
|
||||
#include <QTimer>
|
||||
#include <QFontMetrics>
|
||||
#include <QDir>
|
||||
#include "ui_profileselect.h"
|
||||
#include "SleepLib/profiles.h"
|
||||
#include "newprofile.h"
|
||||
#include "mainwindow.h"
|
||||
|
||||
extern MainWindow * mainwin;
|
||||
|
||||
ProfileSelect::ProfileSelect(QWidget *parent) :
|
||||
QDialog(parent),
|
||||
ui(new Ui::ProfileSelect)
|
||||
{
|
||||
ui->setupUi(this);
|
||||
QStringList str;
|
||||
model = new QStandardItemModel(0, 0);
|
||||
|
||||
int i = 0;
|
||||
int sel = -1;
|
||||
QString name;
|
||||
|
||||
QIcon icon(":/icons/moon.png");
|
||||
|
||||
int w=0;
|
||||
QFont font("Sans Serif", 18, QFont::Bold, false);
|
||||
ui->listView->setFont(font);
|
||||
|
||||
QFontMetrics fm(font);
|
||||
QMap<QString, Profile *>::iterator p;
|
||||
for (p = Profiles::profiles.begin(); p != Profiles::profiles.end(); p++) {
|
||||
name = p.key();
|
||||
|
||||
if (AppSetting->profileName() == name) {
|
||||
sel = i;
|
||||
}
|
||||
|
||||
QStandardItem *item = new QStandardItem(name);
|
||||
|
||||
item->setData(p.key(), Qt::UserRole+2);
|
||||
item->setEditable(false);
|
||||
|
||||
if (!(*p)->checkLock().isEmpty()) {
|
||||
item->setForeground(QBrush(Qt::red));
|
||||
item->setText(name+" (open)");
|
||||
}
|
||||
|
||||
QRect rect = fm.boundingRect(name);
|
||||
if (rect.width() > w) w = rect.width();
|
||||
|
||||
// Profile fonts arern't loaded yet.. Using generic font.
|
||||
item->setFont(font);
|
||||
model->appendRow(item);
|
||||
i++;
|
||||
}
|
||||
w+=20;
|
||||
ui->listView->setMinimumWidth(w);
|
||||
|
||||
proxy = new QSortFilterProxyModel(this);
|
||||
proxy->setSourceModel(model);
|
||||
proxy->setSortCaseSensitivity(Qt::CaseInsensitive);
|
||||
|
||||
ui->listView->setModel(proxy);
|
||||
ui->listView->setSelectionBehavior(QAbstractItemView::SelectRows);
|
||||
ui->listView->setSelectionMode(QAbstractItemView::SingleSelection);
|
||||
|
||||
if (sel >= 0) { ui->listView->setCurrentIndex(proxy->index(sel,0)); } //model->item(sel)->index()); }
|
||||
|
||||
proxy->sort(0, Qt::AscendingOrder);
|
||||
|
||||
m_tries = 0;
|
||||
|
||||
popupMenu = new QMenu(this);
|
||||
popupMenu->addAction(tr("Open Profile"), this, SLOT(openProfile()));
|
||||
popupMenu->addAction(tr("Edit Profile"), this, SLOT(editProfile()));
|
||||
popupMenu->addSeparator();
|
||||
popupMenu->addAction(tr("Delete Profile"), this, SLOT(deleteProfile()));
|
||||
|
||||
ui->labelAppName->setText(STR_TR_OSCAR);
|
||||
ui->labelVersion->setText("");
|
||||
// if (GIT_BRANCH!="master")
|
||||
// ui->labelBuild->setText(GIT_BRANCH);
|
||||
// else ui->labelBuild->setText(QString());
|
||||
ui->labelFolder->setText(GetAppData());
|
||||
ui->labelFolder->setToolTip("Current OSCAR data folder\n" + GetAppData());
|
||||
|
||||
ui->listView->verticalScrollBar()->setStyleSheet("QScrollBar:vertical {border: 0px solid grey; background: transparent; }"
|
||||
"QScrollBar::handle:vertical {"
|
||||
" background: qlineargradient(x1:0, y1:0, x2:1, y2:0,"
|
||||
" stop: 0 rgb(230, 230, 230), stop: 0.5 rgb(255, 255, 255), stop:1 rgb(230, 230, 230));"
|
||||
" min-height: 0px;"
|
||||
" border: 1px solid gray;"
|
||||
" border-radius: 5px;"
|
||||
"}");
|
||||
}
|
||||
|
||||
ProfileSelect::~ProfileSelect()
|
||||
{
|
||||
delete model; // why is this not being cleaned up by Qt?
|
||||
delete popupMenu;
|
||||
delete ui;
|
||||
}
|
||||
void ProfileSelect::earlyExit()
|
||||
{
|
||||
accept();
|
||||
}
|
||||
void ProfileSelect::editProfile()
|
||||
{
|
||||
QString name = ui->listView->currentIndex().data(Qt::UserRole+2).toString();
|
||||
Profile *profile = Profiles::Get(name);
|
||||
|
||||
if (!profile) { return; }
|
||||
|
||||
bool reallyEdit = false;
|
||||
|
||||
if (profile->user->hasPassword()) {
|
||||
QDialog dialog(this, Qt::Dialog);
|
||||
QLineEdit *e = new QLineEdit(&dialog);
|
||||
e->setEchoMode(QLineEdit::Password);
|
||||
dialog.connect(e, SIGNAL(returnPressed()), &dialog, SLOT(accept()));
|
||||
dialog.setWindowTitle(tr("Enter Password for %1").arg(name));
|
||||
dialog.setMinimumWidth(300);
|
||||
QVBoxLayout *lay = new QVBoxLayout();
|
||||
dialog.setLayout(lay);
|
||||
lay->addWidget(e);
|
||||
int tries = 0;
|
||||
|
||||
do {
|
||||
e->setText("");
|
||||
|
||||
if (dialog.exec() != QDialog::Accepted) { break; }
|
||||
|
||||
tries++;
|
||||
|
||||
if (profile->user->checkPassword(e->text())) {
|
||||
reallyEdit = true;
|
||||
break;
|
||||
} else {
|
||||
if (tries < 3) {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error, tr("Incorrect Password"), QMessageBox::Ok);
|
||||
} else {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error, tr("You entered the password wrong too many times."),
|
||||
QMessageBox::Ok);
|
||||
reject();
|
||||
}
|
||||
}
|
||||
} while (tries < 3);
|
||||
} else { reallyEdit = true; }
|
||||
|
||||
if (reallyEdit) {
|
||||
NewProfile newprof(this);
|
||||
newprof.edit(name);
|
||||
newprof.exec();
|
||||
}
|
||||
|
||||
//qDebug() << "edit" << name;
|
||||
}
|
||||
void ProfileSelect::deleteProfile()
|
||||
{
|
||||
QString name = ui->listView->currentIndex().data(Qt::UserRole+2).toString();
|
||||
|
||||
QDialog confirmdlg;
|
||||
QVBoxLayout layout(&confirmdlg);
|
||||
QLabel message(QString("<b>"+STR_MessageBox_Warning+":</b> "+tr("You are about to destroy profile '%1'.")+"<br/><br/>"+tr("Enter the word DELETE below to confirm.")).arg(name), &confirmdlg);
|
||||
layout.insertWidget(0,&message,1);
|
||||
QLineEdit lineedit(&confirmdlg);
|
||||
layout.insertWidget(1, &lineedit, 1);
|
||||
QHBoxLayout layout2;
|
||||
layout.insertLayout(2,&layout2,1);
|
||||
QPushButton cancel(QString("&Cancel"), &confirmdlg);
|
||||
QPushButton accept(QString("&Delete Profile"), &confirmdlg);
|
||||
layout2.addWidget(&cancel);
|
||||
layout2.addStretch(1);
|
||||
layout2.addWidget(&accept);
|
||||
confirmdlg.connect(&cancel, SIGNAL(clicked()), &confirmdlg, SLOT(reject()));
|
||||
confirmdlg.connect(&accept, SIGNAL(clicked()), &confirmdlg, SLOT(accept()));
|
||||
confirmdlg.connect(&lineedit, SIGNAL(returnPressed()), &confirmdlg, SLOT(accept()));
|
||||
|
||||
if (confirmdlg.exec() != QDialog::Accepted)
|
||||
return;
|
||||
|
||||
if (lineedit.text().compare("DELETE")!=0) {
|
||||
QMessageBox::information(NULL, tr("Sorry"), tr("You need to enter DELETE in capital letters."), QMessageBox::Ok);
|
||||
return;
|
||||
}
|
||||
|
||||
Profile * profile = Profiles::profiles[name];
|
||||
p_profile = profile;
|
||||
// Hmmmmm.....
|
||||
// if (!profile->Load()) {
|
||||
// QMessageBox::warning(this, STR_MessageBox_Error,
|
||||
// QString(tr("Could not open profile.. You will need to delete this profile directory manually")+
|
||||
// "\n\n"+tr("You will find it under the following location:")+"\n\n%1").arg(QDir::toNativeSeparators(GetAppData() + "/Profiles/" + profile->user->userName())), QMessageBox::Ok);
|
||||
// return;
|
||||
// }
|
||||
bool reallydelete = false;
|
||||
if (profile->user->hasPassword()) {
|
||||
QDialog dialog(this, Qt::Dialog);
|
||||
QLineEdit *e = new QLineEdit(&dialog);
|
||||
e->setEchoMode(QLineEdit::Password);
|
||||
dialog.connect(e, SIGNAL(returnPressed()), &dialog, SLOT(accept()));
|
||||
dialog.setWindowTitle(tr("Enter Password for %1").arg(name));
|
||||
dialog.setMinimumWidth(300);
|
||||
QVBoxLayout *lay = new QVBoxLayout();
|
||||
dialog.setLayout(lay);
|
||||
lay->addWidget(e);
|
||||
int tries = 0;
|
||||
|
||||
do {
|
||||
e->setText("");
|
||||
|
||||
if (dialog.exec() != QDialog::Accepted) { break; }
|
||||
|
||||
tries++;
|
||||
|
||||
if (profile->user->checkPassword(e->text())) {
|
||||
reallydelete = true;
|
||||
break;
|
||||
} else {
|
||||
if (tries < 3) {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error, tr("You entered an incorrect password"), QMessageBox::Ok);
|
||||
} else {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error,
|
||||
tr("If you're trying to delete because you forgot the password, you need to delete it manually."),
|
||||
QMessageBox::Ok);
|
||||
}
|
||||
}
|
||||
} while (tries < 3);
|
||||
} else { reallydelete = true; }
|
||||
|
||||
if (reallydelete) {
|
||||
QString path = profile->Get(PrefMacro(STR_GEN_DataFolder));
|
||||
|
||||
if (!path.isEmpty()) {
|
||||
if (!removeDir(path)) {
|
||||
QMessageBox::information(this, STR_MessageBox_Error,
|
||||
tr("There was an error deleting the profile directory, you need to manually remove it.")+QString("\n\n%1").arg(path),
|
||||
QMessageBox::Ok);
|
||||
}
|
||||
qDebug() << "Delete" << path;
|
||||
QMessageBox::information(this, STR_MessageBox_Information, tr("Profile '%1' was succesfully deleted").arg(name),QMessageBox::Ok);
|
||||
}
|
||||
|
||||
int row = ui->listView->currentIndex().row();
|
||||
proxy->removeRow(row);
|
||||
delete p_profile;
|
||||
p_profile = nullptr;
|
||||
}
|
||||
}
|
||||
|
||||
//! \fn ProfileSelect::QuickLogin()
|
||||
//! \brief For programmatically bypassing the login window
|
||||
void ProfileSelect::QuickLogin()
|
||||
{
|
||||
on_listView_activated(ui->listView->currentIndex());
|
||||
}
|
||||
|
||||
void ProfileSelect::on_selectButton_clicked()
|
||||
{
|
||||
on_listView_activated(ui->listView->currentIndex());
|
||||
}
|
||||
void ProfileSelect::openProfile()
|
||||
{
|
||||
on_listView_activated(ui->listView->currentIndex());
|
||||
}
|
||||
|
||||
void ProfileSelect::on_newProfileButton_clicked()
|
||||
{
|
||||
NewProfile newprof(this);
|
||||
newprof.skipWelcomeScreen();
|
||||
newprof.setWindowTitle(tr("Create new profile"));
|
||||
|
||||
if (newprof.exec() == NewProfile::Rejected) {
|
||||
// reject();
|
||||
} else { accept(); }
|
||||
}
|
||||
|
||||
|
||||
//! \fn ProfileSelect::on_listView_activated(const QModelIndex &index)
|
||||
//! \brief Process the selected login, requesting passwords if required.
|
||||
void ProfileSelect::on_listView_activated(const QModelIndex &index)
|
||||
{
|
||||
QString name = index.data(Qt::UserRole+2).toString();
|
||||
Profile *profile = Profiles::profiles[name];
|
||||
|
||||
if (!profile) { return; }
|
||||
if (!profile->isOpen()) {
|
||||
p_profile = profile;
|
||||
// Do this in case user renames the directory (otherwise it won't load)
|
||||
// Essentially makes the folder name the user name, but whatever..
|
||||
// TODO: Change the profile editor one day to make it rename the actual folder
|
||||
profile->p_preferences[STR_UI_UserName] = name;
|
||||
}
|
||||
|
||||
if (!profile->user->hasPassword()) {
|
||||
m_selectedProfile = name;
|
||||
AppSetting->setProfileName(name);
|
||||
accept();
|
||||
return;
|
||||
} else {
|
||||
int tries = 0;
|
||||
|
||||
do {
|
||||
QDialog dialog(this, Qt::Dialog);
|
||||
QLineEdit *e = new QLineEdit(&dialog);
|
||||
e->setEchoMode(QLineEdit::Password);
|
||||
dialog.connect(e, SIGNAL(returnPressed()), &dialog, SLOT(accept()));
|
||||
dialog.setWindowTitle(tr("Enter Password"));
|
||||
QVBoxLayout *lay = new QVBoxLayout();
|
||||
dialog.setLayout(lay);
|
||||
lay->addWidget(e);
|
||||
dialog.exec();
|
||||
|
||||
if (profile->user->checkPassword(e->text())) {
|
||||
m_selectedProfile = name;
|
||||
AppSetting->setProfileName(name);
|
||||
accept();
|
||||
return;
|
||||
}
|
||||
|
||||
tries++;
|
||||
|
||||
if (tries < 3) {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error, tr("Incorrect Password"), QMessageBox::Ok);
|
||||
} else {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error,
|
||||
tr("You entered an Incorrect Password too many times. Exiting!"), QMessageBox::Ok);
|
||||
}
|
||||
} while (tries < 3);
|
||||
}
|
||||
|
||||
reject();
|
||||
return;
|
||||
}
|
||||
|
||||
void ProfileSelect::on_listView_customContextMenuRequested(const QPoint &pos)
|
||||
{
|
||||
popupMenu->popup(QWidget::mapToGlobal(pos));
|
||||
}
|
||||
|
||||
void ProfileSelect::on_pushButton_clicked()
|
||||
{
|
||||
mainwin->RestartApplication(false, "-d");
|
||||
}
|
||||
|
||||
void ProfileSelect::on_filter_textChanged(const QString &arg1)
|
||||
{
|
||||
QRegExp regExp("*"+arg1+"*", Qt::CaseInsensitive, QRegExp::Wildcard);
|
||||
proxy->setFilterRegExp(regExp);
|
||||
}
|
@ -1,62 +0,0 @@
|
||||
/* Profile Select Header (Login Screen)
|
||||
*
|
||||
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
||||
*
|
||||
* This file is subject to the terms and conditions of the GNU General Public
|
||||
* License. See the file COPYING in the main directory of the source code
|
||||
* for more details. */
|
||||
|
||||
#ifndef PROFILESELECT_H
|
||||
#define PROFILESELECT_H
|
||||
|
||||
#include <QDialog>
|
||||
#include <QModelIndex>
|
||||
#include <QStandardItemModel>
|
||||
#include <QSortFilterProxyModel>
|
||||
#include <QMenu>
|
||||
|
||||
namespace Ui {
|
||||
class ProfileSelect;
|
||||
}
|
||||
|
||||
/*! \class ProfileSelect
|
||||
\brief Simple Login Window providing a list of all profiles to select from
|
||||
*/
|
||||
class ProfileSelect : public QDialog
|
||||
{
|
||||
Q_OBJECT
|
||||
|
||||
public:
|
||||
explicit ProfileSelect(QWidget *parent = 0);
|
||||
~ProfileSelect();
|
||||
|
||||
QString selectedProfile();
|
||||
void QuickLogin();
|
||||
private slots:
|
||||
void on_selectButton_clicked();
|
||||
|
||||
void on_newProfileButton_clicked();
|
||||
|
||||
void on_listView_activated(const QModelIndex &index);
|
||||
void earlyExit();
|
||||
|
||||
void openProfile();
|
||||
void editProfile();
|
||||
void deleteProfile();
|
||||
|
||||
void on_listView_customContextMenuRequested(const QPoint &pos);
|
||||
|
||||
void on_pushButton_clicked();
|
||||
|
||||
void on_filter_textChanged(const QString &arg1);
|
||||
|
||||
private:
|
||||
Ui::ProfileSelect *ui;
|
||||
QString m_selectedProfile;
|
||||
int m_tries;
|
||||
QMenu *popupMenu;
|
||||
QStandardItemModel *model;
|
||||
QSortFilterProxyModel *proxy;
|
||||
};
|
||||
|
||||
#endif // PROFILESELECT_H
|
@ -1,422 +0,0 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<ui version="4.0">
|
||||
<class>ProfileSelect</class>
|
||||
<widget class="QDialog" name="ProfileSelect">
|
||||
<property name="geometry">
|
||||
<rect>
|
||||
<x>0</x>
|
||||
<y>0</y>
|
||||
<width>418</width>
|
||||
<height>272</height>
|
||||
</rect>
|
||||
</property>
|
||||
<property name="windowTitle">
|
||||
<string>Select Profile</string>
|
||||
</property>
|
||||
<property name="windowIcon">
|
||||
<iconset resource="Resources.qrc">
|
||||
<normaloff>:/icons/logo-sm.png</normaloff>:/icons/logo-sm.png</iconset>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true">QDialog {
|
||||
background: qlineargradient(spread:pad, x1:0, y1:0, x2:1, y2:0, stop:0 rgba(240, 240, 240, 255), stop:1 rgba(220, 220, 220, 255));
|
||||
}
|
||||
|
||||
QGroupBox {
|
||||
background-color: white;
|
||||
border: 1px solid gray;
|
||||
border-radius: 5px;
|
||||
margin-top: 3ex; /* leave space at the top for the title */
|
||||
}
|
||||
|
||||
QGroupBox::title {
|
||||
subcontrol-origin: margin;
|
||||
subcontrol-position: top center; /* position at the top center */
|
||||
padding: 2px;
|
||||
background-color: white;
|
||||
}
|
||||
|
||||
QFrame {
|
||||
background: white;
|
||||
border: 1px solid gray;
|
||||
border-radius: 10px;
|
||||
}
|
||||
|
||||
QLabel {
|
||||
background: transparent;
|
||||
border: 0px;
|
||||
}
|
||||
|
||||
QPushButton {
|
||||
background: qlineargradient(spread:pad, x1:0, y1:0, x2:0, y2:1, stop:0 rgba(240, 240, 240, 255), stop:1 rgba(255, 255, 255, 255));
|
||||
border: 1px solid gray;
|
||||
border-radius: 10px;
|
||||
padding: 3px;
|
||||
padding-left: 20px;
|
||||
padding-right: 20px;
|
||||
}
|
||||
|
||||
QPushButton:hover {
|
||||
background: white;
|
||||
border: 2px solid gray;
|
||||
}
|
||||
|
||||
QPushButton:pressed {
|
||||
background: gray;
|
||||
}
|
||||
|
||||
QLineEdit {
|
||||
border-radius: 5px;
|
||||
border: 1px solid gray;
|
||||
background: white;
|
||||
}</string>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_3">
|
||||
<property name="spacing">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<item>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_2">
|
||||
<property name="spacing">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QFrame" name="frame_3">
|
||||
<property name="styleSheet">
|
||||
<string notr="true"/>
|
||||
</property>
|
||||
<property name="frameShape">
|
||||
<enum>QFrame::StyledPanel</enum>
|
||||
</property>
|
||||
<property name="frameShadow">
|
||||
<enum>QFrame::Raised</enum>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout">
|
||||
<item>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout">
|
||||
<item>
|
||||
<widget class="QLabel" name="label_2">
|
||||
<property name="styleSheet">
|
||||
<string notr="true">border: 0px;</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Search:</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QLineEdit" name="filter">
|
||||
<property name="styleSheet">
|
||||
<string notr="true"/>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QListView" name="listView">
|
||||
<property name="contextMenuPolicy">
|
||||
<enum>Qt::CustomContextMenu</enum>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true">border: 0px;</string>
|
||||
</property>
|
||||
<property name="frameShape">
|
||||
<enum>QFrame::NoFrame</enum>
|
||||
</property>
|
||||
<property name="sizeAdjustPolicy">
|
||||
<enum>QAbstractScrollArea::AdjustToContents</enum>
|
||||
</property>
|
||||
<property name="alternatingRowColors">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
<property name="verticalScrollMode">
|
||||
<enum>QAbstractItemView::ScrollPerPixel</enum>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QFrame" name="frame_2">
|
||||
<property name="styleSheet">
|
||||
<string notr="true">QFrame {
|
||||
background: qlineargradient(spread:pad, x1:0, y1:0, x2:1, y2:0, stop:0 rgba(255, 255, 255, 255), stop:1 rgba(220, 220, 220, 255));
|
||||
};</string>
|
||||
</property>
|
||||
<property name="frameShape">
|
||||
<enum>QFrame::StyledPanel</enum>
|
||||
</property>
|
||||
<property name="frameShadow">
|
||||
<enum>QFrame::Raised</enum>
|
||||
</property>
|
||||
<layout class="QVBoxLayout" name="verticalLayout_2">
|
||||
<property name="spacing">
|
||||
<number>6</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>16</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>16</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>16</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>16</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QPushButton" name="selectButton">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Minimum" vsizetype="Fixed">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="toolTip">
|
||||
<string>Start with the selected user profile.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>&Select User</string>
|
||||
</property>
|
||||
<property name="icon">
|
||||
<iconset resource="Resources.qrc">
|
||||
<normaloff>:/icons/forward.png</normaloff>:/icons/forward.png</iconset>
|
||||
</property>
|
||||
<property name="autoDefault">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="Line" name="line_2">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QPushButton" name="newProfileButton">
|
||||
<property name="toolTip">
|
||||
<string>Create a new user profile.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>New Profile</string>
|
||||
</property>
|
||||
<property name="autoDefault">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="Line" name="line">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Horizontal</enum>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QPushButton" name="pushButton">
|
||||
<property name="toolTip">
|
||||
<string>Choose a different OSCAR data folder.</string>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true"/>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>&Different Folder</string>
|
||||
</property>
|
||||
<property name="autoDefault">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="verticalSpacer_2">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>20</width>
|
||||
<height>40</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QLabel" name="labelAppName">
|
||||
<property name="font">
|
||||
<font>
|
||||
<pointsize>15</pointsize>
|
||||
<weight>75</weight>
|
||||
<bold>true</bold>
|
||||
</font>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true">border: 0px;</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>OSCAR</string>
|
||||
</property>
|
||||
<property name="alignment">
|
||||
<set>Qt::AlignCenter</set>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QLabel" name="labelVersion">
|
||||
<property name="styleSheet">
|
||||
<string notr="true">border: 0px;</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>[version]</string>
|
||||
</property>
|
||||
<property name="alignment">
|
||||
<set>Qt::AlignCenter</set>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<spacer name="verticalSpacer">
|
||||
<property name="orientation">
|
||||
<enum>Qt::Vertical</enum>
|
||||
</property>
|
||||
<property name="sizeHint" stdset="0">
|
||||
<size>
|
||||
<width>20</width>
|
||||
<height>40</height>
|
||||
</size>
|
||||
</property>
|
||||
</spacer>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QPushButton" name="quitButton">
|
||||
<property name="toolTip">
|
||||
<string>Click here if you didn't want to start OSCAR.</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>&Quit</string>
|
||||
</property>
|
||||
<property name="autoDefault">
|
||||
<bool>false</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QFrame" name="frame">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Preferred" vsizetype="Minimum">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true"/>
|
||||
</property>
|
||||
<property name="frameShape">
|
||||
<enum>QFrame::StyledPanel</enum>
|
||||
</property>
|
||||
<property name="frameShadow">
|
||||
<enum>QFrame::Plain</enum>
|
||||
</property>
|
||||
<layout class="QHBoxLayout" name="horizontalLayout_3">
|
||||
<property name="spacing">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="leftMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="topMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="rightMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<property name="bottomMargin">
|
||||
<number>4</number>
|
||||
</property>
|
||||
<item>
|
||||
<widget class="QLabel" name="label">
|
||||
<property name="font">
|
||||
<font>
|
||||
<weight>75</weight>
|
||||
<bold>true</bold>
|
||||
</font>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true">border: 0px;</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>Folder:</string>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
<item>
|
||||
<widget class="QLabel" name="labelFolder">
|
||||
<property name="sizePolicy">
|
||||
<sizepolicy hsizetype="Expanding" vsizetype="Preferred">
|
||||
<horstretch>0</horstretch>
|
||||
<verstretch>0</verstretch>
|
||||
</sizepolicy>
|
||||
</property>
|
||||
<property name="toolTip">
|
||||
<string>The current location of OSCAR data store.</string>
|
||||
</property>
|
||||
<property name="styleSheet">
|
||||
<string notr="true">border: 0px;</string>
|
||||
</property>
|
||||
<property name="text">
|
||||
<string>[data directory]</string>
|
||||
</property>
|
||||
<property name="wordWrap">
|
||||
<bool>true</bool>
|
||||
</property>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
</item>
|
||||
</layout>
|
||||
</widget>
|
||||
<resources>
|
||||
<include location="Resources.qrc"/>
|
||||
</resources>
|
||||
<connections>
|
||||
<connection>
|
||||
<sender>quitButton</sender>
|
||||
<signal>clicked()</signal>
|
||||
<receiver>ProfileSelect</receiver>
|
||||
<slot>reject()</slot>
|
||||
<hints>
|
||||
<hint type="sourcelabel">
|
||||
<x>52</x>
|
||||
<y>276</y>
|
||||
</hint>
|
||||
<hint type="destinationlabel">
|
||||
<x>199</x>
|
||||
<y>149</y>
|
||||
</hint>
|
||||
</hints>
|
||||
</connection>
|
||||
</connections>
|
||||
</ui>
|
@ -353,6 +353,10 @@ void ProfileSelector::on_buttonDestroyProfile_clicked()
|
||||
QString name = proxy->data(proxy->index(ui->profileView->currentIndex().row(), 0, QModelIndex()), Qt::UserRole+2).toString();
|
||||
Profile * profile = Profiles::profiles[name];
|
||||
QString path = profile->Get(PrefMacro(STR_GEN_DataFolder));
|
||||
if (path == (GetAppData() + "/Profiles/")) {
|
||||
QMessageBox::warning(this, STR_MessageBox_Error, tr("The selected profile does not appear to contain any data and cannot be removed by OSCAR"), QMessageBox::Ok);
|
||||
return;
|
||||
}
|
||||
|
||||
bool verified = true;
|
||||
if (profile->user->hasPassword()) {
|
||||
|
Loading…
Reference in New Issue
Block a user