mirror of
https://github.com/qTox/qTox.git
synced 2024-03-22 14:00:36 +08:00
Add GUI for audio groupchats
This commit is contained in:
parent
54462f0276
commit
eac712e68a
|
@ -46,12 +46,15 @@ const QString Core::TOX_EXT = ".tox";
|
||||||
QList<ToxFile> Core::fileSendQueue;
|
QList<ToxFile> Core::fileSendQueue;
|
||||||
QList<ToxFile> Core::fileRecvQueue;
|
QList<ToxFile> Core::fileRecvQueue;
|
||||||
QHash<int, ToxGroupCall> Core::groupCalls;
|
QHash<int, ToxGroupCall> Core::groupCalls;
|
||||||
|
QThread* Core::coreThread{nullptr};
|
||||||
|
|
||||||
Core::Core(Camera* cam, QThread *coreThread, QString loadPath) :
|
Core::Core(Camera* cam, QThread *CoreThread, QString loadPath) :
|
||||||
tox(nullptr), camera(cam), loadPath(loadPath), ready{false}
|
tox(nullptr), camera(cam), loadPath(loadPath), ready{false}
|
||||||
{
|
{
|
||||||
qDebug() << "Core: loading Tox from" << loadPath;
|
qDebug() << "Core: loading Tox from" << loadPath;
|
||||||
|
|
||||||
|
coreThread = CoreThread;
|
||||||
|
|
||||||
videobuf = new uint8_t[videobufsize];
|
videobuf = new uint8_t[videobufsize];
|
||||||
|
|
||||||
for (int i = 0; i < ptCounter; i++)
|
for (int i = 0; i < ptCounter; i++)
|
||||||
|
|
11
src/core.h
11
src/core.h
|
@ -117,6 +117,13 @@ public slots:
|
||||||
void micMuteToggle(int callId);
|
void micMuteToggle(int callId);
|
||||||
void volMuteToggle(int callId);
|
void volMuteToggle(int callId);
|
||||||
|
|
||||||
|
static void joinGroupCall(int groupId); ///< Starts a call in an existing AV groupchat
|
||||||
|
static void leaveGroupCall(int groupId); ///< Will not leave the group, just stop the call
|
||||||
|
static void disableGroupCallMic(int groupId);
|
||||||
|
static void disableGroupCallVol(int groupId);
|
||||||
|
static void enableGroupCallMic(int groupId);
|
||||||
|
static void enableGroupCallVol(int groupId);
|
||||||
|
|
||||||
void setPassword(QString& password, PasswordType passtype, uint8_t* salt = nullptr);
|
void setPassword(QString& password, PasswordType passtype, uint8_t* salt = nullptr);
|
||||||
void clearPassword(PasswordType passtype);
|
void clearPassword(PasswordType passtype);
|
||||||
QByteArray encryptData(const QByteArray& data, PasswordType passtype);
|
QByteArray encryptData(const QByteArray& data, PasswordType passtype);
|
||||||
|
@ -233,8 +240,6 @@ private:
|
||||||
static void onAvPeerTimeout(void* toxav, int32_t call_index, void* core);
|
static void onAvPeerTimeout(void* toxav, int32_t call_index, void* core);
|
||||||
static void onAvMediaChange(void *toxav, int32_t call_index, void* core);
|
static void onAvMediaChange(void *toxav, int32_t call_index, void* core);
|
||||||
|
|
||||||
static void joinGroupCall(int groupId, ToxAv *toxav); ///< Starts a call in an existing AV groupchat
|
|
||||||
static void leaveGroupCall(int groupId, ToxAv *toxav); ///< Will not leave the group, just stop the call
|
|
||||||
static void playGroupAudio(Tox* tox, int groupnumber, int friendgroupnumber, const int16_t* out_audio,
|
static void playGroupAudio(Tox* tox, int groupnumber, int friendgroupnumber, const int16_t* out_audio,
|
||||||
unsigned out_audio_samples, uint8_t decoder_channels, unsigned audio_sample_rate, void* userdata);
|
unsigned out_audio_samples, uint8_t decoder_channels, unsigned audio_sample_rate, void* userdata);
|
||||||
static void sendGroupCallAudio(int groupId, ToxAv* toxav);
|
static void sendGroupCallAudio(int groupId, ToxAv* toxav);
|
||||||
|
@ -282,6 +287,8 @@ private:
|
||||||
|
|
||||||
static ALCdevice* alOutDev, *alInDev;
|
static ALCdevice* alOutDev, *alInDev;
|
||||||
static ALCcontext* alContext;
|
static ALCcontext* alContext;
|
||||||
|
|
||||||
|
static QThread *coreThread;
|
||||||
public:
|
public:
|
||||||
static ALuint alMainSource;
|
static ALuint alMainSource;
|
||||||
};
|
};
|
||||||
|
|
|
@ -592,10 +592,13 @@ void Core::playGroupAudio(Tox* /*tox*/, int groupnumber, int /*friendgroupnumbe
|
||||||
if (!groupCalls[groupnumber].active)
|
if (!groupCalls[groupnumber].active)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
playAudioBuffer(groupCalls[groupnumber].alSource, out_audio, out_audio_samples, decoder_channels, audio_sample_rate);
|
if (!groupCalls[groupbumber].muteVol)
|
||||||
|
return;
|
||||||
|
|
||||||
|
playAudioBuffer(alMainSource, out_audio, out_audio_samples, decoder_channels, audio_sample_rate);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Core::joinGroupCall(int groupId, ToxAv *toxav)
|
void Core::joinGroupCall(int groupId)
|
||||||
{
|
{
|
||||||
qDebug() << QString("Core: Joining group call %1").arg(groupId);
|
qDebug() << QString("Core: Joining group call %1").arg(groupId);
|
||||||
groupCalls[groupId].groupId = groupId;
|
groupCalls[groupId].groupId = groupId;
|
||||||
|
@ -608,10 +611,13 @@ void Core::joinGroupCall(int groupId, ToxAv *toxav)
|
||||||
groupCalls[groupId].codecSettings.max_video_height = TOXAV_MAX_VIDEO_HEIGHT;
|
groupCalls[groupId].codecSettings.max_video_height = TOXAV_MAX_VIDEO_HEIGHT;
|
||||||
|
|
||||||
// Audio
|
// Audio
|
||||||
alGenSources(1, &calls[groupId].alSource);
|
alGenSources(1, &groupCalls[groupId].alSource);
|
||||||
alcCaptureStart(alInDev);
|
alcCaptureStart(alInDev);
|
||||||
|
|
||||||
// Go
|
// Go
|
||||||
|
ToxAv* toxav = Core::getInstance()->toxav;
|
||||||
|
groupCalls[groupId].sendAudioTimer = new QTimer();
|
||||||
|
groupCalls[groupId].sendAudioTimer->moveToThread(coreThread);
|
||||||
groupCalls[groupId].active = true;
|
groupCalls[groupId].active = true;
|
||||||
groupCalls[groupId].sendAudioTimer->setInterval(5);
|
groupCalls[groupId].sendAudioTimer->setInterval(5);
|
||||||
groupCalls[groupId].sendAudioTimer->setSingleShot(true);
|
groupCalls[groupId].sendAudioTimer->setSingleShot(true);
|
||||||
|
@ -619,7 +625,7 @@ void Core::joinGroupCall(int groupId, ToxAv *toxav)
|
||||||
groupCalls[groupId].sendAudioTimer->start();
|
groupCalls[groupId].sendAudioTimer->start();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Core::leaveGroupCall(int groupId, ToxAv *)
|
void Core::leaveGroupCall(int groupId)
|
||||||
{
|
{
|
||||||
qDebug() << QString("Core: Leaving group call %1").arg(groupId);
|
qDebug() << QString("Core: Leaving group call %1").arg(groupId);
|
||||||
groupCalls[groupId].active = false;
|
groupCalls[groupId].active = false;
|
||||||
|
@ -664,3 +670,23 @@ void Core::sendGroupCallAudio(int groupId, ToxAv* toxav)
|
||||||
}
|
}
|
||||||
groupCalls[groupId].sendAudioTimer->start();
|
groupCalls[groupId].sendAudioTimer->start();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Core::disableGroupCallMic(int groupId)
|
||||||
|
{
|
||||||
|
groupCalls[groupId].muteMic = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Core::disableGroupCallVol(int groupId)
|
||||||
|
{
|
||||||
|
groupCalls[groupId].muteVol = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Core::enableGroupCallMic(int groupId)
|
||||||
|
{
|
||||||
|
groupCalls[groupId].muteMic = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Core::enableGroupCallVol(int groupId)
|
||||||
|
{
|
||||||
|
groupCalls[groupId].muteVol = false;
|
||||||
|
}
|
||||||
|
|
|
@ -23,8 +23,8 @@
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QTimer>
|
#include <QTimer>
|
||||||
|
|
||||||
Group::Group(int GroupId, QString Name)
|
Group::Group(int GroupId, QString Name, bool IsAvGroupchat)
|
||||||
: groupId(GroupId), nPeers{0}
|
: groupId(GroupId), nPeers{0}, avGroupchat{IsAvGroupchat}
|
||||||
{
|
{
|
||||||
widget = new GroupWidget(groupId, Name);
|
widget = new GroupWidget(groupId, Name);
|
||||||
chatForm = new GroupChatForm(this);
|
chatForm = new GroupChatForm(this);
|
||||||
|
|
|
@ -30,7 +30,7 @@ class Group : public QObject
|
||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
Group(int GroupId, QString Name);
|
Group(int GroupId, QString Name, bool IsAvGroupchat);
|
||||||
~Group();
|
~Group();
|
||||||
void addPeer(int peerId, QString name);
|
void addPeer(int peerId, QString name);
|
||||||
void removePeer(int peerId);
|
void removePeer(int peerId);
|
||||||
|
@ -43,6 +43,7 @@ public:
|
||||||
GroupWidget* widget;
|
GroupWidget* widget;
|
||||||
GroupChatForm* chatForm;
|
GroupChatForm* chatForm;
|
||||||
int hasNewMessages, userWasMentioned;
|
int hasNewMessages, userWasMentioned;
|
||||||
|
bool avGroupchat;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // GROUP_H
|
#endif // GROUP_H
|
||||||
|
|
|
@ -19,9 +19,9 @@
|
||||||
|
|
||||||
QList<Group*> GroupList::groupList;
|
QList<Group*> GroupList::groupList;
|
||||||
|
|
||||||
Group* GroupList::addGroup(int groupId, const QString& name)
|
Group* GroupList::addGroup(int groupId, const QString& name, bool isAvGroupchat)
|
||||||
{
|
{
|
||||||
Group* newGroup = new Group(groupId, name);
|
Group* newGroup = new Group(groupId, name, isAvGroupchat);
|
||||||
groupList.append(newGroup);
|
groupList.append(newGroup);
|
||||||
return newGroup;
|
return newGroup;
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ class GroupList
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
GroupList();
|
GroupList();
|
||||||
static Group* addGroup(int groupId, const QString& name);
|
static Group* addGroup(int groupId, const QString& name, bool isAvGroupchat);
|
||||||
static Group* findGroup(int groupId);
|
static Group* findGroup(int groupId);
|
||||||
static void removeGroup(int groupId, bool fake = false);
|
static void removeGroup(int groupId, bool fake = false);
|
||||||
|
|
||||||
|
|
|
@ -43,8 +43,6 @@
|
||||||
|
|
||||||
ChatForm::ChatForm(Friend* chatFriend)
|
ChatForm::ChatForm(Friend* chatFriend)
|
||||||
: f(chatFriend)
|
: f(chatFriend)
|
||||||
, audioInputFlag(false)
|
|
||||||
, audioOutputFlag(false)
|
|
||||||
, callId(0)
|
, callId(0)
|
||||||
{
|
{
|
||||||
nameLabel->setText(f->getDisplayedName());
|
nameLabel->setText(f->getDisplayedName());
|
||||||
|
|
|
@ -95,8 +95,6 @@ private:
|
||||||
Friend* f;
|
Friend* f;
|
||||||
CroppingLabel *statusMessageLabel;
|
CroppingLabel *statusMessageLabel;
|
||||||
NetCamView* netcam;
|
NetCamView* netcam;
|
||||||
bool audioInputFlag;
|
|
||||||
bool audioOutputFlag;
|
|
||||||
int callId;
|
int callId;
|
||||||
QLabel *callDuration;
|
QLabel *callDuration;
|
||||||
QTimer *timer;
|
QTimer *timer;
|
||||||
|
|
|
@ -37,6 +37,8 @@
|
||||||
GenericChatForm::GenericChatForm(QWidget *parent) :
|
GenericChatForm::GenericChatForm(QWidget *parent) :
|
||||||
QWidget(parent),
|
QWidget(parent),
|
||||||
earliestMessage(nullptr)
|
earliestMessage(nullptr)
|
||||||
|
, audioInputFlag(false)
|
||||||
|
, audioOutputFlag(false)
|
||||||
{
|
{
|
||||||
curRow = 0;
|
curRow = 0;
|
||||||
|
|
||||||
|
|
|
@ -91,6 +91,8 @@ protected:
|
||||||
QPushButton *sendButton;
|
QPushButton *sendButton;
|
||||||
ChatAreaWidget *chatWidget;
|
ChatAreaWidget *chatWidget;
|
||||||
QDateTime *earliestMessage;
|
QDateTime *earliestMessage;
|
||||||
|
bool audioInputFlag;
|
||||||
|
bool audioOutputFlag;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // GENERICCHATFORM_H
|
#endif // GENERICCHATFORM_H
|
||||||
|
|
|
@ -30,17 +30,25 @@
|
||||||
#include "src/misc/flowlayout.h"
|
#include "src/misc/flowlayout.h"
|
||||||
|
|
||||||
GroupChatForm::GroupChatForm(Group* chatGroup)
|
GroupChatForm::GroupChatForm(Group* chatGroup)
|
||||||
: group(chatGroup)
|
: group(chatGroup), inCall{false}
|
||||||
{
|
{
|
||||||
nusersLabel = new QLabel();
|
nusersLabel = new QLabel();
|
||||||
|
|
||||||
tabber = new TabCompleter(msgEdit, group);
|
tabber = new TabCompleter(msgEdit, group);
|
||||||
|
|
||||||
fileButton->setEnabled(false);
|
fileButton->setEnabled(false);
|
||||||
callButton->setVisible(false);
|
if (group->avGroupchat)
|
||||||
videoButton->setVisible(false);
|
{
|
||||||
volButton->setVisible(false);
|
videoButton->setEnabled(false);
|
||||||
micButton->setVisible(false);
|
videoButton->setObjectName("grey");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
videoButton->setVisible(false);
|
||||||
|
callButton->setVisible(false);
|
||||||
|
volButton->setVisible(false);
|
||||||
|
micButton->setVisible(false);
|
||||||
|
}
|
||||||
|
|
||||||
nameLabel->setText(group->widget->getName());
|
nameLabel->setText(group->widget->getName());
|
||||||
|
|
||||||
|
@ -68,6 +76,9 @@ GroupChatForm::GroupChatForm(Group* chatGroup)
|
||||||
connect(msgEdit, SIGNAL(enterPressed()), this, SLOT(onSendTriggered()));
|
connect(msgEdit, SIGNAL(enterPressed()), this, SLOT(onSendTriggered()));
|
||||||
connect(msgEdit, &ChatTextEdit::tabPressed, tabber, &TabCompleter::complete);
|
connect(msgEdit, &ChatTextEdit::tabPressed, tabber, &TabCompleter::complete);
|
||||||
connect(msgEdit, &ChatTextEdit::keyPressed, tabber, &TabCompleter::reset);
|
connect(msgEdit, &ChatTextEdit::keyPressed, tabber, &TabCompleter::reset);
|
||||||
|
connect(callButton, &QPushButton::clicked, this, &GroupChatForm::onCallClicked);
|
||||||
|
connect(micButton, SIGNAL(clicked()), this, SLOT(onMicMuteToggle()));
|
||||||
|
connect(volButton, SIGNAL(clicked()), this, SLOT(onVolMuteToggle()));
|
||||||
|
|
||||||
setAcceptDrops(true);
|
setAcceptDrops(true);
|
||||||
}
|
}
|
||||||
|
@ -129,3 +140,66 @@ void GroupChatForm::dropEvent(QDropEvent *ev)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GroupChatForm::onMicMuteToggle()
|
||||||
|
{
|
||||||
|
if (audioInputFlag == true)
|
||||||
|
{
|
||||||
|
if (micButton->objectName() == "red")
|
||||||
|
{
|
||||||
|
Core::getInstance()->enableGroupCallMic(group->groupId);
|
||||||
|
micButton->setObjectName("green");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Core::getInstance()->disableGroupCallMic(group->groupId);
|
||||||
|
micButton->setObjectName("red");
|
||||||
|
}
|
||||||
|
|
||||||
|
Style::repolish(micButton);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GroupChatForm::onVolMuteToggle()
|
||||||
|
{
|
||||||
|
if (audioOutputFlag == true)
|
||||||
|
{
|
||||||
|
if (volButton->objectName() == "red")
|
||||||
|
{
|
||||||
|
Core::getInstance()->enableGroupCallVol(group->groupId);
|
||||||
|
volButton->setObjectName("green");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Core::getInstance()->disableGroupCallVol(group->groupId);
|
||||||
|
volButton->setObjectName("red");
|
||||||
|
}
|
||||||
|
|
||||||
|
Style::repolish(volButton);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void GroupChatForm::onCallClicked()
|
||||||
|
{
|
||||||
|
if (!inCall)
|
||||||
|
{
|
||||||
|
Core::getInstance()->joinGroupCall(group->groupId);
|
||||||
|
audioInputFlag = true;
|
||||||
|
audioOutputFlag = true;
|
||||||
|
callButton->setObjectName("red");
|
||||||
|
callButton->style()->polish(callButton);
|
||||||
|
inCall = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Core::getInstance()->leaveGroupCall(group->groupId);
|
||||||
|
audioInputFlag = false;
|
||||||
|
audioOutputFlag = false;
|
||||||
|
micButton->setObjectName("green");
|
||||||
|
micButton->style()->polish(micButton);
|
||||||
|
volButton->setObjectName("green");
|
||||||
|
volButton->style()->polish(volButton);
|
||||||
|
callButton->setObjectName("green");
|
||||||
|
callButton->style()->polish(callButton);
|
||||||
|
inCall = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -34,6 +34,9 @@ public:
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void onSendTriggered();
|
void onSendTriggered();
|
||||||
|
void onMicMuteToggle();
|
||||||
|
void onVolMuteToggle();
|
||||||
|
void onCallClicked();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
// drag & drop
|
// drag & drop
|
||||||
|
@ -45,6 +48,7 @@ private:
|
||||||
FlowLayout* namesListLayout;
|
FlowLayout* namesListLayout;
|
||||||
QLabel *nusersLabel;
|
QLabel *nusersLabel;
|
||||||
TabCompleter* tabber;
|
TabCompleter* tabber;
|
||||||
|
bool inCall;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // GROUPCHATFORM_H
|
#endif // GROUPCHATFORM_H
|
||||||
|
|
|
@ -985,7 +985,7 @@ Group *Widget::createGroup(int groupId)
|
||||||
}
|
}
|
||||||
|
|
||||||
QString groupName = QString("Groupchat #%1").arg(groupId);
|
QString groupName = QString("Groupchat #%1").arg(groupId);
|
||||||
Group* newgroup = GroupList::addGroup(groupId, groupName);
|
Group* newgroup = GroupList::addGroup(groupId, groupName, true);
|
||||||
QLayout* layout = contactListWidget->getGroupLayout();
|
QLayout* layout = contactListWidget->getGroupLayout();
|
||||||
layout->addWidget(newgroup->widget);
|
layout->addWidget(newgroup->widget);
|
||||||
newgroup->widget->updateStatusLight();
|
newgroup->widget->updateStatusLight();
|
||||||
|
|
Loading…
Reference in New Issue
Block a user