Rabbit Remote Control 0.0.33
Loading...
Searching...
No Matches
Client.cpp
1// Author: Kang Lin <kl222@126.com>
2
3#include <QPluginLoader>
4#include <QKeyEvent>
5#include <QtPlugin>
6#include <QFile>
7#include <QApplication>
8#include <QSettings>
9#include <QLoggingCategory>
10#include <QMessageBox>
11#include <QCheckBox>
12
13#include "Client.h"
14#include "RabbitCommonDir.h"
15#include "RabbitCommonTools.h"
16#include "FrmParameterClient.h"
17#include "FrmViewer.h"
18#include "Channel.h"
19#include "ParameterRecordUI.h"
20
21static Q_LOGGING_CATEGORY(log, "Client")
22
23CClient::CClient(QObject *parent, QString szFile) : QObject(parent),
24 m_FileVersion(1) //TODO: update version it if update data
25{
26 bool check = false;
27//#if defined (_DEBUG) || !defined(BUILD_SHARED_LIBS)
28// Q_INIT_RESOURCE(translations_Client);
29//#endif
30
31 qApp->installEventFilter(this);
32
33 m_Translator = RabbitCommon::CTools::Instance()->InstallTranslator(
34 "Client", RabbitCommon::CTools::TranslationType::Library);
35
36 CChannel::InitTranslation();
37 m_szSettingsFile = szFile;
38 m_pParameterClient = new CParameterClient();
39 if(m_pParameterClient) {
40 LoadSettings(m_szSettingsFile);
41 check = connect(m_pParameterClient, SIGNAL(sigHookKeyboardChanged()),
42 this, SLOT(slotHookKeyboardChanged()));
43 Q_ASSERT(check);
44 slotHookKeyboardChanged();
45 } else {
46 qCritical(log) << "new CParameterClient() fail";
47 Q_ASSERT(m_pParameterClient);
48 }
49
50 LoadPlugins();
51}
52
53CClient::~CClient()
54{
55 qDebug(log) << "CClient::~CClient()";
56
57 if(m_pParameterClient)
58 delete m_pParameterClient;
59
60 qApp->installEventFilter(nullptr);
61
62 if(m_Translator)
63 RabbitCommon::CTools::Instance()->RemoveTranslator(m_Translator);
64
65 CChannel::RemoveTranslation();
66
67//#if defined (_DEBUG) || !defined(BUILD_SHARED_LIBS)
68// Q_CLEANUP_RESOURCE(translations_Client);
69//#endif
70}
71
72int CClient::LoadPlugins()
73{
74 int nRet = 0;
75 foreach (QObject *plugin, QPluginLoader::staticInstances())
76 {
77 CPluginClient* p = qobject_cast<CPluginClient*>(plugin);
78 if(p)
79 {
80 if(m_Plugins.find(p->Id()) == m_Plugins.end())
81 {
82 qInfo(log) << "Success: Load plugin" << p->Name();
83 AppendPlugin(p);
84 }
85 else
86 qWarning(log) << "The plugin" << p->Name() << " is exist.";
87 }
88 }
89
90 QString szPath = RabbitCommon::CDir::Instance()->GetDirPlugins();
91#if !defined (Q_OS_ANDROID)
92 szPath = szPath + QDir::separator() + "Client";
93#endif
94
95 QStringList filters;
96#if defined (Q_OS_WINDOWS) || defined(Q_OS_WIN)
97 filters << "*PluginClient*.dll";
98#else
99 filters << "*PluginClient*.so";
100#endif
101 nRet = FindPlugins(szPath, filters);
102 if(!m_szDetails.isEmpty())
103 m_szDetails = tr("### Plugins") + "\n" + m_szDetails;
104
105 qDebug(log) << ("Client details:\n" + Details()).toStdString().c_str();
106 return nRet;
107}
108
109int CClient::FindPlugins(QDir dir, QStringList filters)
110{
111 QString fileName;
112 if(filters.isEmpty())
113 {
114#if defined (Q_OS_WINDOWS) || defined(Q_OS_WIN)
115 filters << "*PluginClient*.dll";
116#else
117 filters << "*PluginClient*.so";
118#endif
119 }
120
121 QString szCurrentPath = QDir::currentPath();
122 QStringList files = dir.entryList(filters, QDir::Files | QDir::NoDotAndDotDot);
123 if(!files.isEmpty())
124 {
125 //This method is invalid
126 //QCoreApplication::addLibraryPath(QDir::cleanPath(dir.absolutePath()));
127
128 QDir::setCurrent(QDir::cleanPath(dir.absolutePath()));
129
130 // This method is valid
131//#if defined(Q_OS_WINDOWS)
132// QString szPath = QString::fromLocal8Bit(qgetenv("PATH"));
133// szPath += ";";
134// szPath += QDir::cleanPath(dir.absolutePath());
135// qputenv("PATH", szPath.toLatin1());
136//#endif
137 }
138
139 foreach (fileName, files) {
140 QString szPlugins = dir.absoluteFilePath(fileName);
141 QPluginLoader loader(szPlugins);
142 QObject *plugin = loader.instance();
143 if (plugin) {
144 CPluginClient* p = qobject_cast<CPluginClient*>(plugin);
145 if(p)
146 {
147 if(m_Plugins.find(p->Id()) == m_Plugins.end())
148 {
149 qInfo(log) << "Success: Load plugin"
150 << p->Name() << "from" << szPlugins;
151 AppendPlugin(p);
152 }
153 else
154 qWarning(log) << "The plugin [" << p->Name() << "] is exist.";
155 }
156 }else{
157 QString szMsg;
158 szMsg = "Error: Load plugin fail from " + szPlugins;
159 if(!loader.errorString().isEmpty())
160 szMsg += "; Error: " + loader.errorString();
161 qCritical(log) << szMsg.toStdString().c_str();
162 }
163 }
164
165 foreach (fileName, dir.entryList(QDir::Dirs | QDir::NoDotAndDotDot)) {
166 QDir pluginDir = dir;
167 if(pluginDir.cd(fileName))
168 FindPlugins(pluginDir, filters);
169 }
170
171 QDir::setCurrent(szCurrentPath);
172
173 return 0;
174}
175
176int CClient::AppendPlugin(CPluginClient *p)
177{
178 if(!p) return -1;
179 m_Plugins.insert(p->Id(), p);
180 //p->InitTranslator();
181 int val = 0;
182 bool bRet = QMetaObject::invokeMethod(
183 p,
184 "InitTranslator",
185 Qt::DirectConnection,
186 Q_RETURN_ARG(int, val));
187 if(!bRet || val)
188 {
189 qCritical(log) << "The plugin" << p->Name()
190 << "initial translator fail" << bRet << val;
191 }
192
193 m_szDetails += "#### " + p->DisplayName() + "\n"
194 + tr("Version:") + " " + p->Version() + " \n"
195 + p->Description() + "\n";
196 if(!p->Details().isEmpty())
197 m_szDetails += p->Details() + "\n";
198
199 return 0;
200}
201
204{
205 CConnecter* pConnecter = nullptr;
206 auto it = m_Plugins.find(id);
207 if(m_Plugins.end() != it)
208 {
209 bool bRet = 0;
210 qDebug(log) << "CreateConnecter id:" << id;
211 auto plugin = it.value();
212 if(plugin) {
213 //p = plugin->CreateConnecter(id);
214 bRet = QMetaObject::invokeMethod(
215 plugin,
216 "CreateConnecter",
217 Qt::DirectConnection,
218 Q_RETURN_ARG(CConnecter*, pConnecter),
219 Q_ARG(QString, id),
220 Q_ARG(CParameterClient*, m_pParameterClient));
221 if(!bRet) {
222 qCritical(log) << "Create CConnecter fail.";
223 return nullptr;
224 }
225 }
226 }
227 return pConnecter;
228}
230
232{
233 qDebug(log) << Q_FUNC_INFO;
234 if(!p) return 0;
235
236 CPluginClient* pPluginClient = nullptr;
237 //pPluginClient->GetPlugClient();
238 bool bRet = QMetaObject::invokeMethod(
239 p,
240 "GetPlugClient",
241 Qt::DirectConnection,
242 Q_RETURN_ARG(CPluginClient*, pPluginClient));
243
244 if(bRet && pPluginClient) {
245 int nRet = 0;
246 //pPluginClient->DeleteConnecter(p);
247 bRet = QMetaObject::invokeMethod(
248 pPluginClient,
249 "DeleteConnecter",
250 Qt::DirectConnection,
251 Q_RETURN_ARG(int, nRet),
252 Q_ARG(CConnecter*, p));
253 if(!bRet) {
254 nRet = -1;
255 qCritical(log) << "Call pPluginClient->DeleteConnecter(p) fail";
256 }
257 return nRet;
258 }
259
260 qCritical(log) << "Get CClient fail.";
261 return -1;
262}
263
264CConnecter* CClient::LoadConnecter(const QString &szFile)
265{
266 CConnecter* pConnecter = nullptr;
267 if(szFile.isEmpty()) return nullptr;
268
269 QSettings set(szFile, QSettings::IniFormat);
270 m_FileVersion = set.value("Manage/FileVersion", m_FileVersion).toInt();
271 QString id = set.value("Plugin/ID").toString();
272 QString protocol = set.value("Plugin/Protocol").toString();
273 QString name = set.value("Plugin/Name").toString();
274 Q_UNUSED(name);
275 qDebug(log) << "LoadConnecter protocol:" << protocol
276 << "name:" << name << "id:" << id;
277 pConnecter = CreateConnecter(id);
278 if(pConnecter) {
279 int nRet = false;
280 //bRet = pConnecter->Load(szFile);
281 bool bRet = QMetaObject::invokeMethod(
282 pConnecter,
283 "Load",
284 Qt::DirectConnection,
285 Q_RETURN_ARG(int, nRet),
286 Q_ARG(QString, szFile));
287 if(!bRet) {
288 qCritical(log) << "Call pConnecter->Load(szFile) fail.";
289 return nullptr;
290 }
291 if(nRet) {
292 qCritical(log) << "Load parameter fail" << nRet;
293 DeleteConnecter(pConnecter);
294 return nullptr;
295 }
296 pConnecter->SetSettingsFile(szFile);
297 }
298 else
299 qCritical(log) << "Don't create connecter:" << protocol;
300
301 return pConnecter;
302}
303
305{
306 if(!pConnecter) return -1;
307
308 QString szFile = pConnecter->GetSettingsFile();
309 if(szFile.isEmpty())
310 szFile = RabbitCommon::CDir::Instance()->GetDirUserData()
311 + QDir::separator()
312 + pConnecter->Id()
313 + ".rrc";
314
315 QSettings set(szFile, QSettings::IniFormat);
316
317 CPluginClient* pPluginClient = nullptr; //pConnecter->m_pPluginClient;
318 bool bRet = QMetaObject::invokeMethod(
319 pConnecter,
320 "GetPlugClient",
321 Qt::DirectConnection,
322 Q_RETURN_ARG(CPluginClient*, pPluginClient));
323 if(!bRet || !pPluginClient)
324 {
325 qCritical(log) << "Get plugin client fail";
326 }
327 Q_ASSERT(pPluginClient);
328
329 set.setValue("Manage/FileVersion", m_FileVersion);
330 set.setValue("Plugin/ID", pPluginClient->Id());
331 set.setValue("Plugin/Protocol", pPluginClient->Protocol());
332 set.setValue("Plugin/Name", pPluginClient->Name());
333 int nRet = 0;
334 //nRet = pConnecter->Save(szFile);
335 bRet = QMetaObject::invokeMethod(
336 pConnecter,
337 "Save",
338 Qt::DirectConnection,
339 Q_RETURN_ARG(int, nRet),
340 Q_ARG(QString, szFile));
341 if(!bRet) {
342 qCritical(log) << "Call pConnecter->Save(szFile) fail.";
343 return -1;
344 }
345 if(nRet) {
346 qCritical(log) << "Save parameter fail" << nRet;
347 return -2;
348 }
349 return 0;
350}
351
352int CClient::LoadSettings(const QString szFile)
353{
354 if(!m_pParameterClient) {
355 qCritical(log) << "The m_pParameterClient is nullptr";
356 Q_ASSERT_X(m_pParameterClient, "CClient", "The m_pParameterClient is nullptr");
357 return -1;
358 }
359
360 QString s = szFile;
361 if(s.isEmpty())
362 s = m_szSettingsFile;
363 return m_pParameterClient->CParameter::Load(s);
364}
365
366int CClient::SaveSettings(const QString szFile)
367{
368 if(!m_pParameterClient) {
369 qCritical(log) << "The m_pParameterClient is nullptr";
370 Q_ASSERT_X(m_pParameterClient, "CClient", "The m_pParameterClient is nullptr");
371 return -1;
372 }
373
374 QString s = szFile;
375 if(s.isEmpty())
376 s = m_szSettingsFile;
377 return m_pParameterClient->CParameter::Save(s);
378}
379
380QList<QWidget*> CClient::GetSettingsWidgets(QWidget* parent)
381{
382 QList<QWidget*> lstWidget;
383
384 CFrmParameterClient* pClient = new CFrmParameterClient(parent);
385 if(pClient) {
386 pClient->SetParameter(m_pParameterClient);
387 lstWidget.push_back(pClient);
388 }
389
390 CParameterRecordUI* pRecord = new CParameterRecordUI(parent);
391 if(pRecord) {
392 pRecord->SetParameter(&m_pParameterClient->m_Record);
393 lstWidget.push_back(pRecord);
394 }
395
396 return lstWidget;
397}
398
400{
401 int nRet = 0;
402 QMap<QString, CPluginClient*>::iterator it;
403 for(it = m_Plugins.begin(); it != m_Plugins.end(); it++)
404 {
405 nRet = handle->onProcess(it.key(), it.value());
406 if(nRet)
407 return nRet;
408 }
409 return nRet;
410}
411
412#if HAS_CPP_11
413int CClient::EnumPlugins(std::function<int(const QString &, CPluginClient *)> cb)
414{
415 int nRet = 0;
416 QMap<QString, CPluginClient*>::iterator it;
417 for(it = m_Plugins.begin(); it != m_Plugins.end(); it++)
418 {
419 nRet = cb(it.key(), it.value());
420 if(nRet)
421 return nRet;
422 }
423 return nRet;
424}
425#endif
426
427const QString CClient::Details() const
428{
429 return m_szDetails;
430}
431
432void CClient::slotHookKeyboardChanged()
433{
434 if(m_pParameterClient->GetHookKeyboard())
435 {
436 if(!RabbitCommon::CTools::Instance()->HasAdministratorPrivilege()
437 && m_pParameterClient->GetHookShowAdministratorPrivilege())
438 {
439 int nRet = 0;
440 QMessageBox msg(
441 QMessageBox::Warning, tr("Warning"),
442 tr("The programe is not administrator privilege.\n"
443 "Don't disable system shortcuts(eg: Ctrl+Alt+del).\n"
444 "Restart program by administrator?"),
445 QMessageBox::Yes | QMessageBox::No);
446 msg.setCheckBox(new QCheckBox(tr("Always shown"), &msg));
447 msg.checkBox()->setCheckable(true);
448 msg.checkBox()->setChecked(
449 m_pParameterClient->GetHookShowAdministratorPrivilege());
450 nRet = msg.exec();
451 if(QMessageBox::Yes == nRet) {
452 RabbitCommon::CTools::Instance()->StartWithAdministratorPrivilege(true);
453 }
454 if(m_pParameterClient->GetHookShowAdministratorPrivilege()
455 != msg.checkBox()->isChecked()) {
456 m_pParameterClient->SetHookShowAdministratorPrivilege(
457 msg.checkBox()->isChecked());
458 SaveSettings();
459 }
460 }
461 m_Hook = QSharedPointer<CHook>(CHook::GetHook());
462 } else {
463 if(m_Hook)
464 m_Hook.reset();
465 }
466}
467
468bool CClient::eventFilter(QObject *watched, QEvent *event)
469{
470 if(QEvent::KeyPress == event->type() || QEvent::KeyRelease == event->type())
471 {
472 //qDebug(log) << "eventFilter:" << event;
473 bool bProcess = false;
474 QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
475 int key = keyEvent->key();
476 switch (key) {
477 case Qt::Key_Meta:
478#if defined(Q_OS_WIN)
479 key = Qt::Key_Super_L;
480#endif
481#if defined(Q_OS_MACOS)
482 key = Qt::Key_Control;
483#endif
484 bProcess = true;
485 break;
486 case Qt::Key_Tab:
487 case Qt::Key_Alt:
488 bProcess = true;
489 break;
490 default:
491 break;
492 }
493
494 if(bProcess) {
495 switch(event->type()) {
496 case QEvent::KeyPress:
497 {
498 CFrmViewer* focus = qobject_cast<CFrmViewer*>(QApplication::focusWidget());
499 if(focus) {
500 emit focus->sigKeyPressEvent(keyEvent);
501 return true;
502 }
503 }
504 case QEvent::KeyRelease:
505 {
506 CFrmViewer* focus = qobject_cast<CFrmViewer*>(QApplication::focusWidget());
507 if(focus) {
508 emit focus->sigKeyReleaseEvent(keyEvent);
509 return true;
510 }
511 }
512 default:
513 break;
514 }
515 }
516 }
517 return QObject::eventFilter(watched, event);
518}
The Handle CConnecter class.
Definition Client.h:151
virtual int onProcess(const QString &id, CPluginClient *pPlug)=0
Process plugins.
manage plugins and connecter
Definition Client.h:49
virtual CConnecter * LoadConnecter(const QString &szFile)
New CConnecter pointer from file, the owner is caller.
Definition Client.cpp:264
virtual CConnecter * CreateConnecter(const QString &id)
New CConnecter pointer, the owner is caller.
Definition Client.cpp:203
virtual int LoadSettings(const QString szFile=QString())
Load Client parameters from file.
Definition Client.cpp:352
virtual int DeleteConnecter(CConnecter *p)
Delete CConnecter.
Definition Client.cpp:231
virtual QList< QWidget * > GetSettingsWidgets(QWidget *parent)
Get parameter settings widget.
Definition Client.cpp:380
virtual int SaveConnecter(CConnecter *pConnecter)
Accept connecter parameters to file.
Definition Client.cpp:304
virtual int SaveSettings(const QString szFile=QString())
Save Client parameters to file.
Definition Client.cpp:366
virtual int EnumPlugins(Handle *handle)
Enum plugins.
Definition Client.cpp:399
Connecter interface.
Definition Connecter.h:62
virtual const QString Id()
Identity.
Definition Connecter.cpp:39
virtual int SetParameter(CParameter *pParameter) override
Set the parameters and initialize the user interface.
A widget which displays output image from a CConnectDesktop and sends input keypresses and mouse acti...
Definition FrmViewer.h:49
The parameters of client.
The plugin interface.
virtual const QString Details() const
Display more information in About dialog or log.
virtual const QString Id() const
ID. Default: Protocol() + ":" + Name()
virtual const QString DisplayName() const
The plugin display name.
virtual const QString Description() const =0
Plugin description.
virtual const QString Version() const =0
Version.
virtual const QString Name() const =0
This name must be the same as the project name (${PROJECT_NAME}). The translation file (${PROJECT_NAM...
virtual const QString Protocol() const =0
Plugin Protocol.