Rabbit Remote Control 0.0.34
Loading...
Searching...
No Matches
ConnectDesktop.cpp
1// Author: Kang Lin <kl222@126.com>
2
3#include <QApplication>
4#include <QClipboard>
5#include <QTimer>
6#include <QLoggingCategory>
7#include <QWheelEvent>
8#include <QVideoFrame>
9#include <QDesktopServices>
10
11#include "ConnectDesktop.h"
12#include "ConnecterThread.h"
13
14static Q_LOGGING_CATEGORY(log, "Client.Connect.Desktop")
15static Q_LOGGING_CATEGORY(logMouse, "Client.Connect.Desktop.Mouse")
16
17#define TypeRecordVideo (QEvent::User + 1)
18class QRecordVideoEvent : public QEvent
19{
20public:
21 QRecordVideoEvent(const QImage& img): QEvent((QEvent::Type)TypeRecordVideo)
22 {
23 m_Image = img;
24 }
25 QImage GetImage()
26 {
27 return m_Image;
28 }
29private:
30 QImage m_Image;
31};
32
33int g_QtKeyboardModifiers = qRegisterMetaType<Qt::KeyboardModifiers>("KeyboardModifiers");
34int g_QtMouseButtons = qRegisterMetaType<Qt::MouseButtons>("MouseButtons");
35int g_QtMouseButton = qRegisterMetaType<Qt::MouseButton>("MouseButton");
36int g_QMessageBox_Icon = qRegisterMetaType<Qt::MouseButton>("QMessageBox::Icon");
37
38CConnectDesktop::CConnectDesktop(CConnecter *pConnecter, bool bDirectConnection)
39 : CConnect(pConnecter)
40#if HAVE_QT6_RECORD
41 , m_pParameterRecord(nullptr)
42 , m_VideoFrameInput(this)
43 , m_AudioBufferInput(this)
44 , m_AudioBufferOutput(this)
45#endif
46{
47 if(pConnecter) {
48 CFrmScroll* pScroll = qobject_cast<CFrmScroll*>(pConnecter->GetViewer());
49 if(pScroll) {
50 CFrmViewer* pView = pScroll->GetViewer();
51 if(pView)
52 SetViewer(pView, pConnecter, bDirectConnection);
53 else {
54 QString szErr = pConnecter->metaObject()->className();
55 szErr += "::GetViewer() is not CFrmViewer";
56 qWarning(log) << szErr.toStdString().c_str();
57 }
58 } else {
59 QString szErr = pConnecter->metaObject()->className();
60 szErr += "::GetViewer() is not CFrmScroll";
61 qWarning(log) << szErr.toStdString().c_str();
62 }
63 SetConnecter(pConnecter);
64 }
65
66#if HAVE_QT6_RECORD
67 bool check = connect(
68 &m_Recorder, &QMediaRecorder::errorOccurred,
69 this, [&](QMediaRecorder::Error error, const QString &errorString) {
70 qDebug(log) << "Recorder error occurred:" << error << errorString;
71 slotRecord(false);
72 emit sigError(error, errorString);
73 });
74 Q_ASSERT(check);
75 check = connect(
76 &m_Recorder, &QMediaRecorder::recorderStateChanged,
77 this, [&](QMediaRecorder::RecorderState state){
78 qDebug(log) << "Recorder state changed:" << state;
79 if(QMediaRecorder::StoppedState == state)
80 {
81 slotRecord(false);
82 if(m_pParameterRecord) {
83 qDebug(log) << "End action:"
84 << m_pParameterRecord->GetEndAction()
85 << m_Recorder.actualLocation();
86 switch(m_pParameterRecord->GetEndAction())
87 {
88 case CParameterRecord::ENDACTION::OpenFile:
89 QDesktopServices::openUrl(m_Recorder.actualLocation());
90 break;
91 case CParameterRecord::ENDACTION::OpenFolder: {
92 QFileInfo fi(m_Recorder.actualLocation().toLocalFile());
93 QDesktopServices::openUrl(
94 QUrl::fromLocalFile(fi.absolutePath()));
95 break;
96 }
97 default:
98 break;
99 }
100 }
101 }
102 });
103 Q_ASSERT(check);
104 check = connect(&m_Recorder, &QMediaRecorder::actualLocationChanged,
105 this, [&](const QUrl &location){
106 qInfo(log) << "Recorder actual location changed:" << location;
107 });
108 Q_ASSERT(check);
109#endif
110}
111
112CConnectDesktop::~CConnectDesktop()
113{
114 qDebug(log) << "CConnectDesktop::~CConnectDesktop()";
115}
116
117int CConnectDesktop::SetConnecter(CConnecter* pConnecter)
118{
119 qDebug(log) << "CConnectDesktop::SetConnecter" << pConnecter;
120 Q_ASSERT(pConnecter);
121 if(!pConnecter) return -1;
122
123 bool check = false;
124 check = connect(this, SIGNAL(sigServerName(const QString&)),
125 pConnecter, SLOT(slotSetServerName(const QString&)));
126 Q_ASSERT(check);
127 check = connect(pConnecter, SIGNAL(sigClipBoardChanged()),
128 this, SLOT(slotClipBoardChanged()));
129 Q_ASSERT(check);
130 check = connect(this, SIGNAL(sigSetClipboard(QMimeData*)),
131 pConnecter, SLOT(slotSetClipboard(QMimeData*)));
132 Q_ASSERT(check);
133#if HAVE_QT6_RECORD
134 CConnecterThread* p = qobject_cast<CConnecterThread*>(pConnecter);
135 if(p) {
136 m_pParameterRecord = &p->GetParameter()->m_Record;
137 check = connect(p, SIGNAL(sigRecord(bool)),
138 this, SLOT(slotRecord(bool)));
139 Q_ASSERT(check);
140
141 check = connect(p, SIGNAL(sigRecordPause(bool)),
142 this, SLOT(slotRecordPause(bool)));
143 Q_ASSERT(check);
144 check = connect(
145 &m_Recorder,
146 SIGNAL(recorderStateChanged(QMediaRecorder::RecorderState)),
147 p, SLOT(slotRecorderStateChanged(QMediaRecorder::RecorderState)));
148 Q_ASSERT(check);
149 }
150#endif
151 return 0;
152}
153
154int CConnectDesktop::SetViewer(CFrmViewer *pView,
155 CConnecter* pConnecter, bool bDirectConnection)
156{
157 Q_ASSERT(pView);
158 if(!pView) return -1;
159
160 bool check = false;
161 check = connect(this, SIGNAL(sigConnected()), pView, SLOT(slotConnected()));
162 Q_ASSERT(check);
163 check = connect(this, SIGNAL(sigSetDesktopSize(int, int)),
164 pView, SLOT(slotSetDesktopSize(int, int)));
165 Q_ASSERT(check);
166 check = connect(this, SIGNAL(sigServerName(const QString&)),
167 pView, SLOT(slotSetName(const QString&)));
168 Q_ASSERT(check);
169
170 check = connect(this, SIGNAL(sigUpdateRect(const QRect&, const QImage&)),
171 pView, SLOT(slotUpdateRect(const QRect&, const QImage&)));
172 Q_ASSERT(check);
173 check = connect(this, SIGNAL(sigUpdateRect(const QImage&)),
174 pView, SLOT(slotUpdateRect(const QImage&)));
175 Q_ASSERT(check);
176 check = connect(this, SIGNAL(sigUpdateCursor(const QCursor&)),
177 pView, SLOT(slotUpdateCursor(const QCursor&)));
178 Q_ASSERT(check);
179 check = connect(this, SIGNAL(sigUpdateCursorPosition(const QPoint&)),
180 pView, SLOT(slotUpdateCursorPosition(const QPoint&)));
181 Q_ASSERT(check);
182 check = connect(this, SIGNAL(sigUpdateLedState(unsigned int)),
183 pView, SLOT(slotUpdateLedState(unsigned int)));
184 Q_ASSERT(check);
185
186#if HAVE_QT6_RECORD
187 check = connect(this, SIGNAL(sigRecordVideo(bool)),
188 pView, SLOT(slotRecordVideo(bool)));
189 Q_ASSERT(check);
190 check = connect(pView, SIGNAL(sigRecordVideo(QImage)),
191 this, SLOT(slotRecordVideo(QImage)),
192 Qt::DirectConnection);
193 Q_ASSERT(check);
194#endif
195 if(bDirectConnection)
196 {
197 /* \~chinese 因为连接可能是在另一个线程中的非Qt事件处理,
198 * 它可能会阻塞线程,那会导致键盘、鼠标事件延迟,
199 * 所以这里用 Qt::DirectConnection
200 * \~english Because the connection may be a non-Qt event processing in another thread,
201 * it may block the thread, which will cause the keyboard and mouse events to be delayed.
202 * So here use Qt::DirectConnection
203 */
204 check = connect(pView, SIGNAL(sigMousePressEvent(QMouseEvent*, QPoint)),
205 this, SLOT(slotMousePressEvent(QMouseEvent*, QPoint)),
206 Qt::DirectConnection);
207 Q_ASSERT(check);
208 check = connect(pView, SIGNAL(sigMouseReleaseEvent(QMouseEvent*, QPoint)),
209 this, SLOT(slotMouseReleaseEvent(QMouseEvent*, QPoint)),
210 Qt::DirectConnection);
211 Q_ASSERT(check);
212 check = connect(pView, SIGNAL(sigMouseMoveEvent(QMouseEvent*, QPoint)),
213 this, SLOT(slotMouseMoveEvent(QMouseEvent*, QPoint)),
214 Qt::DirectConnection);
215 Q_ASSERT(check);
216 check = connect(pView, SIGNAL(sigWheelEvent(QWheelEvent*, QPoint)),
217 this, SLOT(slotWheelEvent(QWheelEvent*, QPoint)),
218 Qt::DirectConnection);
219 Q_ASSERT(check);
220 check = connect(pView, SIGNAL(sigKeyPressEvent(QKeyEvent*)),
221 this, SLOT(slotKeyPressEvent(QKeyEvent*)),
222 Qt::DirectConnection);
223 Q_ASSERT(check);
224 check = connect(pView, SIGNAL(sigKeyReleaseEvent(QKeyEvent*)),
225 this, SLOT(slotKeyReleaseEvent(QKeyEvent*)),
226 Qt::DirectConnection);
227 Q_ASSERT(check);
228 } else {
229 check = connect(pView, SIGNAL(sigMousePressEvent(QMouseEvent*, QPoint)),
230 this, SLOT(slotMousePressEvent(QMouseEvent*, QPoint)));
231 Q_ASSERT(check);
232 check = connect(pView, SIGNAL(sigMouseReleaseEvent(QMouseEvent*, QPoint)),
233 this, SLOT(slotMouseReleaseEvent(QMouseEvent*, QPoint)));
234 Q_ASSERT(check);
235 check = connect(pView, SIGNAL(sigMouseMoveEvent(QMouseEvent*, QPoint)),
236 this, SLOT(slotMouseMoveEvent(QMouseEvent*, QPoint)));
237 Q_ASSERT(check);
238 check = connect(pView, SIGNAL(sigWheelEvent(QWheelEvent*, QPoint)),
239 this, SLOT(slotWheelEvent(QWheelEvent*, QPoint)));
240 Q_ASSERT(check);
241 check = connect(pView, SIGNAL(sigKeyPressEvent(QKeyEvent*)),
242 this, SLOT(slotKeyPressEvent(QKeyEvent*)));
243 Q_ASSERT(check);
244 check = connect(pView, SIGNAL(sigKeyReleaseEvent(QKeyEvent*)),
245 this, SLOT(slotKeyReleaseEvent(QKeyEvent*)));
246 Q_ASSERT(check);
247 }
248
249 return 0;
250}
251
252void CConnectDesktop::slotWheelEvent(QWheelEvent *event, QPoint pos)
253{
254 QWheelEvent* e = new QWheelEvent(
255 pos,
256#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
257 event->globalPosition(),
258#else
259 event->globalPos(),
260#endif
261 event->pixelDelta(), event->angleDelta(), event->buttons(),
262 event->modifiers(), event->phase(), event->inverted(), event->source());
263 QCoreApplication::postEvent(this, e);
264 WakeUp();
265}
266
267void CConnectDesktop::slotMouseMoveEvent(QMouseEvent *event, QPoint pos)
268{
269 QMouseEvent* e = new QMouseEvent(event->type(), pos, event->button(),
270 event->buttons(), event->modifiers());
271 QCoreApplication::postEvent(this, e);
272 WakeUp();
273}
274
275void CConnectDesktop::slotMousePressEvent(QMouseEvent *event, QPoint pos)
276{
277 QMouseEvent* e = new QMouseEvent(event->type(), pos, event->button(),
278 event->buttons(), event->modifiers());
279 QCoreApplication::postEvent(this, e);
280 WakeUp();
281}
282
283void CConnectDesktop::slotMouseReleaseEvent(QMouseEvent *event, QPoint pos)
284{
285 QMouseEvent* e = new QMouseEvent(event->type(), pos, event->button(),
286 event->buttons(), event->modifiers());
287 QCoreApplication::postEvent(this, e);
288 WakeUp();
289}
290
291void CConnectDesktop::slotKeyPressEvent(QKeyEvent *event)
292{
293 QKeyEvent* e = new QKeyEvent(event->type(), event->key(),
294 event->modifiers(), event->text());
295 QCoreApplication::postEvent(this, e);
296 WakeUp();
297}
298
299void CConnectDesktop::slotKeyReleaseEvent(QKeyEvent *event)
300{
301 QKeyEvent* e = new QKeyEvent(event->type(), event->key(),
302 event->modifiers(), event->text());
303 QCoreApplication::postEvent(this, e);
304 WakeUp();
305}
306
307void CConnectDesktop::mouseMoveEvent(QMouseEvent *event)
308{
309 qDebug(logMouse) << "Need to implement CConnectDesktop::mouseMoveEvent";
310}
311
312void CConnectDesktop::mousePressEvent(QMouseEvent *event)
313{
314 qDebug(logMouse) << "Need to implement CConnectDesktop::mousePressEvent";
315}
316
317void CConnectDesktop::mouseReleaseEvent(QMouseEvent *event)
318{
319 qDebug(logMouse) << "Need to implement CConnectDesktop::mouseReleaseEvent";
320}
321
322void CConnectDesktop::wheelEvent(QWheelEvent *event)
323{
324 qDebug(logMouse) << "Need to implement CConnectDesktop::wheelEvent";
325}
326
327void CConnectDesktop::keyPressEvent(QKeyEvent *event)
328{
329 qDebug(logMouse) << "Need to implement CConnectDesktop::keyPressEvent";
330}
331
332void CConnectDesktop::keyReleaseEvent(QKeyEvent *event)
333{
334 qDebug(logMouse) << "Need to implement CConnectDesktop::keyReleaseEvent";
335}
336
338{
339 return 0;
340}
341
342bool CConnectDesktop::event(QEvent *event)
343{
344 //qDebug(log) << "CConnectDesktop::event" << event;
345 switch (event->type()) {
346 case QEvent::MouseButtonPress:
347 case QEvent::MouseButtonDblClick:
348 mousePressEvent((QMouseEvent*)event);
349 break;
350 case QEvent::MouseButtonRelease:
351 mouseReleaseEvent((QMouseEvent*)event);
352 break;
353 case QEvent::MouseMove:
354 mouseMoveEvent((QMouseEvent*)event);
355 break;
356 case QEvent::Wheel:
357 wheelEvent((QWheelEvent*)event);
358 break;
359 case QEvent::KeyPress:
360 keyPressEvent((QKeyEvent*)event);
361 break;
362 case QEvent::KeyRelease:
363 keyReleaseEvent((QKeyEvent*)event);
364 break;
365#if HAVE_QT6_RECORD
366 case TypeRecordVideo:
367 RecordVideo((QRecordVideoEvent*)event);
368 break;
369#endif
370 default:
371 return QObject::event(event);
372 }
373
374 event->accept();
375 return true;
376}
377
378#if HAVE_QT6_RECORD
379void CConnectDesktop::slotRecord(bool bRecord)
380{
381 qDebug(log) << Q_FUNC_INFO << bRecord;
382 if(bRecord) {
383 if(QMediaRecorder::RecordingState == m_Recorder.recorderState())
384 return;
385 (*m_pParameterRecord) >> m_Recorder;
386 m_CaptureSession.setVideoFrameInput(&m_VideoFrameInput);
387 m_CaptureSession.setRecorder(&m_Recorder);
388 m_Recorder.record();
389 } else {
390 m_Recorder.stop();
391 m_CaptureSession.setVideoFrameInput(nullptr);
392 m_CaptureSession.setAudioBufferInput(nullptr);
393 m_CaptureSession.setRecorder(nullptr);
394 }
395 emit sigRecordVideo(bRecord);
396}
397
398void CConnectDesktop::slotRecordPause(bool bPause)
399{
400 qDebug(log) << Q_FUNC_INFO << bPause;
401 if(bPause) {
402 if(m_Recorder.recorderState() == QMediaRecorder::RecordingState)
403 m_Recorder.pause();
404 } else {
405 if(m_Recorder.recorderState() == QMediaRecorder::PausedState)
406 m_Recorder.record();
407 }
408}
409
410void CConnectDesktop::slotRecordVideo(const QImage &img)
411{
413 if(!e) return;
414 QCoreApplication::postEvent(this, e);
415 WakeUp();
416}
417
418void CConnectDesktop::RecordVideo(QRecordVideoEvent *e)
419{
420 qDebug(log) << "Update image";
421 if(!e) return;
422 if(QMediaRecorder::RecordingState != m_Recorder.recorderState()) {
423 qCritical(log) << "Recorder is inavailable";
424 return;
425 }
426 QVideoFrame frame(e->GetImage());
427 bool bRet = m_VideoFrameInput.sendVideoFrame(frame);
428 if(!bRet) {
429 //TODO: 放入未成功发送队列,
430 // 当 QVideoFrameInput::readyToSendVideoFrame() 时,再发送
431 qDebug(log) << "m_VideoFrameInput.sendVideoFrame fail";
432 }
433}
434
436{
437 slotRecord(false);
438 return CConnect::Disconnect();
439}
440#endif
CConnectDesktop(CConnecter *pConnecter, bool bDirectConnection=true)
void sigUpdateRect(const QRect &r, const QImage &image)
Notify the CFrmView update image.
virtual int WakeUp()
Wake up Connect thread(background thread)
virtual void slotClipBoardChanged()=0
Be called when the clip board change.
Connect interface.
Definition Connect.h:45
void sigError(const int nError, const QString &szError=QString())
Triggered when an error is generated.
void sigConnected()
Emitted when the plugin is successfully connected.
virtual int Disconnect()
Disconnect.
Definition Connect.cpp:89
virtual CParameterBase * GetParameter()
Get parameter.
It starts a background thread by default.
Connecter interface.
Definition Connecter.h:62
virtual QWidget * GetViewer()=0
Get Viewer.
The scroll form class.
Definition FrmScroll.h:17
A widget which displays output image from a CConnectDesktop and sends input keypresses and mouse acti...
Definition FrmViewer.h:49