mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-06 11:10:44 +00:00
Improve progress dialogs and show stage messages in ResMed & PRS1 importer
This commit is contained in:
parent
3f4e1b70d0
commit
adf6f1917d
@ -513,6 +513,9 @@ int PRS1Loader::OpenMachine(const QString & path)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Getting Ready..."));
|
||||||
|
QCoreApplication::processEvents();
|
||||||
|
|
||||||
dir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs | QDir::Files | QDir::Hidden | QDir::NoSymLinks);
|
dir.setFilter(QDir::NoDotAndDotDot | QDir::Dirs | QDir::Files | QDir::Hidden | QDir::NoSymLinks);
|
||||||
dir.setSorting(QDir::Name);
|
dir.setSorting(QDir::Name);
|
||||||
QFileInfoList flist = dir.entryInfoList();
|
QFileInfoList flist = dir.entryInfoList();
|
||||||
@ -626,6 +629,9 @@ int PRS1Loader::OpenMachine(const QString & path)
|
|||||||
PRS1Import * task = nullptr;
|
PRS1Import * task = nullptr;
|
||||||
// Note, I have observed p0/p1/etc folders containing duplicates session files (in Robin Sanders data.)
|
// Note, I have observed p0/p1/etc folders containing duplicates session files (in Robin Sanders data.)
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Scanning Files..."));
|
||||||
|
QCoreApplication::processEvents();
|
||||||
|
|
||||||
// for each p0/p1/p2/etc... folder
|
// for each p0/p1/p2/etc... folder
|
||||||
for (int p=0; p < size; ++p) {
|
for (int p=0; p < size; ++p) {
|
||||||
dir.setPath(paths.at(p));
|
dir.setPath(paths.at(p));
|
||||||
@ -738,7 +744,14 @@ int PRS1Loader::OpenMachine(const QString & path)
|
|||||||
int tasks = countTasks();
|
int tasks = countTasks();
|
||||||
unknownCodes.clear();
|
unknownCodes.clear();
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Importing Sessions..."));
|
||||||
|
QCoreApplication::processEvents();
|
||||||
|
|
||||||
runTasks(AppSetting->multithreading());
|
runTasks(AppSetting->multithreading());
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Finishing up..."));
|
||||||
|
QCoreApplication::processEvents();
|
||||||
|
|
||||||
finishAddingSessions();
|
finishAddingSessions();
|
||||||
|
|
||||||
if (unknownCodes.size() > 0) {
|
if (unknownCodes.size() > 0) {
|
||||||
|
@ -2301,6 +2301,9 @@ int ResmedLoader::Open(const QString & dirpath)
|
|||||||
}
|
}
|
||||||
MachineInfo info = newInfo();
|
MachineInfo info = newInfo();
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Parsing Identification File"));
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
// Parse # entries into idmap.
|
// Parse # entries into idmap.
|
||||||
while (!f.atEnd()) {
|
while (!f.atEnd()) {
|
||||||
line = f.readLine().trimmed();
|
line = f.readLine().trimmed();
|
||||||
@ -2397,7 +2400,11 @@ int ResmedLoader::Open(const QString & dirpath)
|
|||||||
|
|
||||||
resdayList.clear();
|
resdayList.clear();
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Locating STR.edf File(s)..."));
|
||||||
|
QCoreApplication::processEvents();
|
||||||
|
|
||||||
// List all STR.edf backups and tag on latest for processing
|
// List all STR.edf backups and tag on latest for processing
|
||||||
|
|
||||||
QMap<QDate, STRFile> STRmap;
|
QMap<QDate, STRFile> STRmap;
|
||||||
|
|
||||||
QDir dir;
|
QDir dir;
|
||||||
@ -2528,6 +2535,9 @@ int ResmedLoader::Open(const QString & dirpath)
|
|||||||
///////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////
|
||||||
// Build a Date map of all records in STR.edf files, populating ResDayList
|
// Build a Date map of all records in STR.edf files, populating ResDayList
|
||||||
///////////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////////
|
||||||
|
emit updateMessage(QObject::tr("Processing STR.edf File(s)..."));
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
ParseSTR(mach, STRmap);
|
ParseSTR(mach, STRmap);
|
||||||
|
|
||||||
// We are done with the Parsed STR EDF objects, so delete them
|
// We are done with the Parsed STR EDF objects, so delete them
|
||||||
@ -2567,11 +2577,17 @@ int ResmedLoader::Open(const QString & dirpath)
|
|||||||
QFile impfile(mach->getDataPath()+"/imported_files.csv");
|
QFile impfile(mach->getDataPath()+"/imported_files.csv");
|
||||||
if (impfile.exists()) impfile.remove();
|
if (impfile.exists()) impfile.remove();
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Searching for EDF Files..."));
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
scanFiles(mach, newpath);
|
scanFiles(mach, newpath);
|
||||||
|
|
||||||
// Now at this point we have resdayList populated with processable summary and EDF files data
|
// Now at this point we have resdayList populated with processable summary and EDF files data
|
||||||
// that can be processed in threads..
|
// that can be processed in threads..
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Queing Import Jobs..."));
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
for (auto rdi=resdayList.begin(), rend=resdayList.end(); rdi != rend; rdi++) {
|
for (auto rdi=resdayList.begin(), rend=resdayList.end(); rdi != rend; rdi++) {
|
||||||
QDate date = rdi.key();
|
QDate date = rdi.key();
|
||||||
|
|
||||||
@ -2619,6 +2635,7 @@ int ResmedLoader::Open(const QString & dirpath)
|
|||||||
}
|
}
|
||||||
|
|
||||||
sessionCount = 0;
|
sessionCount = 0;
|
||||||
|
emit updateMessage(QObject::tr("Importing Sessions..."));
|
||||||
runTasks();
|
runTasks();
|
||||||
int num_new_sessions = sessionCount;
|
int num_new_sessions = sessionCount;
|
||||||
|
|
||||||
@ -2627,6 +2644,8 @@ int ResmedLoader::Open(const QString & dirpath)
|
|||||||
// Now look for any new summary data that can be extracted from STR.edf records
|
// Now look for any new summary data that can be extracted from STR.edf records
|
||||||
////////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
emit updateMessage(QObject::tr("Finishing Up..."));
|
||||||
|
QApplication::processEvents();
|
||||||
|
|
||||||
//int size = m->sessionlist.size();
|
//int size = m->sessionlist.size();
|
||||||
//int cnt=0;
|
//int cnt=0;
|
||||||
|
@ -375,6 +375,7 @@ protected:
|
|||||||
*/
|
*/
|
||||||
class ResmedLoader : public CPAPLoader
|
class ResmedLoader : public CPAPLoader
|
||||||
{
|
{
|
||||||
|
Q_OBJECT
|
||||||
friend class ResmedImport;
|
friend class ResmedImport;
|
||||||
friend class ResmedImportStage2;
|
friend class ResmedImportStage2;
|
||||||
public:
|
public:
|
||||||
|
@ -606,12 +606,14 @@ bool Machine::Load()
|
|||||||
|
|
||||||
QPixmap image = getPixmap().scaled(64,64);
|
QPixmap image = getPixmap().scaled(64,64);
|
||||||
popup->setPixmap(image);
|
popup->setPixmap(image);
|
||||||
popup->setMessage(QObject::tr("Loading %1 data...").arg(info.brand));
|
popup->setMessage(QObject::tr("Loading %1 data for %2...").arg(info.brand).arg(p_profile->user->userName()));
|
||||||
popup->show();
|
popup->show();
|
||||||
|
|
||||||
QProgressBar * progress = popup->progress;
|
QProgressBar * saveQProgress = qprogress;
|
||||||
|
|
||||||
if (!LoadSummary(progress)) {
|
qprogress = popup->progress;
|
||||||
|
|
||||||
|
if (!LoadSummary(qprogress)) {
|
||||||
// No XML index file, so assume upgrading, or it simply just got screwed up or deleted...
|
// No XML index file, so assume upgrading, or it simply just got screwed up or deleted...
|
||||||
QTime time;
|
QTime time;
|
||||||
time.start();
|
time.start();
|
||||||
@ -631,17 +633,15 @@ bool Machine::Load()
|
|||||||
QStringList filelist = dir.entryList();
|
QStringList filelist = dir.entryList();
|
||||||
int size = filelist.size();
|
int size = filelist.size();
|
||||||
|
|
||||||
if (progress) {
|
if (qprogress) {
|
||||||
progress->setMinimum(0);
|
qprogress->setMinimum(0);
|
||||||
progress->setMaximum(0);
|
qprogress->setValue(0);
|
||||||
progress->setValue(0);
|
|
||||||
QApplication::processEvents();
|
QApplication::processEvents();
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i=0; i < size; i++) {
|
// Legacy crap.. Summary and Event stuff used to be in one big pile in the machine folder root
|
||||||
QString filename = filelist.at(i);
|
for (auto & filename : filelist) {
|
||||||
QFile::copy(path+filename, summarypath+filename);
|
QFile::rename(path+filename, summarypath+filename);
|
||||||
QFile::remove(path+filename);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy old Event files to folder
|
// Copy old Event files to folder
|
||||||
@ -654,14 +654,13 @@ bool Machine::Load()
|
|||||||
if (!dir.exists(eventpath)) dir.mkpath(eventpath);
|
if (!dir.exists(eventpath)) dir.mkpath(eventpath);
|
||||||
for (int i=0; i< size; i++) {
|
for (int i=0; i< size; i++) {
|
||||||
if ((i % 50) == 0) { // This is slow.. :-/
|
if ((i % 50) == 0) { // This is slow.. :-/
|
||||||
if (progress) { progress->setValue((float(i) / float(size) * 100.0)); }
|
if (qprogress) { qprogress->setValue((float(i) / float(size) * 100.0)); }
|
||||||
|
|
||||||
QApplication::processEvents();
|
QApplication::processEvents();
|
||||||
}
|
}
|
||||||
|
|
||||||
QString filename = filelist.at(i);
|
QString filename = filelist.at(i);
|
||||||
QFile::copy(path+filename, eventpath+filename);
|
QFile::rename(path+filename, eventpath+filename);
|
||||||
QFile::remove(path+filename);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -675,10 +674,10 @@ bool Machine::Load()
|
|||||||
filelist = dir.entryList();
|
filelist = dir.entryList();
|
||||||
size = filelist.size();
|
size = filelist.size();
|
||||||
|
|
||||||
if (progress) {
|
if (qprogress) {
|
||||||
progress->setMinimum(0);
|
qprogress->setMinimum(0);
|
||||||
progress->setMaximum(size);
|
qprogress->setMaximum(size);
|
||||||
progress->setValue(0);
|
qprogress->setValue(0);
|
||||||
QApplication::processEvents();
|
QApplication::processEvents();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -689,7 +688,7 @@ bool Machine::Load()
|
|||||||
for (int i=0; i < size; i++) {
|
for (int i=0; i < size; i++) {
|
||||||
|
|
||||||
if ((i % 50) == 0) { // This is slow.. :-/
|
if ((i % 50) == 0) { // This is slow.. :-/
|
||||||
if (progress) { progress->setValue(i); }
|
if (qprogress) { qprogress->setValue(i); }
|
||||||
|
|
||||||
QApplication::processEvents();
|
QApplication::processEvents();
|
||||||
}
|
}
|
||||||
@ -713,15 +712,17 @@ bool Machine::Load()
|
|||||||
|
|
||||||
SaveSummary();
|
SaveSummary();
|
||||||
qDebug() << "Loaded" << info.model << "data in" << time.elapsed() << "ms";
|
qDebug() << "Loaded" << info.model << "data in" << time.elapsed() << "ms";
|
||||||
if (progress) { progress->setValue(size); }
|
if (qprogress) { qprogress->setValue(size); }
|
||||||
} else {
|
} else {
|
||||||
if (progress) { progress->setValue(100); }
|
if (qprogress) { qprogress->setValue(100); }
|
||||||
}
|
}
|
||||||
loadSessionInfo();
|
loadSessionInfo();
|
||||||
QApplication::processEvents();
|
QApplication::processEvents();
|
||||||
popup->hide();
|
popup->hide();
|
||||||
delete popup;
|
delete popup;
|
||||||
|
|
||||||
|
qprogress = saveQProgress;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -893,16 +894,15 @@ void Machine::runTasks()
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
QThreadPool * threadpool = QThreadPool::globalInstance();
|
QThreadPool * threadpool = QThreadPool::globalInstance();
|
||||||
// int m_totaltasks=m_tasklist.size();
|
int m_totaltasks=m_tasklist.size();
|
||||||
int m_currenttask=0;
|
int m_currenttask=0;
|
||||||
|
qprogress->setMaximum(m_totaltasks);
|
||||||
while (!m_tasklist.isEmpty()) {
|
while (!m_tasklist.isEmpty()) {
|
||||||
if (threadpool->tryStart(m_tasklist.at(0))) {
|
if (threadpool->tryStart(m_tasklist.at(0))) {
|
||||||
m_tasklist.pop_front();
|
m_tasklist.pop_front();
|
||||||
// float f = float(m_currenttask) / float(m_totaltasks) * 100.0;
|
qprogress->setValue(m_currenttask++);
|
||||||
// qprogress->setValue(f);
|
QApplication::processEvents();
|
||||||
m_currenttask++;
|
|
||||||
}
|
}
|
||||||
//QApplication::processEvents();
|
|
||||||
}
|
}
|
||||||
QThreadPool::globalInstance()->waitForDone(-1);
|
QThreadPool::globalInstance()->waitForDone(-1);
|
||||||
|
|
||||||
|
@ -109,6 +109,7 @@ class MachineLoader: public QObject
|
|||||||
|
|
||||||
signals:
|
signals:
|
||||||
void updateProgress(int cnt, int total);
|
void updateProgress(int cnt, int total);
|
||||||
|
void updateMessage(QString);
|
||||||
void machineUnsupported(Machine *);
|
void machineUnsupported(Machine *);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@ -151,6 +152,7 @@ public:
|
|||||||
virtual ChannelID HumidifierLevel() { return CPAP_HumidSetting; }
|
virtual ChannelID HumidifierLevel() { return CPAP_HumidSetting; }
|
||||||
virtual ChannelID CPAPModeChannel() { return CPAP_Mode; }
|
virtual ChannelID CPAPModeChannel() { return CPAP_Mode; }
|
||||||
virtual void initChannels() {}
|
virtual void initChannels() {}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ImportPath
|
struct ImportPath
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SleepLib Progress Dialog Header
|
/* SleepLib Progress Dialog Header
|
||||||
*
|
*
|
||||||
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
* Copyright (c) 2011-2018 Mark Watkins <mark@jedimark.net>
|
||||||
*
|
*
|
||||||
@ -9,7 +9,7 @@
|
|||||||
#include "progressdialog.h"
|
#include "progressdialog.h"
|
||||||
|
|
||||||
ProgressDialog::ProgressDialog(QWidget * parent):
|
ProgressDialog::ProgressDialog(QWidget * parent):
|
||||||
QDialog(parent, Qt::SplashScreen)
|
QDialog(parent, Qt::Popup)
|
||||||
{
|
{
|
||||||
waitmsg = new QLabel(QObject::tr("PLease Wait..."));
|
waitmsg = new QLabel(QObject::tr("PLease Wait..."));
|
||||||
hlayout = new QHBoxLayout;
|
hlayout = new QHBoxLayout;
|
||||||
@ -36,3 +36,7 @@ void ProgressDialog::doUpdateProgress(int cnt, int total)
|
|||||||
progress->setValue(cnt);
|
progress->setValue(cnt);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void ProgressDialog::setMessage(QString msg) {
|
||||||
|
waitmsg->setText(msg); update();
|
||||||
|
}
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
/* SleepLib Progress Dialog Header
|
/* SleepLib Progress Dialog Header
|
||||||
*
|
*
|
||||||
* Copyright (C) 2011-2018 Mark Watkins <mark@jedimark.net>
|
* Copyright (C) 2011-2018 Mark Watkins <mark@jedimark.net>
|
||||||
*
|
*
|
||||||
@ -16,14 +16,15 @@
|
|||||||
#include <QProgressBar>
|
#include <QProgressBar>
|
||||||
|
|
||||||
class ProgressDialog:public QDialog {
|
class ProgressDialog:public QDialog {
|
||||||
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
explicit ProgressDialog(QWidget * parent);
|
explicit ProgressDialog(QWidget * parent);
|
||||||
virtual ~ProgressDialog();
|
virtual ~ProgressDialog();
|
||||||
|
|
||||||
void setMessage(QString msg) { waitmsg->setText(msg); }
|
|
||||||
void setPixmap(QPixmap &pixmap) { imglabel->setPixmap(pixmap); }
|
void setPixmap(QPixmap &pixmap) { imglabel->setPixmap(pixmap); }
|
||||||
QProgressBar * progress;
|
QProgressBar * progress;
|
||||||
public slots:
|
public slots:
|
||||||
|
void setMessage(QString msg);
|
||||||
void doUpdateProgress(int cnt, int total);
|
void doUpdateProgress(int cnt, int total);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
@ -463,13 +463,12 @@ void MainWindow::OpenProfile(QString profileName)
|
|||||||
p_profile->p_preferences[STR_PREF_ReimportBackup]=false;
|
p_profile->p_preferences[STR_PREF_ReimportBackup]=false;
|
||||||
}
|
}
|
||||||
|
|
||||||
p_profile->LoadMachineData();
|
|
||||||
|
|
||||||
QList<MachineLoader *> loaders = GetLoaders();
|
QList<MachineLoader *> loaders = GetLoaders();
|
||||||
for (int i=0; i<loaders.size(); ++i) {
|
for (int i=0; i<loaders.size(); ++i) {
|
||||||
connect(loaders.at(i), SIGNAL(machineUnsupported(Machine*)), this, SLOT(MachineUnsupported(Machine*)));
|
connect(loaders.at(i), SIGNAL(machineUnsupported(Machine*)), this, SLOT(MachineUnsupported(Machine*)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
p_profile->LoadMachineData();
|
||||||
|
|
||||||
ui->statStartDate->setDate(p_profile->FirstDay());
|
ui->statStartDate->setDate(p_profile->FirstDay());
|
||||||
ui->statEndDate->setDate(p_profile->LastDay());
|
ui->statEndDate->setDate(p_profile->LastDay());
|
||||||
@ -536,7 +535,9 @@ void MainWindow::CloseProfile()
|
|||||||
|
|
||||||
void MainWindow::Startup()
|
void MainWindow::Startup()
|
||||||
{
|
{
|
||||||
|
for (auto & loader : GetLoaders()) {
|
||||||
|
loader->setParent(this);
|
||||||
|
}
|
||||||
QString lastProfile = AppSetting->profileName();
|
QString lastProfile = AppSetting->profileName();
|
||||||
|
|
||||||
if (Profiles::profiles.contains(lastProfile) && AppSetting->autoOpenLastUsed()) {
|
if (Profiles::profiles.contains(lastProfile) && AppSetting->autoOpenLastUsed()) {
|
||||||
@ -557,27 +558,21 @@ int MainWindow::importCPAP(ImportPath import, const QString &message)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
QDialog * popup = new QDialog(this);
|
ProgressDialog * progdlg = new ProgressDialog(this);
|
||||||
QLabel * waitmsg = new QLabel(message);
|
|
||||||
QHBoxLayout *hlayout = new QHBoxLayout;
|
|
||||||
|
|
||||||
QLabel * imglabel = new QLabel(popup);
|
|
||||||
|
|
||||||
QPixmap image = import.loader->getPixmap(import.loader->PeekInfo(import.path).series);
|
QPixmap image = import.loader->getPixmap(import.loader->PeekInfo(import.path).series);
|
||||||
// QPixmap image(getCPAPPixmap(import.loader->loaderName()));
|
|
||||||
image = image.scaled(64,64);
|
image = image.scaled(64,64);
|
||||||
imglabel->setPixmap(image);
|
progdlg->setPixmap(image);
|
||||||
|
|
||||||
|
QProgressBar *saveQprogress = qprogress;
|
||||||
|
qprogress = progdlg->progress;
|
||||||
|
|
||||||
QVBoxLayout * vlayout = new QVBoxLayout;
|
progdlg->show();
|
||||||
popup->setLayout(vlayout);
|
progdlg->setMessage(message);
|
||||||
vlayout->addLayout(hlayout);
|
|
||||||
hlayout->addWidget(imglabel);
|
import.loader->setParent(this);
|
||||||
hlayout->addWidget(waitmsg,1,Qt::AlignCenter);
|
connect(import.loader, SIGNAL(updateMessage(QString)), progdlg, SLOT(setMessage(QString)));
|
||||||
vlayout->addWidget(qprogress,1);
|
|
||||||
|
|
||||||
qprogress->setVisible(true);
|
|
||||||
popup->show();
|
|
||||||
int c = import.loader->Open(import.path);
|
int c = import.loader->Open(import.path);
|
||||||
|
|
||||||
if (c > 0) {
|
if (c > 0) {
|
||||||
@ -587,17 +582,17 @@ int MainWindow::importCPAP(ImportPath import, const QString &message)
|
|||||||
} else {
|
} else {
|
||||||
Notify(tr("Couldn't find any valid Machine Data at\n\n%1").arg(import.path),tr("Import Problem"));
|
Notify(tr("Couldn't find any valid Machine Data at\n\n%1").arg(import.path),tr("Import Problem"));
|
||||||
}
|
}
|
||||||
|
disconnect(import.loader, SIGNAL(updateMessage(QString)), progdlg, SLOT(setMessage(QString)));
|
||||||
|
|
||||||
popup->hide();
|
progdlg->hide();
|
||||||
vlayout->removeWidget(qprogress);
|
|
||||||
ui->statusbar->insertWidget(1,qprogress,1);
|
delete progdlg;
|
||||||
qprogress->setVisible(false);
|
|
||||||
|
|
||||||
delete popup;
|
|
||||||
if (AppSetting->openTabAfterImport()>0) {
|
if (AppSetting->openTabAfterImport()>0) {
|
||||||
ui->tabWidget->setCurrentIndex(AppSetting->openTabAfterImport());
|
ui->tabWidget->setCurrentIndex(AppSetting->openTabAfterImport());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
qprogress=saveQprogress;
|
||||||
return c;
|
return c;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user