mirror of https://github.com/qTox/qTox.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
208 lines
4.8 KiB
208 lines
4.8 KiB
/* |
|
Copyright (C) 2014 by Project Tox <https://tox.im> |
|
|
|
This file is part of qTox, a Qt-based graphical interface for Tox. |
|
|
|
This program is libre 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 COPYING file for more details. |
|
*/ |
|
|
|
#include "camera.h" |
|
#include "src/cameraworker.h" |
|
#include <QDebug> |
|
#include <QThread> |
|
#include <QMutexLocker> |
|
|
|
Camera* Camera::instance = nullptr; |
|
|
|
Camera::Camera() |
|
: refcount(0) |
|
, workerThread(nullptr) |
|
, worker(nullptr) |
|
{ |
|
worker = new CameraWorker(0); |
|
workerThread = new QThread(); |
|
|
|
worker->moveToThread(workerThread); |
|
|
|
connect(workerThread, &QThread::started, worker, &CameraWorker::onStart); |
|
connect(workerThread, &QThread::finished, worker, &CameraWorker::deleteLater); |
|
connect(worker, &CameraWorker::started, this, &Camera::onWorkerStarted); |
|
connect(worker, &CameraWorker::newFrameAvailable, this, &Camera::onNewFrameAvailable); |
|
connect(worker, &CameraWorker::resProbingFinished, this, &Camera::onResProbingFinished); |
|
workerThread->start(); |
|
} |
|
|
|
void Camera::onWorkerStarted() |
|
{ |
|
worker->probeResolutions(); |
|
} |
|
|
|
Camera::~Camera() |
|
{ |
|
workerThread->exit(); |
|
workerThread->deleteLater(); |
|
} |
|
|
|
void Camera::subscribe() |
|
{ |
|
if (refcount <= 0) |
|
worker->resume(); |
|
|
|
refcount++; |
|
} |
|
|
|
void Camera::unsubscribe() |
|
{ |
|
refcount--; |
|
|
|
if (refcount <= 0) |
|
{ |
|
worker->suspend(); |
|
refcount = 0; |
|
} |
|
} |
|
|
|
vpx_image Camera::getLastVPXImage() |
|
{ |
|
QMutexLocker lock(&mutex); |
|
|
|
vpx_image img; |
|
|
|
if (currFrame.isNull()) |
|
{ |
|
img.w = 0; |
|
img.h = 0; |
|
return img; |
|
} |
|
|
|
const int w = currFrame.resolution.width(); |
|
const int h = currFrame.resolution.height(); |
|
|
|
// I420 "It comprises an NxM Y plane followed by (N/2)x(M/2) V and U planes." |
|
// http://fourcc.org/yuv.php#IYUV |
|
vpx_img_alloc(&img, VPX_IMG_FMT_VPXI420, w, h, 1); |
|
|
|
for (int y = 0; y < h; ++y) |
|
{ |
|
for (int x = 0; x < w; ++x) |
|
{ |
|
u_int8_t b = currFrame.frameData.data()[(x + y * w) * 3 + 0]; |
|
u_int8_t g = currFrame.frameData.data()[(x + y * w) * 3 + 1]; |
|
u_int8_t r = currFrame.frameData.data()[(x + y * w) * 3 + 2]; |
|
|
|
img.planes[VPX_PLANE_Y][x + y * img.stride[VPX_PLANE_Y]] = ((66 * r + 129 * g + 25 * b) >> 8) + 16; |
|
|
|
if (!(x % 2) && !(y % 2)) |
|
{ |
|
const int i = x / 2; |
|
const int j = y / 2; |
|
|
|
img.planes[VPX_PLANE_U][i + j * img.stride[VPX_PLANE_U]] = ((112 * r + -94 * g + -18 * b) >> 8) + 128; |
|
img.planes[VPX_PLANE_V][i + j * img.stride[VPX_PLANE_V]] = ((-38 * r + -74 * g + 112 * b) >> 8) + 128; |
|
} |
|
} |
|
} |
|
|
|
return img; |
|
} |
|
|
|
QList<QSize> Camera::getSupportedResolutions() |
|
{ |
|
return resolutions; |
|
} |
|
|
|
QSize Camera::getBestVideoMode() |
|
{ |
|
int bestScore = 0; |
|
QSize bestRes; |
|
|
|
for (QSize res : getSupportedResolutions()) |
|
{ |
|
int score = res.width() * res.height(); |
|
|
|
if (score > bestScore) |
|
{ |
|
bestScore = score; |
|
bestRes = res; |
|
} |
|
} |
|
|
|
return bestRes; |
|
} |
|
|
|
void Camera::setResolution(QSize res) |
|
{ |
|
worker->setProp(CV_CAP_PROP_FRAME_WIDTH, res.width()); |
|
worker->setProp(CV_CAP_PROP_FRAME_HEIGHT, res.height()); |
|
} |
|
|
|
QSize Camera::getResolution() |
|
{ |
|
return QSize(worker->getProp(CV_CAP_PROP_FRAME_WIDTH), worker->getProp(CV_CAP_PROP_FRAME_HEIGHT)); |
|
} |
|
|
|
void Camera::setProp(Camera::Prop prop, double val) |
|
{ |
|
switch (prop) |
|
{ |
|
case BRIGHTNESS: |
|
worker->setProp(CV_CAP_PROP_BRIGHTNESS, val); |
|
break; |
|
case SATURATION: |
|
worker->setProp(CV_CAP_PROP_SATURATION, val); |
|
break; |
|
case CONTRAST: |
|
worker->setProp(CV_CAP_PROP_CONTRAST, val); |
|
break; |
|
case HUE: |
|
worker->setProp(CV_CAP_PROP_HUE, val); |
|
break; |
|
} |
|
} |
|
|
|
double Camera::getProp(Camera::Prop prop) |
|
{ |
|
switch (prop) |
|
{ |
|
case BRIGHTNESS: |
|
return worker->getProp(CV_CAP_PROP_BRIGHTNESS); |
|
case SATURATION: |
|
return worker->getProp(CV_CAP_PROP_SATURATION); |
|
case CONTRAST: |
|
return worker->getProp(CV_CAP_PROP_CONTRAST); |
|
case HUE: |
|
return worker->getProp(CV_CAP_PROP_HUE); |
|
} |
|
|
|
return 0.0; |
|
} |
|
|
|
void Camera::onNewFrameAvailable(const VideoFrame frame) |
|
{ |
|
emit frameAvailable(frame); |
|
|
|
mutex.lock(); |
|
currFrame = frame; |
|
mutex.unlock(); |
|
} |
|
|
|
void Camera::onResProbingFinished(QList<QSize> res) |
|
{ |
|
resolutions = res; |
|
} |
|
|
|
Camera* Camera::getInstance() |
|
{ |
|
if (!instance) |
|
instance = new Camera(); |
|
|
|
return instance; |
|
}
|
|
|