MainPanel.cpp 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005
  1. #include "MainPanel.h"
  2. #include "utils/iconutils.h"
  3. // Qt Core
  4. #include <QApplication>
  5. #include <QDebug>
  6. #include <QJsonDocument>
  7. #include <QJsonObject>
  8. #include <QJsonParseError>
  9. #include <QResizeEvent>
  10. #include <QScreen>
  11. #include <QSizePolicy>
  12. #include <QTimer>
  13. #include <QtConcurrent>
  14. // Qt Widgets
  15. #include <QComboBox>
  16. #include <QFrame>
  17. #include <QHBoxLayout>
  18. #include <QLabel>
  19. #include <QSplitter>
  20. #include <QVBoxLayout>
  21. // Third Party
  22. #include <qtpromise/qpromise.h>
  23. #include <qtpromise/qpromisefuture.h>
  24. #include <qtpromise/qpromisehelpers.h>
  25. // Project Includes
  26. #include "AVPlayer/avplayerwidget.h"
  27. #include "api/roomapi.h"
  28. #include "appevent.h"
  29. #include "network/websocketclient.h"
  30. #include "themesettingswidget.h"
  31. #include "widgets/bubbletip.h"
  32. #include "widgets/chatView/chatwindow.h"
  33. #include "widgets/framelessbase.h"
  34. #include "widgets/functionbutton.h"
  35. #include "widgets/maskoverlay.h"
  36. #include "widgets/recorderwidget.h"
  37. #include "widgets/statswidget.h"
  38. #include "widgets/userprofilewidget.h"
  39. #include <util/jsonmapper.h>
  40. MainPanel::MainPanel(QWidget *parent)
  41. : TWidget(parent)
  42. , m_chatTabWidget(nullptr)
  43. , m_chatWindow(nullptr)
  44. , m_onlineUsersWidget(nullptr)
  45. {
  46. setAttribute(Qt::WA_StyledBackground, true);
  47. // ========== 初始化播放控制组件 ==========
  48. initPlaybackControls();
  49. // ========== 初始化聊天相关组件 ==========
  50. initChatComponents();
  51. // ========== 初始化布局组件 ==========
  52. initLayoutComponents();
  53. // ========== 初始化功能按钮 ==========
  54. initFunctionButtons();
  55. // ========== 连接信号槽 ==========
  56. connectSignals();
  57. // ========== 初始化音频设备 ==========
  58. initAudioDeviceSelectors();
  59. }
  60. void MainPanel::initPlaybackControls()
  61. {
  62. m_debounceTimer = new QTimer(this);
  63. m_debounceTimer->setInterval(500);
  64. m_debounceTimer->setSingleShot(true);
  65. connect(m_debounceTimer, &QTimer::timeout, this, &MainPanel::handleDebouncedPlay);
  66. }
  67. void MainPanel::initChatComponents()
  68. {
  69. webSocketClient = new WebSocketClient(this);
  70. // 启用WebSocket自动重连,提升聊天室稳定性
  71. if (webSocketClient) {
  72. webSocketClient->setAutoReconnect(true);
  73. }
  74. // 创建聊天窗口
  75. m_chatWindow = new ChatWindow(webSocketClient);
  76. m_chatWindow->setMinimumWidth(400);
  77. // 防御:明确不随关闭销毁,避免父窗口关闭时误删
  78. m_chatWindow->setAttribute(Qt::WA_DeleteOnClose, false);
  79. // 连接聊天窗口关闭请求信号
  80. connect(m_chatWindow, &ChatWindow::windowCloseRequested, this, &MainPanel::onChatWindowCloseRequested);
  81. // 创建在线用户管理widget
  82. m_onlineUsersWidget = new OnlineUsersWidget(this);
  83. // 连接在线用户widget的信号
  84. connect(m_onlineUsersWidget, &OnlineUsersWidget::userDoubleClicked,
  85. [this](const QString &userId, const QString &username) {
  86. qDebug() << "用户双击:" << username << "ID:" << userId;
  87. // TODO: 实现私聊功能
  88. });
  89. connect(m_onlineUsersWidget, &OnlineUsersWidget::privateMessageRequested,
  90. [this](const QString &userId, const QString &username) {
  91. qDebug() << "请求私信:" << username << "ID:" << userId;
  92. // TODO: 实现私信功能
  93. });
  94. // 创建Tab容器
  95. m_chatTabWidget = new QTabWidget(this);
  96. m_chatTabWidget->addTab(m_chatWindow, "聊天");
  97. m_chatTabWidget->addTab(m_onlineUsersWidget, "在线用户");
  98. m_chatTabWidget->setMinimumWidth(400);
  99. }
  100. void MainPanel::initLayoutComponents()
  101. {
  102. // 创建右侧面板
  103. m_rightWidget = new QWidget;
  104. QVBoxLayout *vbox = new QVBoxLayout(m_rightWidget);
  105. vbox->setContentsMargins(0, 0, 0, 0);
  106. // 创建聊天窗口容器
  107. m_chatContainer = new QWidget(m_rightWidget);
  108. QVBoxLayout *chatLayout = new QVBoxLayout(m_chatContainer);
  109. chatLayout->setContentsMargins(0, 0, 0, 0);
  110. chatLayout->addWidget(m_chatTabWidget);
  111. vbox->addWidget(m_chatContainer, 1);
  112. // 创建主分割器
  113. splitter = new QSplitter(Qt::Horizontal, this);
  114. playerContainer = new QWidget(this);
  115. splitter->addWidget(playerContainer);
  116. splitter->addWidget(m_rightWidget);
  117. splitter->setStretchFactor(0, 60);
  118. splitter->setStretchFactor(1, 30);
  119. this->mainLayout()->addWidget(splitter, 1);
  120. this->mainLayout()->setContentsMargins(0, 0, 0, 0);
  121. this->mainLayout()->setSpacing(0);
  122. // 为playerContainer设置初始布局
  123. QVBoxLayout *playerLayout = new QVBoxLayout(playerContainer);
  124. playerLayout->setContentsMargins(0, 0, 0, 0);
  125. playerLayout->setSpacing(0);
  126. }
  127. void MainPanel::initFunctionButtons()
  128. {
  129. buttonGroup = new PopoverButtonGroup(Qt::Horizontal, playerContainer);
  130. // 添加功能按钮
  131. // FunctionButton *settingsBtn = new FunctionButton(IconUtils::createSettingsIcon(), "设置", this);
  132. // // 移除 Popover 箭头,改为直接打开设置窗口
  133. // buttonGroup->addButton(settingsBtn, nullptr);
  134. // connect(settingsBtn, &QPushButton::clicked, this, &MainPanel::onSettingsButtonClicked);
  135. // 移除:搜索按钮(暂不使用)
  136. // FunctionButton *searchBtn = new FunctionButton(IconUtils::createSearchIcon(), "搜索", this);
  137. // Popover *searchPopover = new Popover(this);
  138. // buttonGroup->addButton(searchBtn, searchPopover);
  139. // FunctionButton *userBtn = new FunctionButton(IconUtils::createUserIcon(), "用户", this);
  140. // Popover *userPopover = new Popover(this);
  141. // // 构建用户Popover内容:提供“退出”操作
  142. // {
  143. // QWidget *userContent = new QWidget(userPopover);
  144. // QVBoxLayout *userLayout = new QVBoxLayout(userContent);
  145. // userLayout->setContentsMargins(8, 8, 8, 8);
  146. // userLayout->setSpacing(8);
  147. // QPushButton *logoutBtn = new QPushButton(tr("退出"), userContent);
  148. // logoutBtn->setMinimumWidth(120);
  149. // connect(logoutBtn, &QPushButton::clicked, this, [this]() {
  150. // emit logoutClicked();
  151. // });
  152. // userLayout->addWidget(logoutBtn);
  153. // userPopover->setContentWidget(userContent);
  154. // }
  155. // buttonGroup->addButton(userBtn, userPopover);
  156. // 添加音频设备选择按钮
  157. // FunctionButton *audioDeviceBtn = new FunctionButton(IconUtils::createAudioDeviceIcon(), "音频设备", this);
  158. // Popover *audioDevicePopover = new Popover(this);
  159. // // // 使用解耦合版本的音频设备选择器
  160. // // m_audioDeviceSelectorDecoupled = new AudioDeviceSelectorIconDecoupled(this);
  161. // // audioDevicePopover->setContentWidget(m_audioDeviceSelectorDecoupled);
  162. // // 使用 RecorderAudioWidget 作为 Popover 内容(麦克风 + 扬声器)
  163. // QWidget *audioContent = new QWidget(audioDevicePopover);
  164. // QVBoxLayout *audioLayout = new QVBoxLayout(audioContent);
  165. // audioLayout->setContentsMargins(8, 8, 8, 8);
  166. // audioLayout->setSpacing(8);
  167. // // 麦克风区域
  168. // QLabel *micTitle = new QLabel(tr("麦克风"), audioContent);
  169. // micTitle->setStyleSheet("font-weight:600;");
  170. // m_micWidget = new QComboBox(audioContent);
  171. // m_micWidget->setEditable(false);
  172. // audioLayout->addWidget(micTitle);
  173. // audioLayout->addWidget(m_micWidget);
  174. // // 扬声器区域
  175. // QLabel *speakerTitle = new QLabel(tr("扬声器"), audioContent);
  176. // speakerTitle->setStyleSheet("font-weight:600;");
  177. // m_speakerWidget = new QComboBox(audioContent);
  178. // m_speakerWidget->setEditable(false);
  179. // audioLayout->addWidget(speakerTitle);
  180. // audioLayout->addWidget(m_speakerWidget);
  181. // // 视频编码器区域
  182. // QLabel *encoderTitle = new QLabel(tr("视频编码器"), audioContent);
  183. // encoderTitle->setStyleSheet("font-weight:600;");
  184. // m_encoderWidget = new QComboBox(audioContent);
  185. // m_encoderWidget->setEditable(false);
  186. // audioLayout->addWidget(encoderTitle);
  187. // audioLayout->addWidget(m_encoderWidget);
  188. // audioDevicePopover->setContentWidget(audioContent);
  189. // buttonGroup->addButton(audioDeviceBtn, audioDevicePopover);
  190. // 新增:独立的推流按钮(无弹层)
  191. m_streamButton = new FunctionButton(IconUtils::createStreamIcon(), tr("推流"), this);
  192. buttonGroup->addButton(m_streamButton);
  193. // 新增:聊天按钮(推流:显示/隐藏;非推流:弹出/嵌入)
  194. m_chatButton = new FunctionButton(IconUtils::createChatIcon(), tr("聊天"), this);
  195. buttonGroup->addButton(m_chatButton);
  196. // 移除:执行操作按钮(暂不使用)
  197. // FunctionButton *actionButton = new FunctionButton(IconUtils::createSettingsIcon(),
  198. // "执行操作",
  199. // this);
  200. // buttonGroup->addButton(actionButton, nullptr);
  201. // 将buttonGroup添加到playerContainer的布局中
  202. QVBoxLayout *playerLayout = qobject_cast<QVBoxLayout*>(playerContainer->layout());
  203. if (!playerLayout) {
  204. playerLayout = new QVBoxLayout(playerContainer);
  205. playerLayout->setContentsMargins(0, 0, 0, 0);
  206. playerLayout->setSpacing(0);
  207. }
  208. // 移除addStretch,让buttonGroup紧贴播放器组件,消除空白区域
  209. playerLayout->addWidget(buttonGroup, 0); // 添加buttonGroup,不拉伸
  210. buttonGroup->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed); // 使用固定大小策略
  211. }
  212. void MainPanel::connectSignals()
  213. {
  214. // 连接功能按钮信号
  215. connect(m_streamButton, &QPushButton::clicked, this, &MainPanel::onStreamButtonClicked);
  216. connect(m_chatButton, &QPushButton::clicked, this, &MainPanel::onChatButtonClicked);
  217. // 连接WebSocket相关信号
  218. // 已移除与 UserProfileWidget 相关的在线状态更新
  219. // connect(webSocketClient, &WebSocketClient::statsUpdate, statsWidget, &StatsWidget::updateStats); // 暂时移除统计在主面板的更新
  220. // 连接用户列表更新信号到在线用户widget
  221. connect(webSocketClient, &WebSocketClient::userListUpdate,
  222. m_onlineUsersWidget, &OnlineUsersWidget::onUserListUpdate);
  223. connect(webSocketClient, &WebSocketClient::liveStatus, this, [this](const QString &msg) {
  224. // 这里可以处理 liveStatus 相关逻辑
  225. QJsonParseError err;
  226. QJsonDocument doc = QJsonDocument::fromJson(msg.toUtf8(), &err);
  227. if (err.error != QJsonParseError::NoError || !doc.isObject()) {
  228. qDebug() << "[MainPanel] liveStatus: 解析失败" << err.errorString();
  229. return;
  230. }
  231. QJsonObject obj = doc.object();
  232. int liveStatus = obj.value("liveStatus").toInt(0); // 默认-1
  233. if (liveStatus == 1) {
  234. qDebug() << "[MainPanel] liveStatus: 直播中" << m_chatWindow;
  235. if (m_chatWindow) {
  236. const QString id = webSocketClient->roomId();
  237. // 使用防抖机制处理频繁的请求
  238. m_pendingRoomId = id;
  239. m_debounceTimer->start(); // 重新开始计时,如果在500ms内再次收到请求,会重置定时器
  240. }
  241. }
  242. });
  243. // 初始化音频设备列表
  244. initAudioDeviceSelectors();
  245. }
  246. MainPanel::~MainPanel()
  247. {
  248. if (m_avPlayerStandalone) {
  249. m_avPlayerStandalone->deleteLater();
  250. m_avPlayerStandalone = nullptr;
  251. }
  252. if (m_recorderStandalone) {
  253. m_recorderStandalone->deleteLater();
  254. m_recorderStandalone = nullptr;
  255. }
  256. }
  257. void MainPanel::setRole(const QStringList &roleList)
  258. {
  259. bool isRec = roleList.contains("role.admin") || roleList.contains("role.recorder") || roleList.contains("录制");
  260. if (isRec) {
  261. auto rec = new RecorderWidget(this);
  262. setPlayerWidget(rec);
  263. } else {
  264. auto av = new AVPlayerWidget(this);
  265. setPlayerWidget(av);
  266. }
  267. if (m_streamButton) {
  268. m_streamButton->setVisible(isRec);
  269. m_streamButton->setText(tr("推流"));
  270. }
  271. }
  272. void MainPanel::setPushRoomId(const QString &id)
  273. {
  274. if (!webSocketClient) return;
  275. // 初始化聊天室连接
  276. webSocketClient->connectToRoom(id);
  277. if (m_chatWindow) {
  278. m_chatWindow->initWebsocket(id);
  279. }
  280. // 设置在线用户widget的房间ID
  281. if (m_onlineUsersWidget) {
  282. m_onlineUsersWidget->setCurrentRoomId(id);
  283. }
  284. // 若当前是播放器,使用防抖播放
  285. m_pendingRoomId = id;
  286. m_debounceTimer->start();
  287. }
  288. void MainPanel::setPlayerWidget(QWidget *newPlayer)
  289. {
  290. if (!newPlayer) return;
  291. if (!playerContainer) return;
  292. if (playerWidget) {
  293. playerWidget->deleteLater();
  294. playerWidget = nullptr;
  295. }
  296. playerWidget = newPlayer;
  297. // 将新播放器添加到容器布局
  298. if (auto layout = qobject_cast<QVBoxLayout*>(playerContainer->layout())) {
  299. layout->insertWidget(0, playerWidget, 1); // 使用拉伸因子1,让播放器组件充分利用空间
  300. } else {
  301. auto layout2 = new QVBoxLayout(playerContainer);
  302. layout2->setContentsMargins(0, 0, 0, 0);
  303. layout2->setSpacing(0);
  304. layout2->addWidget(playerWidget, 1); // 使用拉伸因子1,让播放器组件充分利用空间
  305. }
  306. // 更新推流按钮可见性
  307. if (m_streamButton) {
  308. const bool isRec = qobject_cast<RecorderWidget*>(playerWidget) != nullptr;
  309. m_streamButton->setVisible(isRec);
  310. if (!isRec) {
  311. m_isStreaming = false;
  312. m_streamButton->setText(tr("推流"));
  313. }
  314. }
  315. // 连接 RecorderWidget 的推流信号,保持 UI 状态同步
  316. if (auto rec = qobject_cast<RecorderWidget*>(playerWidget)) {
  317. connect(rec, &RecorderWidget::streamingStarted, this, [this, rec]() {
  318. m_isStreaming = true;
  319. if (m_streamButton) m_streamButton->setText(tr("停止推流"));
  320. // 进入极简模式
  321. if (QWidget *tlw = window()) {
  322. tlw->setUpdatesEnabled(false);
  323. }
  324. rec->hidePreview();
  325. if (m_chatTabWidget) m_chatTabWidget->hide();
  326. if (m_rightWidget) m_rightWidget->hide();
  327. if (splitter) splitter->hide();
  328. // 使用自定义标题的紧凑浮窗承载工具栏
  329. if (!m_compactFrame) {
  330. m_compactFrame = new TMainWindow();
  331. m_compactFrame->setWindowTitle(tr("共享控制"));
  332. // m_compactFrame->setWindowFlag(Qt::Tool, true);
  333. // m_compactFrame->setAttribute(Qt::WA_DeleteOnClose);
  334. if (buttonGroup) {
  335. if (auto layout = qobject_cast<QVBoxLayout*>(playerContainer->layout())) {
  336. layout->removeWidget(buttonGroup);
  337. }
  338. auto container = new QWidget(m_compactFrame);
  339. auto lay = new QHBoxLayout(container);
  340. lay->setContentsMargins(12, 8, 12, 8);
  341. lay->setSpacing(10);
  342. buttonGroup->setParent(container);
  343. lay->addWidget(buttonGroup);
  344. m_compactFrame->setCentralWidget(container);
  345. // 使用 sizeHint 直接设置初始尺寸,避免反复 adjustSize 带来的重算
  346. const QSize hint = buttonGroup->sizeHint();
  347. int w = qMax(360, hint.width() + 24);
  348. int h = qMax(80, hint.height() + 16);
  349. int titleH = 0;
  350. if (auto mw = m_compactFrame->menuWidget()) {
  351. mw->adjustSize();
  352. titleH = mw->sizeHint().height();
  353. }
  354. m_compactFrame->setMinimumSize(w, h + titleH);
  355. m_compactFrame->resize(w, h + titleH);
  356. }
  357. }
  358. m_compactFrame->show();
  359. m_compactFrame->raise();
  360. m_compactFrame->activateWindow();
  361. // 将窗口移动到屏幕中上位置
  362. QRect screenGeometry = QApplication::primaryScreen()->availableGeometry();
  363. int x = (screenGeometry.width() - m_compactFrame->width()) / 2;
  364. int y = screenGeometry.height() / 4; // 屏幕高度的1/4位置,即中上
  365. m_compactFrame->move(screenGeometry.x() + x, screenGeometry.y() + y);
  366. m_compactMode = true;
  367. // 主窗口保持原几何但隐藏
  368. if (QWidget *tlw = window()) {
  369. m_savedWindowGeometry = tlw->geometry();
  370. tlw->hide();
  371. }
  372. });
  373. connect(rec, &RecorderWidget::streamingStopped, this, [this, rec]() {
  374. m_isStreaming = false;
  375. if (m_streamButton) m_streamButton->setText(tr("推流"));
  376. // 退出极简模式
  377. if (QWidget *tlw = window()) {
  378. tlw->setUpdatesEnabled(false);
  379. }
  380. rec->showPreview();
  381. if (splitter) splitter->show();
  382. if (m_rightWidget) m_rightWidget->show();
  383. // 销毁紧凑浮窗并将 buttonGroup 归还
  384. if (m_compactFrame) {
  385. if (buttonGroup) {
  386. buttonGroup->hide();
  387. buttonGroup->setParent(playerContainer);
  388. if (auto layout = qobject_cast<QVBoxLayout*>(playerContainer->layout())) {
  389. layout->addWidget(buttonGroup, 0);
  390. }
  391. buttonGroup->show();
  392. }
  393. m_compactFrame->close();
  394. m_compactFrame->deleteLater();
  395. m_compactFrame = nullptr;
  396. }
  397. m_compactMode = false;
  398. // 恢复主窗口显示
  399. if (QWidget *tlw = window()) {
  400. tlw->setGeometry(m_savedWindowGeometry);
  401. tlw->show();
  402. tlw->raise();
  403. tlw->setUpdatesEnabled(true);
  404. }
  405. showChatEmbedded();
  406. if (m_chatButton) m_chatButton->setText(tr("聊天"));
  407. applyModeLayout(); // 恢复布局为非推流状态
  408. });
  409. }
  410. // 根据当前模式(录制/播放)应用一次自适应布局
  411. applyModeLayout();
  412. }
  413. void MainPanel::handleDebouncedPlay()
  414. {
  415. // 在定时器触发时执行播放逻辑
  416. const QString id = m_pendingRoomId;
  417. if (id.isEmpty()) return;
  418. qDebug() << "[MainPanel] Debounced startPlay for room" << id;
  419. RecorderWidget *rec = qobject_cast<RecorderWidget *>(playerWidget);
  420. if (rec) {
  421. rec->setLiveName(id);
  422. }
  423. AVPlayerWidget *av = qobject_cast<AVPlayerWidget*>(playerWidget);
  424. if (av) {
  425. // 连接异步播放的错误信号(如果尚未连接)
  426. static bool signalsConnected = false;
  427. if (!signalsConnected) {
  428. connect(av, &AVPlayerWidget::playError, this, [this](const QString &error) {
  429. qDebug() << "[MainPanel] Play error:" << error;
  430. // 显示错误提示
  431. BubbleTip::showTip(this, error, BubbleTip::Bottom, 5000, BubbleTip::Error);
  432. });
  433. connect(av, &AVPlayerWidget::playLoadingStarted, this, [this]() {
  434. qDebug() << "[MainPanel] Play loading started";
  435. // 可以在这里添加全局加载状态指示
  436. });
  437. connect(av, &AVPlayerWidget::playLoadingFinished, this, [this]() {
  438. qDebug() << "[MainPanel] Play loading finished";
  439. // 可以在这里隐藏全局加载状态指示
  440. });
  441. signalsConnected = true;
  442. }
  443. av->stopPlay();
  444. av->setPlayRoomId(id);
  445. av->startPlayAsync(); // 使用异步播放方法
  446. return;
  447. }
  448. }
  449. void MainPanel::initAudioDeviceSelectors()
  450. {
  451. // TODO: 枚举音频设备、填充 m_micWidget 和 m_speakerWidget;枚举编码器填充 m_encoderWidget
  452. }
  453. void MainPanel::showRecorderStandalone()
  454. {
  455. if (m_recorderFrame) {
  456. m_recorderFrame->raise();
  457. m_recorderFrame->activateWindow();
  458. return;
  459. }
  460. m_recorderFrame = new TMainWindow();
  461. m_recorderFrame->setAttribute(Qt::WA_DeleteOnClose);
  462. m_recorderFrame->setWindowTitle(tr("录制器"));
  463. m_recorderStandalone = new RecorderWidget(m_recorderFrame);
  464. m_recorderFrame->setCentralWidget(m_recorderStandalone);
  465. m_recorderFrame->resize(960, 600);
  466. m_recorderFrame->show();
  467. connect(m_recorderFrame, &QObject::destroyed, this, [this]() {
  468. m_recorderFrame = nullptr;
  469. m_recorderStandalone = nullptr;
  470. });
  471. }
  472. void MainPanel::showPlayerStandalone()
  473. {
  474. if (m_playerFrame) {
  475. m_playerFrame->raise();
  476. m_playerFrame->activateWindow();
  477. return;
  478. }
  479. m_playerFrame = new TMainWindow();
  480. m_playerFrame->setAttribute(Qt::WA_DeleteOnClose);
  481. m_playerFrame->setWindowTitle(tr("播放器"));
  482. m_avPlayerStandalone = new AVPlayerWidget(m_playerFrame);
  483. m_playerFrame->setCentralWidget(m_avPlayerStandalone);
  484. m_playerFrame->resize(960, 600);
  485. m_playerFrame->show();
  486. connect(m_playerFrame, &QObject::destroyed, this, [this]() {
  487. m_playerFrame = nullptr;
  488. m_avPlayerStandalone = nullptr;
  489. });
  490. }
  491. void MainPanel::showChatStandalone()
  492. {
  493. // 显示独立聊天窗口 - 调用 showChatInFrame 实现
  494. showChatInFrame();
  495. }
  496. void MainPanel::showChatInFrame()
  497. {
  498. if (!m_chatTabWidget) return;
  499. if (m_isStreaming) return; // 推流时保持仅浮窗
  500. // - if (m_chatFrame) {
  501. // - m_chatFrame->raise();
  502. // - m_chatFrame->activateWindow();
  503. // - return;
  504. // - }
  505. if (m_chatFrame) {
  506. // 确保 m_chatTabWidget 已正确作为中央部件挂载
  507. if (m_chatFrame->centralWidget() != m_chatTabWidget && m_chatTabWidget) {
  508. m_chatTabWidget->hide();
  509. if (m_chatTabWidget->parent() != m_chatFrame) {
  510. m_chatTabWidget->setParent(m_chatFrame);
  511. }
  512. m_chatFrame->setCentralWidget(m_chatTabWidget);
  513. m_chatTabWidget->show();
  514. }
  515. if (!m_chatFrame->isVisible())
  516. m_chatFrame->show();
  517. m_chatFrame->raise();
  518. m_chatFrame->activateWindow();
  519. return;
  520. }
  521. // 从嵌入容器移除
  522. if (m_chatContainer) {
  523. if (auto l = qobject_cast<QVBoxLayout*>(m_chatContainer->layout())) {
  524. l->removeWidget(m_chatTabWidget);
  525. }
  526. }
  527. m_chatFrame = new TMainWindow();
  528. m_chatFrame->setAttribute(Qt::WA_DeleteOnClose);
  529. m_chatFrame->setWindowTitle(tr("聊天"));
  530. m_chatFrame->installEventFilter(this);
  531. m_chatTabWidget->setParent(m_chatFrame);
  532. m_chatFrame->setCentralWidget(m_chatTabWidget);
  533. m_chatFrame->resize(380, 540);
  534. m_chatFrame->show();
  535. // - connect(m_chatFrame, &QObject::destroyed, this, [this]() {
  536. // - m_chatFrame = nullptr;
  537. // - onChatWindowCloseRequested();
  538. // - });
  539. connect(m_chatFrame, &QObject::destroyed, this, [this]() { m_chatFrame = nullptr; });
  540. }
  541. void MainPanel::showChatEmbedded()
  542. {
  543. if (!m_chatTabWidget || !m_chatContainer) return;
  544. // 如存在独立窗口包装,先将 m_chatTabWidget 移回容器再关闭窗口,避免被父窗口销毁
  545. if (m_chatFrame) {
  546. // 断开临时回调,避免 destroyed 中的副作用
  547. disconnect(m_chatFrame, nullptr, this, nullptr);
  548. if (m_chatTabWidget->parent() == m_chatFrame) {
  549. // 使用 takeCentralWidget 而非 setCentralWidget(nullptr) 避免误删中央部件
  550. if (m_chatFrame->centralWidget()) {
  551. QWidget *w = m_chatFrame->takeCentralWidget();
  552. Q_UNUSED(w);
  553. }
  554. }
  555. m_chatTabWidget->hide();
  556. m_chatTabWidget->setParent(m_chatContainer);
  557. m_chatTabWidget->setWindowTitle(QString());
  558. if (auto l = qobject_cast<QVBoxLayout *>(m_chatContainer->layout())) {
  559. if (l->indexOf(m_chatTabWidget) < 0)
  560. l->addWidget(m_chatTabWidget);
  561. } else {
  562. auto l2 = new QVBoxLayout(m_chatContainer);
  563. l2->setContentsMargins(0, 0, 0, 0);
  564. l2->addWidget(m_chatTabWidget);
  565. }
  566. m_chatTabWidget->show();
  567. // 现在安全地关闭独立窗口
  568. m_chatFrame->close();
  569. m_chatFrame = nullptr;
  570. } else {
  571. // 无独立窗口,仅确保嵌入
  572. m_chatTabWidget->hide();
  573. m_chatTabWidget->setParent(m_chatContainer);
  574. m_chatTabWidget->setWindowTitle(QString());
  575. if (auto l = qobject_cast<QVBoxLayout *>(m_chatContainer->layout())) {
  576. if (l->indexOf(m_chatTabWidget) < 0)
  577. l->addWidget(m_chatTabWidget);
  578. } else {
  579. auto l2 = new QVBoxLayout(m_chatContainer);
  580. l2->setContentsMargins(0, 0, 0, 0);
  581. l2->addWidget(m_chatTabWidget);
  582. }
  583. m_chatTabWidget->show();
  584. }
  585. }
  586. void MainPanel::onRecordButtonClicked()
  587. {
  588. // TODO: 录制控制逻辑
  589. }
  590. void MainPanel::onStreamButtonClicked()
  591. {
  592. RecorderWidget *rec = qobject_cast<RecorderWidget*>(playerWidget);
  593. if (!rec) {
  594. qDebug() << "[MainPanel] 当前不处于录制器模式,无法推流";
  595. return;
  596. }
  597. if (!m_isStreaming) {
  598. rec->startStreaming(); // 成功与否由信号驱动 UI
  599. } else {
  600. rec->stopStreaming();
  601. }
  602. }
  603. void MainPanel::onChatWindowCloseRequested()
  604. {
  605. // 如果处在推流状态,认为是“隐藏聊天”而不是关闭程序
  606. if (m_isStreaming) {
  607. // 显式隐藏独立聊天窗口,统一行为到 eventFilter 的隐藏逻辑
  608. if (m_chatFrame && m_chatFrame->isVisible()) {
  609. m_chatFrame->hide();
  610. }
  611. if (m_chatButton) m_chatButton->setText(tr("显示聊天"));
  612. return; // ChatWindow 本身不要销毁
  613. }
  614. // 非推流:回归嵌入显示
  615. showChatEmbedded();
  616. if (m_chatButton) m_chatButton->setText(tr("聊天"));
  617. applyModeLayout();
  618. }
  619. void MainPanel::onChatButtonClicked()
  620. {
  621. if (!m_chatTabWidget) return;
  622. if (m_isStreaming) {
  623. // 推流时:只在独立聊天窗口上 显示/隐藏 切换,避免频繁 reparent
  624. if (!m_chatFrame) {
  625. // 如果当前在嵌入容器里,先从布局移除
  626. if (m_chatContainer) {
  627. if (auto l = qobject_cast<QVBoxLayout*>(m_chatContainer->layout())) {
  628. l->removeWidget(m_chatTabWidget);
  629. }
  630. }
  631. m_chatFrame = new TMainWindow();
  632. m_chatFrame->setAttribute(Qt::WA_DeleteOnClose);
  633. m_chatFrame->setWindowTitle(tr("聊天"));
  634. m_chatFrame->installEventFilter(this);
  635. m_chatTabWidget->setParent(m_chatFrame);
  636. m_chatFrame->setCentralWidget(m_chatTabWidget);
  637. m_chatTabWidget->show();
  638. m_chatFrame->resize(380, 540);
  639. m_chatFrame->show();
  640. m_chatFrame->raise();
  641. m_chatFrame->activateWindow();
  642. connect(m_chatFrame, &QObject::destroyed, this, [this]() { m_chatFrame = nullptr; });
  643. if (m_chatButton) m_chatButton->setText(tr("隐藏聊天"));
  644. return; // 结束推流分支处理
  645. } else {
  646. // 已有独立窗口:切换显示/隐藏
  647. if (m_chatTabWidget->parent() != m_chatFrame) {
  648. m_chatTabWidget->setParent(m_chatFrame);
  649. if (m_chatFrame->centralWidget() != m_chatTabWidget)
  650. m_chatFrame->setCentralWidget(m_chatTabWidget);
  651. }
  652. m_chatTabWidget->show();
  653. if (m_chatFrame->isVisible()) {
  654. m_chatFrame->hide();
  655. if (m_chatButton) m_chatButton->setText(tr("显示聊天"));
  656. } else {
  657. m_chatFrame->show();
  658. m_chatFrame->raise();
  659. m_chatFrame->activateWindow();
  660. if (m_chatButton) m_chatButton->setText(tr("隐藏聊天"));
  661. }
  662. return; // 结束推流分支处理
  663. }
  664. }
  665. // 非推流:在弹出与嵌入间切换
  666. if (m_chatFrame) {
  667. // 已经是弹出状态,切回嵌入
  668. showChatEmbedded();
  669. if (m_chatButton)
  670. m_chatButton->setText(tr("聊天"));
  671. applyModeLayout();
  672. } else {
  673. // 目前为嵌入状态,弹出为独立窗口
  674. showChatInFrame();
  675. if (m_chatButton)
  676. m_chatButton->setText(tr("嵌入聊天"));
  677. applyModeLayout();
  678. }
  679. }
  680. // ===== 浮动工具栏 =====
  681. void MainPanel::showFloatingToolbar()
  682. {
  683. // 改为使用 m_compactFrame 承载,无需再在主窗口中悬浮
  684. if (!m_compactFrame)
  685. return;
  686. m_compactFrame->show();
  687. m_compactFrame->raise();
  688. }
  689. void MainPanel::hideFloatingToolbar()
  690. {
  691. if (m_compactFrame)
  692. m_compactFrame->hide();
  693. }
  694. // 新增:根据模式与窗口大小自适应布局
  695. void MainPanel::applyModeLayout()
  696. {
  697. // 若控件未初始化,直接返回
  698. if (!splitter || !playerContainer) return;
  699. // 推流期间,主窗口处于隐藏/极简状态,这里只保证分割区域隐藏
  700. if (m_isStreaming) {
  701. if (m_rightWidget) m_rightWidget->hide();
  702. splitter->hide();
  703. return;
  704. }
  705. const bool isRecorder = qobject_cast<RecorderWidget*>(playerWidget) != nullptr;
  706. const bool isPlayer = qobject_cast<AVPlayerWidget*>(playerWidget) != nullptr;
  707. // 保证分割窗口可见,用于布局
  708. splitter->show();
  709. const int panelW = width() > 0 ? width() : 1200;
  710. if (isPlayer) {
  711. // 播放模式:根据聊天是否嵌入来决定右侧面板显示
  712. const bool embeddedChat = (m_chatTabWidget && m_chatContainer && m_chatTabWidget->parent() == m_chatContainer);
  713. if (embeddedChat) {
  714. if (m_rightWidget) m_rightWidget->show();
  715. int chatW = panelW / 3;
  716. chatW = qBound(300, chatW, 420);
  717. const int leftW = qMax(0, panelW - chatW);
  718. QList<int> sizes; sizes << leftW << chatW;
  719. splitter->setSizes(sizes);
  720. } else {
  721. if (m_rightWidget) m_rightWidget->hide();
  722. QList<int> sizes; sizes << panelW << 0;
  723. splitter->setSizes(sizes);
  724. }
  725. return;
  726. }
  727. // 录制模式(非推流):展示右侧面板,根据聊天当前状态决定布局
  728. if (m_rightWidget) m_rightWidget->show();
  729. // 检查聊天当前是否为嵌入状态
  730. const bool embeddedChat = (m_chatTabWidget && m_chatContainer && m_chatTabWidget->parent() == m_chatContainer);
  731. if (embeddedChat) {
  732. // 聊天为嵌入状态,设置分割布局
  733. int chatW = panelW / 3;
  734. chatW = qBound(300, chatW, 420);
  735. const int leftW = qMax(0, panelW - chatW);
  736. QList<int> sizes; sizes << leftW << chatW;
  737. splitter->setSizes(sizes);
  738. } else {
  739. // 聊天为独立窗口状态,隐藏右侧面板
  740. if (m_rightWidget) m_rightWidget->hide();
  741. QList<int> sizes; sizes << panelW << 0;
  742. splitter->setSizes(sizes);
  743. }
  744. }
  745. void MainPanel::resizeEvent(QResizeEvent* event)
  746. {
  747. QWidget::resizeEvent(event);
  748. // 随窗口大小调整左右比例
  749. applyModeLayout();
  750. }
  751. bool MainPanel::eventFilter(QObject *watched, QEvent *event)
  752. {
  753. // 拦截聊天独立窗口的关闭事件
  754. if (watched == m_chatFrame && event->type() == QEvent::Close) {
  755. // 推流期间:关闭操作改为隐藏窗口,避免销毁与频繁 reparent 导致异常
  756. if (m_isStreaming) {
  757. event->ignore();
  758. if (m_chatFrame) {
  759. m_chatFrame->hide();
  760. }
  761. if (m_chatButton)
  762. m_chatButton->setText(tr("显示聊天"));
  763. return true; // 事件已处理,不再继续关闭
  764. }
  765. // 非推流:允许关闭,但先把 m_chatTabWidget 放回嵌入容器,避免被父窗口销毁
  766. if (m_chatTabWidget && m_chatContainer && m_chatTabWidget->parent() == m_chatFrame) {
  767. // 取走中央部件,避免 setCentralWidget(nullptr) 触发潜在删除
  768. if (m_chatFrame->centralWidget()) {
  769. QWidget *w = m_chatFrame->takeCentralWidget();
  770. Q_UNUSED(w);
  771. }
  772. m_chatTabWidget->hide();
  773. m_chatTabWidget->setParent(m_chatContainer);
  774. m_chatTabWidget->setWindowTitle(QString());
  775. if (auto l = qobject_cast<QVBoxLayout *>(m_chatContainer->layout())) {
  776. if (l->indexOf(m_chatTabWidget) < 0)
  777. l->addWidget(m_chatTabWidget);
  778. } else {
  779. auto l2 = new QVBoxLayout(m_chatContainer);
  780. l2->setContentsMargins(0, 0, 0, 0);
  781. l2->addWidget(m_chatTabWidget);
  782. }
  783. m_chatTabWidget->show();
  784. if (m_chatButton)
  785. m_chatButton->setText(tr("聊天"));
  786. applyModeLayout();
  787. }
  788. return false; // 继续关闭流程
  789. }
  790. return QWidget::eventFilter(watched, event);
  791. }
  792. void MainPanel::startStreamingTest(const QString &roomId, bool enableRecording)
  793. {
  794. qDebug() << "开始推流拉流测试,房间ID:" << roomId << ",启用录制:" << enableRecording;
  795. // 1. 创建或获取RecorderWidget实例用于推流
  796. if (!m_recorderStandalone) {
  797. m_recorderStandalone = new RecorderWidget();
  798. // 连接推流相关信号
  799. connect(m_recorderStandalone, &RecorderWidget::streamingStarted, this, [this]() {
  800. qDebug() << "推流已开始";
  801. });
  802. connect(m_recorderStandalone, &RecorderWidget::streamingStopped, this, [this]() {
  803. qDebug() << "推流已停止";
  804. });
  805. connect(m_recorderStandalone, &RecorderWidget::errorOccurred, this, [this](const QString &error) {
  806. qDebug() << "推流错误:" << error;
  807. });
  808. }
  809. // 2. 创建或获取AVPlayerWidget实例用于拉流
  810. if (!m_avPlayerStandalone) {
  811. m_avPlayerStandalone = new AVPlayerWidget();
  812. // 连接拉流相关信号
  813. connect(m_avPlayerStandalone, &AVPlayerWidget::playStateChanged, this, [this](bool isPlaying) {
  814. qDebug() << "播放状态变化:" << (isPlaying ? "播放中" : "已停止");
  815. });
  816. connect(m_avPlayerStandalone, &AVPlayerWidget::playError, this, [this](const QString &error) {
  817. qDebug() << "播放错误:" << error;
  818. });
  819. connect(m_avPlayerStandalone, &AVPlayerWidget::playLoadingStarted, this, [this]() {
  820. qDebug() << "开始加载流媒体";
  821. });
  822. connect(m_avPlayerStandalone, &AVPlayerWidget::playLoadingFinished, this, [this]() {
  823. qDebug() << "流媒体加载完成";
  824. });
  825. }
  826. // 3. 配置推流参数
  827. m_recorderStandalone->setLiveName(roomId);
  828. // 4. 配置拉流参数
  829. m_avPlayerStandalone->setPlayRoomId(roomId);
  830. // 5. 显示推流和拉流窗口
  831. m_recorderStandalone->show();
  832. m_recorderStandalone->resize(800, 600);
  833. m_recorderStandalone->move(0, 0);
  834. m_recorderStandalone->setWindowTitle(QString("推流测试 - 房间ID: %1").arg(roomId));
  835. m_avPlayerStandalone->show();
  836. m_avPlayerStandalone->resize(800, 600);
  837. m_recorderStandalone->move(800, 600);
  838. m_avPlayerStandalone->setWindowTitle(QString("拉流测试 - 房间ID: %1").arg(roomId));
  839. // 6. 启动推流
  840. if (m_recorderStandalone->startStreaming()) {
  841. qDebug() << "推流启动成功";
  842. } else {
  843. qDebug() << "推流启动失败";
  844. }
  845. // 7. 延迟启动拉流,等待推流稳定
  846. QTimer::singleShot(5000, this, [this]() {
  847. if (m_avPlayerStandalone) {
  848. m_avPlayerStandalone->startPlayAsync();
  849. qDebug() << "延迟启动拉流";
  850. }
  851. });
  852. // 8. 如果启用录制,同时开始录制
  853. if (enableRecording) {
  854. QTimer::singleShot(1000, this, [this]() {
  855. if (m_recorderStandalone) {
  856. m_recorderStandalone->startRecording();
  857. qDebug() << "同时启动录制功能";
  858. }
  859. });
  860. }
  861. }
  862. void MainPanel::stopStreamingTest()
  863. {
  864. qDebug() << "停止推流拉流测试";
  865. // 1. 停止推流
  866. if (m_recorderStandalone) {
  867. m_recorderStandalone->stopStreaming();
  868. m_recorderStandalone->stopRecording(); // 同时停止录制
  869. m_recorderStandalone->hide();
  870. }
  871. // 2. 停止拉流
  872. if (m_avPlayerStandalone) {
  873. m_avPlayerStandalone->stop();
  874. m_avPlayerStandalone->hide();
  875. }
  876. qDebug() << "推流拉流测试已停止";
  877. }