344 lines
9.0 KiB
C++
344 lines
9.0 KiB
C++
/*
|
||
* Copyright (C) 2023, KylinSoft Co., Ltd.
|
||
*
|
||
* 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, 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 program; if not, see <http://www.gnu.org/licenses/>.
|
||
*
|
||
**/
|
||
#include "biometricauthwidget.h"
|
||
#include <QLabel>
|
||
#include <QDebug>
|
||
#include <QDBusUnixFileDescriptor>
|
||
#include <unistd.h>
|
||
#include <pwd.h>
|
||
|
||
#include <opencv2/core/core.hpp>
|
||
#include <opencv2/imgproc/imgproc.hpp>
|
||
#include <opencv2/highgui/highgui.hpp>
|
||
#include <opencv2/opencv.hpp>
|
||
|
||
#include "giodbus.h"
|
||
|
||
BiometricAuthWidget::BiometricAuthWidget(BiometricProxy *proxy, QWidget *parent) :
|
||
QWidget(parent),
|
||
proxy(proxy),
|
||
isInAuth(false),
|
||
movieTimer(nullptr),
|
||
failedCount(0),
|
||
timeoutCount(0),
|
||
beStopped(false),
|
||
retrytimer(nullptr),
|
||
usebind(false)
|
||
{
|
||
usebind = getAuthDouble();
|
||
initUI();
|
||
resize(400, 260);
|
||
|
||
if(this->proxy)
|
||
{
|
||
connect(this->proxy, &BiometricProxy::StatusChanged,
|
||
this, &BiometricAuthWidget::onStatusChanged);
|
||
|
||
connect(this->proxy, &BiometricProxy::FrameWritten,
|
||
this, &BiometricAuthWidget::onFrameWritten);
|
||
}
|
||
|
||
}
|
||
|
||
void BiometricAuthWidget::initUI()
|
||
{
|
||
//显示提示信息
|
||
lblNotify = new QLabel(this);
|
||
lblNotify->setWordWrap(true);
|
||
lblNotify->setAlignment(Qt::AlignHCenter | Qt::AlignBottom);
|
||
|
||
//显示当前设备
|
||
lblDevice = new QLabel(this);
|
||
lblDevice->setWordWrap(true);
|
||
lblDevice->setAlignment(Qt::AlignCenter);
|
||
|
||
//显示图片
|
||
lblImage = new QLabel(this);
|
||
lblImage->setFixedSize(100, 100);
|
||
|
||
}
|
||
|
||
|
||
void BiometricAuthWidget::resizeEvent(QResizeEvent */*event*/)
|
||
{
|
||
lblNotify->setGeometry(0, 0, width(), 45);
|
||
lblDevice->setGeometry(0, lblNotify->geometry().bottom()+5, width(), 30);
|
||
lblImage->setGeometry((width() - lblImage->width()) / 2,
|
||
lblDevice->geometry().bottom() + 10,
|
||
lblImage->width(), lblImage->height());
|
||
//qDebug()
|
||
}
|
||
|
||
void BiometricAuthWidget::startAuth(DeviceInfoPtr device, int uid)
|
||
{
|
||
if(!proxy)
|
||
{
|
||
qWarning() << "BiometricProxy doesn't exist.";
|
||
return;
|
||
}
|
||
|
||
if(isInAuth)
|
||
{
|
||
qDebug() << "Identification is currently under way, stop it";
|
||
stopAuth();
|
||
}
|
||
|
||
this->device = device;
|
||
this->uid = uid;
|
||
this->userName = getpwuid(uid)->pw_name;
|
||
this->failedCount = 0;
|
||
this->timeoutCount = 0;
|
||
this->beStopped = false;
|
||
proxy->StopOps(device->id);
|
||
startAuth_();
|
||
|
||
if(device->deviceType != DeviceType::Type::Face){
|
||
updateImage(1);
|
||
}
|
||
|
||
}
|
||
|
||
void BiometricAuthWidget::startAuth_()
|
||
{
|
||
lblDevice->setText(tr("Current device: ") + device->shortName);
|
||
|
||
//qDebug().noquote() << QString("Identify:[drvid: %1, uid: %2]").arg(1).arg(2);
|
||
|
||
isInAuth = true;
|
||
dup_fd = -1;
|
||
|
||
QDBusPendingCall call = proxy->Identify(device->id, uid);
|
||
QDBusPendingCallWatcher *watcher = new QDBusPendingCallWatcher(call, this);
|
||
connect(watcher, &QDBusPendingCallWatcher::finished,
|
||
this, &BiometricAuthWidget::onIdentifyComplete);
|
||
|
||
}
|
||
|
||
void BiometricAuthWidget::stopAuth()
|
||
{
|
||
beStopped = true;
|
||
if(!isInAuth)
|
||
{
|
||
return;
|
||
}
|
||
|
||
proxy->StopOps(device->id);
|
||
if(retrytimer&&retrytimer->isActive()){
|
||
retrytimer->stop();
|
||
delete retrytimer;
|
||
retrytimer = nullptr;
|
||
}
|
||
isInAuth = false;
|
||
updateImage(0);
|
||
}
|
||
|
||
void BiometricAuthWidget::onIdentifyComplete(QDBusPendingCallWatcher *watcher)
|
||
{
|
||
QDBusPendingReply<int, int> reply = *watcher;
|
||
if(reply.isError())
|
||
{
|
||
qWarning() << "Identify error: " << reply.error().message();
|
||
Q_EMIT authComplete(false);
|
||
updateImage(0);
|
||
return;
|
||
}
|
||
int result = reply.argumentAt(0).toInt();
|
||
int authUid = reply.argumentAt(1).toInt();
|
||
|
||
// 特征识别成功,而且用户id匹配
|
||
if(result == DBUS_RESULT_SUCCESS && authUid == uid)
|
||
{
|
||
qDebug() << "Identify success";
|
||
Q_EMIT authComplete(true);
|
||
}
|
||
// 特征识别不匹配
|
||
else if(result == DBUS_RESULT_NOTMATCH)
|
||
{
|
||
if(usebind){
|
||
Q_EMIT authComplete(false);
|
||
return;
|
||
}
|
||
qDebug() << "Identify failed";
|
||
failedCount++;
|
||
if(failedCount >= GetMaxFailedAutoRetry(userName))
|
||
{
|
||
Q_EMIT authComplete(false);
|
||
}
|
||
else
|
||
{
|
||
lblNotify->setText(tr("Identify failed, Please retry."));
|
||
if(!beStopped){
|
||
// QTimer::singleShot(1000, this, &BiometricAuthWidget::startAuth_);
|
||
if(!retrytimer){
|
||
retrytimer = new QTimer(this);
|
||
retrytimer->setSingleShot(true);
|
||
connect(retrytimer, &QTimer::timeout, this, &BiometricAuthWidget::startAuth_);
|
||
}
|
||
retrytimer->start(1000);
|
||
}
|
||
}
|
||
}
|
||
//识别发生错误
|
||
else if(result == DBUS_RESULT_ERROR)
|
||
{
|
||
if(usebind){
|
||
Q_EMIT authComplete(false);
|
||
return;
|
||
}
|
||
StatusReslut ret = proxy->UpdateStatus(device->id);
|
||
//识别操作超时
|
||
if(ret.result == 0 && ret.opsStatus == OPS_IDENTIFY_TIMEOUT)
|
||
{
|
||
timeoutCount++;
|
||
if(timeoutCount >= GetMaxTimeoutAutoRetry(userName))
|
||
{
|
||
Q_EMIT authComplete(false);
|
||
}
|
||
else
|
||
{
|
||
QTimer::singleShot(1000, [&]{
|
||
if(!beStopped)
|
||
{
|
||
startAuth_();
|
||
}
|
||
});
|
||
}
|
||
}else{
|
||
Q_EMIT authComplete(false);
|
||
}
|
||
}else{
|
||
Q_EMIT authComplete(false);
|
||
}
|
||
updateImage(0);
|
||
}
|
||
|
||
void BiometricAuthWidget::onFrameWritten(int drvid)
|
||
{
|
||
|
||
if(dup_fd == -1){
|
||
dup_fd = get_server_gvariant_stdout(drvid);
|
||
}
|
||
|
||
if(dup_fd <= 0)
|
||
return ;
|
||
|
||
cv::Mat img;
|
||
lseek(dup_fd, 0, SEEK_SET);
|
||
char base64_bufferData[1024*1024];
|
||
int rc = read(dup_fd, base64_bufferData, 1024*1024);
|
||
printf("rc = %d\n", rc);
|
||
|
||
cv::Mat mat2(1, sizeof(base64_bufferData), CV_8U, base64_bufferData);
|
||
img = cv::imdecode(mat2, cv::IMREAD_COLOR);
|
||
cv::cvtColor(img,img,cv::COLOR_BGR2RGB);
|
||
|
||
QImage srcQImage = QImage((uchar*)(img.data), img.cols, img.rows, QImage::Format_RGB888);
|
||
lblImage->setFixedSize(160,160);
|
||
lblImage->setGeometry((width() - lblImage->width()) / 2,
|
||
lblDevice->geometry().bottom() + 10,
|
||
lblImage->width(), lblImage->height());
|
||
lblImage->setPixmap(QPixmap::fromImage(srcQImage).scaled(lblImage->size()));
|
||
|
||
}
|
||
|
||
void BiometricAuthWidget::onStatusChanged(int drvid, int status)
|
||
{
|
||
if(!isInAuth)
|
||
{
|
||
return;
|
||
}
|
||
if(drvid != device->id)
|
||
{
|
||
return;
|
||
}
|
||
|
||
// 显示来自服务的提示信息
|
||
if(status == STATUS_NOTIFY)
|
||
{
|
||
QString notifyMsg = proxy->GetNotifyMesg(drvid);
|
||
lblNotify->setText(notifyMsg);
|
||
}
|
||
}
|
||
|
||
static int count = 0;
|
||
void BiometricAuthWidget::updateImage(int type)
|
||
{
|
||
if(device->deviceType == DeviceType::Type::Face)
|
||
return ;
|
||
|
||
if(type == 0)
|
||
{
|
||
if(movieTimer && movieTimer->isActive())
|
||
{
|
||
movieTimer->stop();
|
||
}
|
||
|
||
QString imagePath = QString(UKUI_BIOMETRIC_IMAGES_PATH "%1/01.png")
|
||
.arg(DeviceType::getDeviceType(device->deviceType));
|
||
setImage(imagePath);
|
||
}
|
||
else
|
||
{
|
||
if(!movieTimer)
|
||
{
|
||
movieTimer = new QTimer(this);
|
||
movieTimer->setInterval(100);
|
||
connect(movieTimer, &QTimer::timeout,
|
||
this, &BiometricAuthWidget::onMoviePixmapUpdate);
|
||
}
|
||
count = 0;
|
||
movieTimer->start();
|
||
}
|
||
}
|
||
|
||
void BiometricAuthWidget::onMoviePixmapUpdate()
|
||
{
|
||
if(count >= 18)
|
||
{
|
||
count = 0;
|
||
}
|
||
count++;
|
||
QString fileName = (count < 10 ? "0" : "") + QString::number(count);
|
||
QString imagePath = QString(UKUI_BIOMETRIC_IMAGES_PATH "%1/%2.png")
|
||
.arg(DeviceType::getDeviceType(device->deviceType))
|
||
.arg(fileName);
|
||
setImage(imagePath);
|
||
}
|
||
|
||
void BiometricAuthWidget::setImage(const QString &path)
|
||
{
|
||
QPixmap image(path);
|
||
image = image.scaled(lblImage->width(), lblImage->height(),
|
||
Qt::IgnoreAspectRatio, Qt::SmoothTransformation);
|
||
|
||
lblImage->setPixmap(image);
|
||
}
|
||
|
||
bool BiometricAuthWidget::getAuthDouble()
|
||
{
|
||
QSettings settings("/etc/biometric-auth/ukui-biometric.conf", QSettings::IniFormat);
|
||
bool distribId = settings.value("DoubleAuth").toBool();
|
||
return distribId;
|
||
}
|
||
|
||
void BiometricAuthWidget::setMinImage(float val)
|
||
{
|
||
resize(400,100+100*val);
|
||
lblImage->setFixedSize(100*val, 100*val);
|
||
}
|