#include "pylonrecorder.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if defined(QT_PRINTSUPPORT_LIB) # include # if QT_CONFIG(printdialog) #include # include # endif #endif PylonRecorder::PylonRecorder(QWidget *parent) : QMainWindow(parent), imageLabel(new QLabel), scrollArea(new QScrollArea), singlecamgrabber(nullptr), dualcamgrabber(nullptr), writer(nullptr), buffer(nullptr), singlecam(nullptr), dualcam(nullptr), dryRun(false), cameraOpened(false), camsconfigured(false) { imageLabel->setBackgroundRole(QPalette::Base); imageLabel->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored); imageLabel->setScaledContents(true); scrollArea->setBackgroundRole(QPalette::Dark); scrollArea->setWidget(imageLabel); scrollArea->setVisible(false); setCentralWidget(scrollArea); frameTimer = new QTimer(this); connect(frameTimer, &QTimer::timeout, this, &PylonRecorder::displaySingleFrame); pressureTimer = new QTimer(this); connect(pressureTimer, &QTimer::timeout, this, &PylonRecorder::displayBufferPressure); labelTimer = new QTimer(this); connect(labelTimer, &QTimer::timeout, this, &PylonRecorder::displayActivity); pressureBar = new QProgressBar(this); pressureBar->setRange(0, 100); pressureBar->setTextVisible(true); pressureBar->setFixedSize(200, 25); QColor color = progressColor(0); QPalette progressPalette = pressureBar->palette(); progressPalette.setBrush(QPalette::Highlight, QBrush(color)); pressureBar->setPalette(progressPalette); QLabel *preassureLabel = new QLabel("Buffer preassure:", this); preassureLabel->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial;font-weight: Bold}"); loadBar = new QProgressBar(this); loadBar->setRange(0, defaultBufferSize); loadBar->setFixedSize(200, 25); QLabel *loadLabel = new QLabel("Load:", this); loadLabel->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial;font-weight: Bold}"); writingLabel = new QLabel("writing"); writingLabel->setEnabled(false); writingLabel->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial;}"); grabbingLabel = new QLabel("grabbing"); grabbingLabel->setEnabled(false); grabbingLabel->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial;}"); labelSwitch = false; cameraConnectedLabel = new QLabel("not connected"); cameraConnectedLabel->setStyleSheet("QLabel { color : red; }"); cameraConnectedLabel->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial;}"); fileLabel = new QLabel(); fileLabel->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial;}"); QLabel *camHeader = new QLabel("Camera:"); camHeader->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial; font-weight: Bold}"); QLabel *statusHeader = new QLabel("Status:"); statusHeader->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial; font-weight: Bold}"); QLabel *fileHeader = new QLabel("Output file:"); fileHeader->setStyleSheet("QLabel{font-size: 11pt;font-family: Arial; font-weight: Bold}"); statusBar()->addWidget(camHeader); statusBar()->addWidget(cameraConnectedLabel); statusBar()->addWidget(preassureLabel); statusBar()->addWidget(pressureBar); statusBar()->addWidget(loadLabel); statusBar()->addWidget(loadBar); statusBar()->addWidget(statusHeader); statusBar()->addWidget(grabbingLabel); statusBar()->addWidget(writingLabel); statusBar()->addWidget(fileHeader); statusBar()->addWidget(fileLabel); resize(QGuiApplication::primaryScreen()->availableSize() * 3 / 5); detectCameras(); createActions(); updateActions(); applySettings(); } void PylonRecorder::detectCameras() { qDebug() << "Detecting devices!"; Pylon::CTlFactory& TlFactory = Pylon::CTlFactory::GetInstance(); TlFactory.EnumerateDevices(deviceList); qDebug() << "Found devices!" << deviceList.size(); } PylonRecorder::~PylonRecorder(){ qDebug() << "Destructing PylonRecorder"; if (singlecamgrabber != nullptr && singlecamgrabber->isRunning()) { singlecamgrabber->requestStop(); singlecamgrabber->wait(1000); } if (writer != nullptr && writer->isRunning()) { writer->forceStop(); writer->wait(1000); } storeSettings(); if (singlecam != nullptr) { qDebug() << "Deleting singlecam"; delete singlecam; singlecam = nullptr; } if (dualcam != nullptr) { qDebug() << "Deleting dualcam"; delete dualcam; dualcam = nullptr; } if (buffer != nullptr) { qDebug() << "Deleting buffer"; delete buffer; buffer = nullptr; } if (singlecamgrabber != nullptr) { qDebug() << "Deleting grabber"; delete singlecamgrabber; singlecamgrabber = nullptr; } if (dualcamgrabber != nullptr) { qDebug() << "Deleting grabber"; delete dualcamgrabber; dualcamgrabber = nullptr; } if (writer != nullptr) { qDebug() << "Deleting writer"; delete writer; writer = nullptr; } qDebug() << "Deleting setting"; delete settings; } void PylonRecorder::applySettings() { int bufferSize = settings->value("recorder/buffersize", defaultBufferSize).toInt(); int exposureTime = settings->value("camera/exposure", defaultExposureTime).toInt(); int frameRate = settings->value("camera/framerate", defaultFrameRate).toInt(); int gain = settings->value("camera/gain", defaultGain).toInt(); storageLocation = settings->value("recorder/storagelocation", "").toString(); if (!storageLocation.isEmpty()) { selectStorageAction->setStatusTip(tr("Default storage location: ") + storageLocation); } buffersizeSpinner->setValue(bufferSize); exposureSpinner->setValue(exposureTime); gainSpinner->setValue(gain); framerateSpinner->setValue(frameRate); } void PylonRecorder::storeSettings() { // FIXME store cam layout to settings settings->setValue("camera/exposure", exposureSpinner->value()); settings->setValue("camera/framerate", framerateSpinner->value()); settings->setValue("camera/gain", gainSpinner->value()); settings->setValue("recorder/buffersize", buffersizeSpinner->value()); settings->setValue("recorder/storagelocation", storageLocation); } bool PylonRecorder::loadFile(const QString &fileName) { QImageReader reader(fileName); reader.setAutoTransform(true); const QImage newImage = reader.read(); if (newImage.isNull()) { QMessageBox::information(this, QGuiApplication::applicationDisplayName(), tr("Cannot load %1: %2") .arg(QDir::toNativeSeparators(fileName), reader.errorString())); return false; } setImage(newImage); setWindowFilePath(fileName); const QString message = tr("Opened \"%1\", %2x%3, Depth: %4") .arg(QDir::toNativeSeparators(fileName)).arg(image.width()).arg(image.height()).arg(image.depth()); statusBar()->showMessage(message); return true; } void PylonRecorder::setImage(const QImage &newImage) { //FIXME figure out how to display both images. extract to extra class... image = newImage; // (image.colorSpace().isValid()) // image.convertToColorSpace(QColorSpace::SRgb); imageLabel->setPixmap(QPixmap::fromImage(image)); // scaleFactor = 1.0; scrollArea->setVisible(true); printAct->setEnabled(true); fitToWindowAct->setEnabled(true); //updateActions(); if (!fitToWindowAct->isChecked()) { imageLabel->adjustSize(); applyScaling(); } this->update(); } bool PylonRecorder::saveFile(const QString &fileName) { QImageWriter writer(fileName); if (!writer.write(image)) { QMessageBox::information(this, QGuiApplication::applicationDisplayName(), tr("Cannot write %1: %2") .arg(QDir::toNativeSeparators(fileName)), writer.errorString()); return false; } const QString message = tr("Wrote \"%1\"").arg(QDir::toNativeSeparators(fileName)); statusBar()->showMessage(message); return true; } static void initializeImageFileDialog(QFileDialog &dialog, QFileDialog::AcceptMode acceptMode) { static bool firstDialog = true; if (firstDialog) { firstDialog = false; const QStringList picturesLocations = QStandardPaths::standardLocations(QStandardPaths::PicturesLocation); dialog.setDirectory(picturesLocations.isEmpty() ? QDir::currentPath() : picturesLocations.last()); } QStringList mimeTypeFilters; const QByteArrayList supportedMimeTypes = acceptMode == QFileDialog::AcceptOpen ? QImageReader::supportedMimeTypes() : QImageWriter::supportedMimeTypes(); for (const QByteArray &mimeTypeName : supportedMimeTypes) mimeTypeFilters.append(mimeTypeName); mimeTypeFilters.sort(); dialog.setMimeTypeFilters(mimeTypeFilters); dialog.selectMimeTypeFilter("image/jpeg"); if (acceptMode == QFileDialog::AcceptSave) dialog.setDefaultSuffix("jpg"); } void PylonRecorder::open() { QFileDialog dialog(this, tr("Open File")); initializeImageFileDialog(dialog, QFileDialog::AcceptOpen); while (dialog.exec() == QDialog::Accepted && !loadFile(dialog.selectedFiles().first())) {} } void PylonRecorder::saveAs() { QFileDialog dialog(this, tr("Save File As")); initializeImageFileDialog(dialog, QFileDialog::AcceptSave); while (dialog.exec() == QDialog::Accepted && !saveFile(dialog.selectedFiles().first())) {} } void PylonRecorder::print() { QPixmap map = imageLabel->pixmap(Qt::ReturnByValue); Q_ASSERT(!map.isNull()); #if defined(QT_PRINTSUPPORT_LIB) && QT_CONFIG(printdialog) QPrintDialog dialog(&printer, this); if (dialog.exec()) { QPainter painter(&printer); QRect rect = painter.viewport(); QSize size = map.size(); size.scale(rect.size(), Qt::KeepAspectRatio); painter.setViewport(rect.x(), rect.y(), size.width(), size.height()); painter.setWindow(map.rect()); painter.drawPixmap(0, 0, map); } #endif } void PylonRecorder::copy() { #ifndef QT_NO_CLIPBOARD QGuiApplication::clipboard()->setImage(image); #endif // !QT_NO_CLIPBOARD } #ifndef QT_NO_CLIPBOARD static QImage clipboardImage() { if (const QMimeData *mimeData = QGuiApplication::clipboard()->mimeData()) { if (mimeData->hasImage()) { const QImage image = qvariant_cast(mimeData->imageData()); if (!image.isNull()) return image; } } return QImage(); } #endif // !QT_NO_CLIPBOARD void PylonRecorder::paste() { #ifndef QT_NO_CLIPBOARD const QImage newImage = clipboardImage(); if (newImage.isNull()) { statusBar()->showMessage(tr("No image in clipboard")); } else { setImage(newImage); setWindowFilePath(QString()); const QString message = tr("Obtained image from clipboard, %1x%2, Depth: %3") .arg(newImage.width()).arg(newImage.height()).arg(newImage.depth()); statusBar()->showMessage(message); } #endif // !QT_NO_CLIPBOARD } void PylonRecorder::zoomIn() { scaleImage(1.25); } void PylonRecorder::zoomOut() { scaleImage(0.8); } void PylonRecorder::normalSize() { imageLabel->adjustSize(); scaleFactor = 1.0; } void PylonRecorder::fitToWindow() { bool fitToWindow = fitToWindowAct->isChecked(); scrollArea->setWidgetResizable(fitToWindow); if (!fitToWindow) normalSize(); updateActions(); } void PylonRecorder::about() { QMessageBox::about(this, tr("About Pylon Recorder"), tr("

Pylon Recorder
Simple recorder for video grabbing from pylon USB3 monochrome camera." "Usage: In order to grab image, a camera device needs to be opened, press the connect button to do so. Push the record button to record continuously.

" "

videos are stored in the same folder as the tool is called from. Videos are compressed to mp4." " As the buffer preassure reaches 100% frames will be lost! You may want to increase the buffer size" " or reduce the framerate.

In order to run you need to install the pylon libraries" " and the mp4 video codec avalable for download at Basler website

" "

by Jan Grewe, Neuroethology Lab, University of Tuebingen.

")); } void PylonRecorder::createActions() { const QIcon connect_icon(":/images/connect.png"); const QIcon disconnect_icon(":/images/disconnect.png"); const QIcon snapshot_icon(":/images/snapshot.png"); const QIcon grab_icon(":/images/record.png"); const QIcon stop_icon(":/images/stop.png"); const QIcon exit_icon(":/images/exit.png"); QMenu *fileMenu = menuBar()->addMenu(tr("&File")); QAction *openAct = fileMenu->addAction(tr("&Open..."), this, &PylonRecorder::open); openAct->setShortcut(QKeySequence::Open); saveAsAct = fileMenu->addAction(tr("&Save As..."), this, &PylonRecorder::saveAs); saveAsAct->setEnabled(false); printAct = fileMenu->addAction(tr("&Print..."), this, &PylonRecorder::print); printAct->setShortcut(QKeySequence::Print); printAct->setEnabled(false); fileMenu->addSeparator(); QAction *exitAct = fileMenu->addAction(exit_icon, tr("E&xit"), this, &PylonRecorder::quitApplication); exitAct->setShortcut(tr("Ctrl+Q")); QMenu *editMenu = menuBar()->addMenu(tr("&Edit")); copyAct = editMenu->addAction(tr("&Copy"), this, &PylonRecorder::copy); copyAct->setShortcut(QKeySequence::Copy); copyAct->setEnabled(false); QAction *pasteAct = editMenu->addAction(tr("&Paste"), this, &PylonRecorder::paste); pasteAct->setShortcut(QKeySequence::Paste); QMenu *viewMenu = menuBar()->addMenu(tr("&View")); zoomInAct = viewMenu->addAction(tr("Zoom &In (25%)"), this, &PylonRecorder::zoomIn); zoomInAct->setShortcut(QKeySequence::ZoomIn); zoomInAct->setEnabled(false); zoomOutAct = viewMenu->addAction(tr("Zoom &Out (25%)"), this, &PylonRecorder::zoomOut); zoomOutAct->setShortcut(QKeySequence::ZoomOut); zoomOutAct->setEnabled(false); normalSizeAct = viewMenu->addAction(tr("&Normal Size"), this, &PylonRecorder::normalSize); normalSizeAct->setShortcut(tr("Ctrl+S")); normalSizeAct->setEnabled(false); viewMenu->addSeparator(); fitToWindowAct = viewMenu->addAction(tr("&Fit to Window"), this, &PylonRecorder::fitToWindow); fitToWindowAct->setEnabled(false); fitToWindowAct->setCheckable(true); fitToWindowAct->setShortcut(tr("Ctrl+F")); QMenu *camera_menu = menuBar()->addMenu(tr("&Camera")); connect_camera_action = camera_menu->addAction(connect_icon, tr("&connect"), this, &PylonRecorder::connectCamera); connect_camera_action->setStatusTip(tr("Connect to to camera and open device")); disconnect_camera_action = camera_menu->addAction(disconnect_icon, tr("&disconnect"), this, &PylonRecorder::disconnectCamera); disconnect_camera_action->setStatusTip(tr("Disconnect from the camera device")); camera_menu->addSeparator(); grab_still_action = camera_menu->addAction(snapshot_icon, tr("&grab still"), this, &PylonRecorder::grabStillFromPylon); grab_still_action->setStatusTip(tr("Grab single image from Pylon camera")); grab_still_action->setShortcut(tr("Ctrl+ ")); grab_continuous_action = camera_menu->addAction(grab_icon, tr("&grab continuous"), this, &PylonRecorder::startRecording); grab_continuous_action->setShortcut(tr("Ctrl+Enter")); grab_stop_action = camera_menu->addAction(stop_icon, tr("&stop grabbing"), this, &PylonRecorder::stopRecording); grab_stop_action->setShortcut(tr("Ctrl+Shift+Enter")); selectStorageAction = new QAction(tr("storage location"), this); selectStorageAction->setStatusTip(tr("Select a storage location for the recorded videos")); selectStorageAction->setToolTip(tr("Select a storage location for the recorded videos")); connect(selectStorageAction, &QAction::triggered, this, &PylonRecorder::selectStorageLocation); storeSettingsAction = new QAction(tr("store settings"), this); storeSettingsAction->setStatusTip(tr("store current settings as defaults")); connect(storeSettingsAction, &QAction::triggered, this, &PylonRecorder::storeSettings); projectSettingsAction = new QAction(tr("project metadata"), this); projectSettingsAction->setStatusTip(tr("Edit project metadata")); connect(projectSettingsAction, &QAction::triggered, this, &PylonRecorder::editProjectMetadata); QMenu *settingsMenu = menuBar()->addMenu(tr("&Settings")); settingsMenu->addAction(selectStorageAction); settingsMenu->addAction(projectSettingsAction); settingsMenu->addSeparator(); settingsMenu->setToolTipsVisible(true); settingsMenu->addAction(storeSettingsAction); QMenu *helpMenu = menuBar()->addMenu(tr("&Help")); helpMenu->addAction(tr("&About"), this, &PylonRecorder::about); helpMenu->addAction(tr("About &Qt"), &QApplication::aboutQt); framerateSpinner = new QSpinBox(); framerateSpinner->setRange(1, 120); framerateSpinner->setSuffix("Hz"); framerateSpinner->setFixedSize(80, 25); framerateSpinner->setValue(defaultFrameRate); framerateSpinner->setStyleSheet("QSpinBox{font-size: 10pt;font-family: Arial;color: rgb(0, 0, 0);background-color: rgb(255,255,255);}"); //framerateSpinner->setStyleSheet("QSpinBox{font-size: 10px;font-family: Arial;color: rgb(255, 255, 255);background-color: rgb(38,56,76);}"); buffersizeSpinner = new QSpinBox(); buffersizeSpinner->setRange(100, 5000); buffersizeSpinner->setSingleStep(25); buffersizeSpinner->setValue(defaultBufferSize); buffersizeSpinner->setFixedSize(100, 25); buffersizeSpinner->setStyleSheet("QSpinBox{font-size: 10pt;font-family: Arial;color: rgb(0, 0, 0);background-color: rgb(255,255,255);}"); exposureSpinner = new QSpinBox(); exposureSpinner->setRange(10, 50000); exposureSpinner->setSingleStep(50); exposureSpinner->setValue(defaultExposureTime); exposureSpinner->setFixedSize(120, 25); exposureSpinner->setStyleSheet("QSpinBox{font-size: 10pt;font-family: Arial;color: rgb(0, 0, 0);background-color: rgb(255,255,255);}"); gainSpinner = new QSpinBox(); gainSpinner->setRange(0, 24); gainSpinner->setSingleStep(1); gainSpinner->setValue(defaultGain); gainSpinner->setFixedSize(80, 25); gainSpinner->setStyleSheet("QSpinBox{font-size: 10pt;font-family: Arial;color: rgb(0, 0, 0);background-color: rgb(255,255,255);}"); dryRunCheckBox = new QCheckBox("Dry run, no writing"); dryRunCheckBox->setChecked(false); dryRunCheckBox->setFixedHeight(25); dryRunCheckBox->setStyleSheet("QCheckBox{font-size: 12pt;font-family: Arial;color: rgb(0, 0, 0);}"); QToolBar *toolbar = addToolBar("main toolbar"); toolbar->addAction(exitAct); toolbar->addSeparator(); toolbar->addAction(connect_camera_action); toolbar->addAction(disconnect_camera_action); toolbar->addSeparator(); toolbar->addAction(grab_still_action); toolbar->addAction(grab_continuous_action); toolbar->addAction(grab_stop_action); toolbar->addSeparator(); toolbar->addWidget(new QLabel("frame rate (Hz):")); toolbar->addWidget(framerateSpinner); toolbar->addSeparator(); toolbar->addWidget(new QLabel("gain (dB):")); toolbar->addWidget(gainSpinner); toolbar->addSeparator(); toolbar->addWidget(new QLabel("exposure time (us):")); toolbar->addWidget(exposureSpinner); toolbar->addSeparator(); toolbar->addWidget(new QLabel("buffer size:")); toolbar->addWidget(buffersizeSpinner); toolbar->addSeparator(); toolbar->addWidget(dryRunCheckBox); } void PylonRecorder::updateActions() { qInfo() << "Update Actions"; saveAsAct->setEnabled(!image.isNull()); copyAct->setEnabled(!image.isNull()); zoomInAct->setEnabled(!fitToWindowAct->isChecked()); zoomOutAct->setEnabled(!fitToWindowAct->isChecked()); normalSizeAct->setEnabled(!fitToWindowAct->isChecked()); disconnect_camera_action->setEnabled(deviceList.size() > 0); connect_camera_action->setEnabled(true); grab_still_action->setEnabled(deviceList.size() > 0); grab_continuous_action->setEnabled(cameraOpened && !grabbing); // grab_continuous_action->setEnabled(!grabbing); grab_stop_action->setEnabled(grabbing); } void PylonRecorder::scaleImage(double factor) { QPixmap map = imageLabel->pixmap(Qt::ReturnByValue); Q_ASSERT(!map.isNull()); scaleFactor *= factor; imageLabel->resize(scaleFactor * map.size()); adjustScrollBar(scrollArea->horizontalScrollBar(), factor); adjustScrollBar(scrollArea->verticalScrollBar(), factor); zoomInAct->setEnabled(scaleFactor < 3.0); zoomOutAct->setEnabled(scaleFactor > 0.333); } void PylonRecorder::applyScaling(){ imageLabel->resize(scaleFactor * imageLabel->pixmap(Qt::ReturnByValue).size()); } void PylonRecorder::quitApplication() { qDebug() << "QuitApplication: Quit Application!"; if (grabbing) { std::cerr << "QuitApplication: Stop grabbing\n"; qDebug() << "QuitApplication: Stop grabbing"; stopRecording(); } qDebug() << "QuitApplication done!"; this->close(); std::cerr << "QuitApplication: done,\n"; } void PylonRecorder::adjustScrollBar(QScrollBar *scrollBar, double factor) { scrollBar->setValue(int(factor * scrollBar->value() + ((factor - 1) * scrollBar->pageStep()/2))); } void PylonRecorder::cameraConfiguration(){ d = new CamConfigurator(deviceList, this); connect(d, SIGNAL(accepted()), SLOT(cameraConfigurationAccepted())); connect(d, SIGNAL(rejected()), SLOT(cameraConfigurationAborted())); // QObject::connect(&d, SIGNAL(column_visibility_changed(QString, QString,bool)), this, SLOT(visible_columns_update(QString, QString,bool))); d->exec(); } void PylonRecorder::cameraConfigurationAccepted() { qDebug() << "Cameras setting " << ((d->result()) ? "Accepted" : "Discarded"); this->layout = d->layout(); camsconfigured = true; delete d; } void PylonRecorder::cameraConfigurationAborted() { qDebug() << "Camera configuration aborted!"; camsconfigured = false; } void PylonRecorder::connectCamera() { this->layout = CameraLayout(); qDebug() << "connecting camera(s)"; if (deviceList.size() == 0) { detectCameras(); QMessageBox msgBox; QString msg = "

No camera device found!


Connect camera and try again!

"; msgBox.setText(msg); msgBox.exec(); qWarning() << msg.toStdString().c_str(); return; } cameraConfiguration(); if (!camsconfigured) { qDebug() << "cameras have not been properly configured!"; return; } if (layout.mode == CameraMode::single && layout.devices.size() == 1) { qDebug() << "single camera mode"; std::string cname = layout.devices[0]; std::string message; qDebug() << "connecting to camera " << cname.c_str(); singlecam = new PylonWrapper(layout); bool success = singlecam->openCamera(message); if (success) { cameraConnectedLabel->setText("connected"); cameraConnectedLabel->setStyleSheet("QLabel { font-size: 10px;font-family: Arial;color: green;}"); cameraOpened = true; } else { QMessageBox msgBox; QString msg = "

Could not open camera device!

" + QString::fromStdString(message) + "

"; msgBox.setText(msg); msgBox.exec(); cameraOpened = false; } statusBar()->showMessage(QString::fromStdString(message)); updateActions(); } if (layout.mode == CameraMode::dual && layout.devices.size() == 2) { qDebug() << "dual camera mode"; std::string message; dualcam = new DualcamWrapper(layout); bool success = dualcam->openCameras(message); if (success) { cameraConnectedLabel->setText("connected"); cameraConnectedLabel->setStyleSheet("QLabel { font-size: 10px;font-family: Arial;color: green;}"); cameraOpened = true; } else { QMessageBox msgBox; QString msg = "

Could not open camera device!

" + QString::fromStdString(message) + "

"; msgBox.setText(msg); msgBox.exec(); cameraOpened = false; } statusBar()->showMessage(QString::fromStdString(message)); updateActions(); } qDebug() << "connecting cam(s) done!"; } void PylonRecorder::disconnectCamera() { qDebug() << "disconnecting camera"; if (singlecam != nullptr && singlecam->isOpen()) { singlecam->closeCamera(); } if (dualcam != nullptr && dualcam->isOpen()) { dualcam->closeCameras(); } statusBar()->showMessage(tr("Camera closed!")); cameraConnectedLabel->setText("not connected"); cameraConnectedLabel->setStyleSheet("QLabel { font-size: 10px;font-family: Arial;color: red;}"); updateActions(); camsconfigured = false; cameraOpened = false; qDebug() << "disconnecting cameras done"; } VideoSpecs PylonRecorder::getVideoSpecs(const ImageSettings &settings) { VideoSpecs s = VideoSpecs(); if (!this->layout.devices.size() > 0) { return s; } s.fps = framerateSpinner->value(); s.exposureTime = static_cast(exposureSpinner->value()); s.detectorGain = static_cast(gainSpinner->value()); s.pixelType = settings.pixelType; s.orientation = settings.orientation; s.quality = 95; if (layout.mode == CameraMode::single) { s.width = static_cast(this->layout.rois[0].width); s.height = static_cast(this->layout.rois[0].height); } else if (layout.mode == CameraMode::dual) { s.width = static_cast(this->layout.rois[0].width * 2); s.height = static_cast(this->layout.rois[0].height); } return s; } void PylonRecorder::startSinglecamRecording() { qDebug() << "start single-camera recording!"; std::string filename = createFilename("", ".mp4"); fileLabel->setText(QString::fromStdString(filename)); qDebug() << "storing to file " << filename.c_str(); ImageSettings settings = singlecam->getImageSettings(); qDebug() << "got image settings"; VideoSpecs specs = getVideoSpecs(settings); specs.filename = filename; specs.format = VideoFormat::mp4; qDebug() << "got video specifications"; if (buffer != nullptr) { buffer->clear(); delete buffer; buffer = nullptr; } qDebug() << "setting image buffer to size " << buffersizeSpinner->value(); buffer = new ImageBuffer(defaultBufferSize); if (buffersizeSpinner->value() != static_cast(buffer->capacity())) { buffer->resize(static_cast(buffersizeSpinner->value())); loadBar->setRange(0, buffersizeSpinner->value()); } qDebug() << "setting up grabber"; if (singlecamgrabber != nullptr) { delete singlecamgrabber; singlecamgrabber = nullptr; } singlecamgrabber = new Grabber(singlecam, buffer, defaultFrameRate); if (framerateSpinner->value() != singlecamgrabber->currentFramerate()) singlecamgrabber->setFrameRate(framerateSpinner->value()); if (exposureSpinner->value() != int(singlecamgrabber->currentExposureTime())) singlecamgrabber->setExposureTime(static_cast(exposureSpinner->value())); if (gainSpinner->value() != int(singlecamgrabber->currentGain())) singlecamgrabber->setGain(static_cast(gainSpinner->value())); qDebug() << "setup writer"; if (writer != nullptr) { delete writer; writer = nullptr; } writer = new Writer(buffer, 0); connect(writer, SLOT(writingDone()), this, SLOT(writerDone())); writer->setVideoSpecs(specs); QSettings s; this->mdata.read(s); writer->setProjectMetadata(mdata); buffer->clear(); if (!dryRunCheckBox->isChecked()) { writer->start(); writing = true; } dryRun = dryRunCheckBox->isChecked(); singlecamgrabber->start(); grabbing = true; stopRequest = false; } void PylonRecorder::startDualcamRecording() { qDebug() << "start dual-camera recording!"; std::string filename = createFilename("", ".mp4"); fileLabel->setText(QString::fromStdString(filename)); qDebug() << "storing to file " << filename.c_str(); ImageSettings settings = dualcam->getImageSettings(0); //FIXME! qDebug() << "got image settings"; VideoSpecs specs = getVideoSpecs(settings); specs.filename = filename; specs.format = VideoFormat::mp4; qDebug() << "got video specifications"; if (buffer != nullptr) { buffer->clear(); delete buffer; buffer = nullptr; } qDebug() << "setting image buffer to size " << buffersizeSpinner->value(); buffer = new ImageBuffer(defaultBufferSize); if (buffersizeSpinner->value() != static_cast(buffer->capacity())) { buffer->resize(static_cast(buffersizeSpinner->value())); loadBar->setRange(0, buffersizeSpinner->value()); } qDebug() << "setting up grabber"; if (dualcamgrabber != nullptr) { delete dualcamgrabber; dualcamgrabber = nullptr; } dualcamgrabber = new DualcamGrabber(dualcam, buffer, defaultFrameRate); if (framerateSpinner->value() != dualcamgrabber->currentFramerate()) dualcamgrabber->setFrameRate(framerateSpinner->value()); if (exposureSpinner->value() != int(dualcamgrabber->currentExposureTime())) dualcamgrabber->setExposureTime(static_cast(exposureSpinner->value())); if (gainSpinner->value() != int(dualcamgrabber->currentGain())) dualcamgrabber->setGain(static_cast(gainSpinner->value())); qDebug() << "setting up writers"; if (writer != nullptr) { delete writer; writer = nullptr; } writer = new Writer(buffer, 0); connect(writer, SIGNAL(writingDone()), this, SLOT(writerDone())); writer->setVideoSpecs(specs); qDebug() << "push metadata to writer"; QSettings s; this->mdata.read(s); writer->setProjectMetadata(mdata); dryRun = dryRunCheckBox->isChecked(); buffer->clear(); if (!dryRun) { writer->start(); writing = true; } dualcamgrabber->start(); grabbing = true; stopRequest = false; } void PylonRecorder::startRecording() { if (layout.mode == CameraMode::single) { startSinglecamRecording(); } else if (layout.mode == CameraMode::dual){ startDualcamRecording(); } else { qDebug() << "invalid camera mode!"; } pressureTimer->start(50); frameTimer->start(100); labelTimer->start(650); updateActions(); } void PylonRecorder::stopRecording() { qDebug() << "StopRecording!"; if (!stopRequest) { qDebug() << "StopRecording: stop frame timer!"; frameTimer->stop(); qDebug() << "StopRecording: stop grabber!"; if (singlecamgrabber !=nullptr) singlecamgrabber->requestStop(); if (dualcamgrabber !=nullptr) dualcamgrabber->requestStop(); qDebug() << "StopRecording: stop writer!"; if (writer != nullptr) writer->requestStop(); grabbing = false; stopRequest = true; grab_stop_action->setEnabled(false); qDebug() << "StopRecording: clear buffer!"; if(buffer != nullptr) { buffer->clear(); if (dryRun) writerDone(); } } qDebug() << "StopRecording done!"; } void PylonRecorder::writerDone() { pressureTimer->stop(); pressureBar->reset(); loadBar->reset(); labelTimer->stop(); writingLabel->setStyleSheet(inactiveLabelStyle); grabbingLabel->setStyleSheet(inactiveLabelStyle); if (layout.mode == CameraMode::single) { singlecamgrabber->wait(10000); } else { dualcamgrabber->wait(10000); } if (writer != nullptr) writer->wait(10000); writing = false; updateActions(); qInfo() << "writer is Done!"; } void PylonRecorder::displayActivity() { grabbingLabel->setStyleSheet((labelSwitch && grabbing) ? activeLabelStyleHigh : activeLabelStyleLow); writingLabel->setStyleSheet((labelSwitch && writing) ? activeLabelStyleHigh : activeLabelStyleLow); labelSwitch = !labelSwitch; } void PylonRecorder::displaySingleFrame() { MyImage *img; size_t fc = 0; img = buffer->readLast(fc); if (img != nullptr){ QImage qimg(static_cast(img->data()), img->width(), img->height(), QImage::Format::Format_Grayscale8); setImage(qimg); }else { std::cerr << "Error reading last image" << std::endl; } } QColor PylonRecorder::progressColor(int value) { int c, m, k = 0, y = 255; c = 255 - 255 * value/100; if (value < 50) { m = 127 - 127 * value/50; } else { m = 255 * (value-50)/50; } return QColor::fromCmyk(c, m, y, k); } std::string PylonRecorder::createFilename(const std::string &suffix, const std::string &extension) { QDateTime dt(QDateTime::currentDateTimeUtc()); QDate date = dt.date(); std::string base = (date.toString("yyyy.MM.dd") + "_").toStdString(); QString idx = QString::number(movieCount); std::string fname = base + idx.toStdString() + suffix + extension; while (QFile::exists(QString::fromStdString(fname))) { movieCount++; fname = base + QString::number(movieCount).toStdString() + extension; } return fname; } void PylonRecorder::displayBufferPressure() { int value = static_cast(round(buffer->bufferPressure())); pressureBar->setValue(value); QColor color = progressColor(value); progressPalette.setBrush(QPalette::Highlight, QBrush(color)); pressureBar->setPalette(progressPalette); int load = static_cast(buffer->bufferLoad()); loadBar->setValue(load); } void PylonRecorder::grabStillFromPylon() { qDebug() << "Grab still image form camera!"; if (singlecam != nullptr && singlecam->isOpen()) { MyImage img; bool valid = singlecam->grabFrame(img); if (valid) { QImage qimg(static_cast(img.data()), img.width(), img.height(), QImage::Format::Format_Grayscale8); setImage(qimg); } } else { statusBar()->showMessage(tr("Camera is not open! Connect to camera first!")); } //FIXME does not work for single camera mode! qDebug() << "grabbing still image done!"; } void PylonRecorder::selectStorageLocation() { QString dir = QFileDialog::getExistingDirectory(this, tr("Open Directory"), "/home", QFileDialog::ShowDirsOnly | QFileDialog::DontResolveSymlinks); if (!dir.isEmpty()) { this->storageLocation = dir; selectStorageAction->setStatusTip(tr("Storage location: ") + storageLocation); } } void PylonRecorder::editProjectMetadata(){ ProjectSettings *dlg = new ProjectSettings(this); dlg->setModal(true); int res = dlg->exec(); if (res == QDialog::Accepted) { this->mdata = dlg->getMetadata(); } }