// Hyperion includes #include #include #include // utils includes #include // qt #include GrabberWrapper* GrabberWrapper::instance = nullptr; GrabberWrapper::GrabberWrapper(const QString& grabberName, Grabber * ggrabber, unsigned width, unsigned height, unsigned updateRate_Hz) : _grabberName(grabberName) , _timer(new QTimer(this)) , _updateInterval_ms(1000/updateRate_Hz) , _log(Logger::getInstance(grabberName.toUpper())) , _ggrabber(ggrabber) , _image(0,0) { GrabberWrapper::instance = this; // Configure the timer to generate events every n milliseconds _timer->setInterval(_updateInterval_ms); _image.resize(width, height); connect(_timer, &QTimer::timeout, this, &GrabberWrapper::action); // connect the image forwarding (_grabberName.startsWith("V4L")) ? connect(this, &GrabberWrapper::systemImage, GlobalSignals::getInstance(), &GlobalSignals::setV4lImage) : connect(this, &GrabberWrapper::systemImage, GlobalSignals::getInstance(), &GlobalSignals::setSystemImage); // listen for source requests connect(GlobalSignals::getInstance(), &GlobalSignals::requestSource, this, &GrabberWrapper::handleSourceRequest); } GrabberWrapper::~GrabberWrapper() { Debug(_log,"Close grabber: %s", QSTRING_CSTR(_grabberName)); } bool GrabberWrapper::start() { if (!_timer->isActive()) { // Start the timer with the pre configured interval Debug(_log,"Grabber start()"); _timer->start(); } return _timer->isActive(); } void GrabberWrapper::stop() { if (_timer->isActive()) { // Stop the timer, effectivly stopping the process Debug(_log,"Grabber stop()"); _timer->stop(); } } bool GrabberWrapper::isActive() const { return _timer->isActive(); } QStringList GrabberWrapper::getActive(int inst) const { QStringList result = QStringList(); if(GRABBER_V4L_CLIENTS.contains(inst)) result << GRABBER_V4L_CLIENTS.value(inst); if(GRABBER_SYS_CLIENTS.contains(inst)) result << GRABBER_SYS_CLIENTS.value(inst); return result; } QStringList GrabberWrapper::availableGrabbers() { QStringList grabbers; #ifdef ENABLE_DISPMANX grabbers << "dispmanx"; #endif #if defined(ENABLE_V4L2) || defined(ENABLE_MF) grabbers << "v4l2"; #endif #ifdef ENABLE_FB grabbers << "framebuffer"; #endif #ifdef ENABLE_AMLOGIC grabbers << "amlogic"; #endif #ifdef ENABLE_OSX grabbers << "osx"; #endif #ifdef ENABLE_X11 grabbers << "x11"; #endif #ifdef ENABLE_XCB grabbers << "xcb"; #endif #ifdef ENABLE_QT grabbers << "qt"; #endif #ifdef ENABLE_DX grabbers << "dx"; #endif return grabbers; } void GrabberWrapper::setVideoMode(VideoMode mode) { if (_ggrabber != nullptr) { Info(_log,"setvideomode"); _ggrabber->setVideoMode(mode); } } void GrabberWrapper::setCropping(unsigned cropLeft, unsigned cropRight, unsigned cropTop, unsigned cropBottom) { _ggrabber->setCropping(cropLeft, cropRight, cropTop, cropBottom); } void GrabberWrapper::updateTimer(int interval) { if(_updateInterval_ms != interval) { _updateInterval_ms = interval; const bool& timerWasActive = _timer->isActive(); _timer->stop(); _timer->setInterval(_updateInterval_ms); if(timerWasActive) _timer->start(); } } void GrabberWrapper::handleSettingsUpdate(settings::type type, const QJsonDocument& config) { if(type == settings::SYSTEMCAPTURE && !_grabberName.startsWith("V4L")) { // extract settings const QJsonObject& obj = config.object(); // width/height _ggrabber->setWidthHeight(obj["width"].toInt(96), obj["height"].toInt(96)); // display index for MAC _ggrabber->setDisplayIndex(obj["display"].toInt(0)); // device path for Framebuffer _ggrabber->setDevicePath(obj["device"].toString("/dev/fb0")); // pixel decimation for x11 _ggrabber->setPixelDecimation(obj["pixelDecimation"].toInt(8)); // crop for system capture _ggrabber->setCropping( obj["cropLeft"].toInt(0), obj["cropRight"].toInt(0), obj["cropTop"].toInt(0), obj["cropBottom"].toInt(0)); // eval new update time updateTimer(1000/obj["frequency_Hz"].toInt(10)); } } void GrabberWrapper::handleSourceRequest(hyperion::Components component, int hyperionInd, bool listen) { if(component == hyperion::Components::COMP_GRABBER && !_grabberName.startsWith("V4L")) { if(listen) GRABBER_SYS_CLIENTS.insert(hyperionInd, _grabberName); else GRABBER_SYS_CLIENTS.remove(hyperionInd); if(GRABBER_SYS_CLIENTS.empty()) stop(); else start(); } else if(component == hyperion::Components::COMP_V4L && _grabberName.startsWith("V4L")) { if(listen) GRABBER_V4L_CLIENTS.insert(hyperionInd, _grabberName); else GRABBER_V4L_CLIENTS.remove(hyperionInd); if(GRABBER_V4L_CLIENTS.empty()) stop(); else start(); } } void GrabberWrapper::tryStart() { // verify start condition if((_grabberName.startsWith("V4L") && !GRABBER_V4L_CLIENTS.empty()) || (!_grabberName.startsWith("V4L") && !GRABBER_SYS_CLIENTS.empty())) { start(); } } QStringList GrabberWrapper::getV4L2devices() const { if(_grabberName.startsWith("V4L")) return _ggrabber->getV4L2devices(); return QStringList(); } QString GrabberWrapper::getV4L2deviceName(const QString& devicePath) const { if(_grabberName.startsWith("V4L")) return _ggrabber->getV4L2deviceName(devicePath); return QString(); } QMultiMap GrabberWrapper::getV4L2deviceInputs(const QString& devicePath) const { if(_grabberName.startsWith("V4L")) return _ggrabber->getV4L2deviceInputs(devicePath); return QMultiMap(); } QStringList GrabberWrapper::getV4L2EncodingFormats(const QString& devicePath) const { if(_grabberName.startsWith("V4L")) return _ggrabber->getV4L2EncodingFormats(devicePath); return QStringList(); } QStringList GrabberWrapper::getResolutions(const QString& devicePath) const { if(_grabberName.startsWith("V4L")) return _ggrabber->getResolutions(devicePath); return QStringList(); } QStringList GrabberWrapper::getFramerates(const QString& devicePath) const { if(_grabberName.startsWith("V4L")) return _ggrabber->getFramerates(devicePath); return QStringList(); }