MainPanel.cpp 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995
  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. connect(webSocketClient, &WebSocketClient::liveStatus, this, [this](const QString &msg) {
  221. // 这里可以处理 liveStatus 相关逻辑
  222. QJsonParseError err;
  223. QJsonDocument doc = QJsonDocument::fromJson(msg.toUtf8(), &err);
  224. if (err.error != QJsonParseError::NoError || !doc.isObject()) {
  225. qDebug() << "[MainPanel] liveStatus: 解析失败" << err.errorString();
  226. return;
  227. }
  228. QJsonObject obj = doc.object();
  229. int liveStatus = obj.value("liveStatus").toInt(0); // 默认-1
  230. if (liveStatus == 1) {
  231. qDebug() << "[MainPanel] liveStatus: 直播中" << m_chatWindow;
  232. if (m_chatWindow) {
  233. const QString id = webSocketClient->roomId();
  234. // 使用防抖机制处理频繁的请求
  235. m_pendingRoomId = id;
  236. m_debounceTimer->start(); // 重新开始计时,如果在500ms内再次收到请求,会重置定时器
  237. }
  238. }
  239. });
  240. // 初始化音频设备列表
  241. initAudioDeviceSelectors();
  242. }
  243. MainPanel::~MainPanel()
  244. {
  245. if (m_avPlayerStandalone) {
  246. m_avPlayerStandalone->deleteLater();
  247. m_avPlayerStandalone = nullptr;
  248. }
  249. if (m_recorderStandalone) {
  250. m_recorderStandalone->deleteLater();
  251. m_recorderStandalone = nullptr;
  252. }
  253. }
  254. void MainPanel::setRole(const QStringList &roleList)
  255. {
  256. bool isRec = roleList.contains("role.admin") || roleList.contains("role.recorder") || roleList.contains("录制");
  257. if (isRec) {
  258. auto rec = new RecorderWidget(this);
  259. setPlayerWidget(rec);
  260. } else {
  261. auto av = new AVPlayerWidget(this);
  262. setPlayerWidget(av);
  263. }
  264. if (m_streamButton) {
  265. m_streamButton->setVisible(isRec);
  266. m_streamButton->setText(tr("推流"));
  267. }
  268. }
  269. void MainPanel::setPushRoomId(const QString &id)
  270. {
  271. if (!webSocketClient) return;
  272. // 初始化聊天室连接
  273. webSocketClient->connectToRoom(id);
  274. if (m_chatWindow) {
  275. m_chatWindow->initWebsocket(id);
  276. }
  277. // 若当前是播放器,使用防抖播放
  278. m_pendingRoomId = id;
  279. m_debounceTimer->start();
  280. }
  281. void MainPanel::setPlayerWidget(QWidget *newPlayer)
  282. {
  283. if (!newPlayer) return;
  284. if (!playerContainer) return;
  285. if (playerWidget) {
  286. playerWidget->deleteLater();
  287. playerWidget = nullptr;
  288. }
  289. playerWidget = newPlayer;
  290. // 将新播放器添加到容器布局
  291. if (auto layout = qobject_cast<QVBoxLayout*>(playerContainer->layout())) {
  292. layout->insertWidget(0, playerWidget, 1); // 使用拉伸因子1,让播放器组件充分利用空间
  293. } else {
  294. auto layout2 = new QVBoxLayout(playerContainer);
  295. layout2->setContentsMargins(0, 0, 0, 0);
  296. layout2->setSpacing(0);
  297. layout2->addWidget(playerWidget, 1); // 使用拉伸因子1,让播放器组件充分利用空间
  298. }
  299. // 更新推流按钮可见性
  300. if (m_streamButton) {
  301. const bool isRec = qobject_cast<RecorderWidget*>(playerWidget) != nullptr;
  302. m_streamButton->setVisible(isRec);
  303. if (!isRec) {
  304. m_isStreaming = false;
  305. m_streamButton->setText(tr("推流"));
  306. }
  307. }
  308. // 连接 RecorderWidget 的推流信号,保持 UI 状态同步
  309. if (auto rec = qobject_cast<RecorderWidget*>(playerWidget)) {
  310. connect(rec, &RecorderWidget::streamingStarted, this, [this, rec]() {
  311. m_isStreaming = true;
  312. if (m_streamButton) m_streamButton->setText(tr("停止推流"));
  313. // 进入极简模式
  314. if (QWidget *tlw = window()) {
  315. tlw->setUpdatesEnabled(false);
  316. }
  317. rec->hidePreview();
  318. if (m_chatTabWidget) m_chatTabWidget->hide();
  319. if (m_rightWidget) m_rightWidget->hide();
  320. if (splitter) splitter->hide();
  321. // 使用自定义标题的紧凑浮窗承载工具栏
  322. if (!m_compactFrame) {
  323. m_compactFrame = new TMainWindow();
  324. m_compactFrame->setWindowTitle(tr("共享控制"));
  325. // m_compactFrame->setWindowFlag(Qt::Tool, true);
  326. // m_compactFrame->setAttribute(Qt::WA_DeleteOnClose);
  327. if (buttonGroup) {
  328. if (auto layout = qobject_cast<QVBoxLayout*>(playerContainer->layout())) {
  329. layout->removeWidget(buttonGroup);
  330. }
  331. auto container = new QWidget(m_compactFrame);
  332. auto lay = new QHBoxLayout(container);
  333. lay->setContentsMargins(12, 8, 12, 8);
  334. lay->setSpacing(10);
  335. buttonGroup->setParent(container);
  336. lay->addWidget(buttonGroup);
  337. m_compactFrame->setCentralWidget(container);
  338. // 使用 sizeHint 直接设置初始尺寸,避免反复 adjustSize 带来的重算
  339. const QSize hint = buttonGroup->sizeHint();
  340. int w = qMax(360, hint.width() + 24);
  341. int h = qMax(80, hint.height() + 16);
  342. int titleH = 0;
  343. if (auto mw = m_compactFrame->menuWidget()) {
  344. mw->adjustSize();
  345. titleH = mw->sizeHint().height();
  346. }
  347. m_compactFrame->setMinimumSize(w, h + titleH);
  348. m_compactFrame->resize(w, h + titleH);
  349. }
  350. }
  351. m_compactFrame->show();
  352. m_compactFrame->raise();
  353. m_compactFrame->activateWindow();
  354. // 将窗口移动到屏幕中上位置
  355. QRect screenGeometry = QApplication::primaryScreen()->availableGeometry();
  356. int x = (screenGeometry.width() - m_compactFrame->width()) / 2;
  357. int y = screenGeometry.height() / 4; // 屏幕高度的1/4位置,即中上
  358. m_compactFrame->move(screenGeometry.x() + x, screenGeometry.y() + y);
  359. m_compactMode = true;
  360. // 主窗口保持原几何但隐藏
  361. if (QWidget *tlw = window()) {
  362. m_savedWindowGeometry = tlw->geometry();
  363. tlw->hide();
  364. }
  365. });
  366. connect(rec, &RecorderWidget::streamingStopped, this, [this, rec]() {
  367. m_isStreaming = false;
  368. if (m_streamButton) m_streamButton->setText(tr("推流"));
  369. // 退出极简模式
  370. if (QWidget *tlw = window()) {
  371. tlw->setUpdatesEnabled(false);
  372. }
  373. rec->showPreview();
  374. if (splitter) splitter->show();
  375. if (m_rightWidget) m_rightWidget->show();
  376. // 销毁紧凑浮窗并将 buttonGroup 归还
  377. if (m_compactFrame) {
  378. if (buttonGroup) {
  379. buttonGroup->hide();
  380. buttonGroup->setParent(playerContainer);
  381. if (auto layout = qobject_cast<QVBoxLayout*>(playerContainer->layout())) {
  382. layout->addWidget(buttonGroup, 0);
  383. }
  384. buttonGroup->show();
  385. }
  386. m_compactFrame->close();
  387. m_compactFrame->deleteLater();
  388. m_compactFrame = nullptr;
  389. }
  390. m_compactMode = false;
  391. // 恢复主窗口显示
  392. if (QWidget *tlw = window()) {
  393. tlw->setGeometry(m_savedWindowGeometry);
  394. tlw->show();
  395. tlw->raise();
  396. tlw->setUpdatesEnabled(true);
  397. }
  398. showChatEmbedded();
  399. if (m_chatButton) m_chatButton->setText(tr("聊天"));
  400. applyModeLayout(); // 恢复布局为非推流状态
  401. });
  402. }
  403. // 根据当前模式(录制/播放)应用一次自适应布局
  404. applyModeLayout();
  405. }
  406. void MainPanel::handleDebouncedPlay()
  407. {
  408. // 在定时器触发时执行播放逻辑
  409. const QString id = m_pendingRoomId;
  410. if (id.isEmpty()) return;
  411. qDebug() << "[MainPanel] Debounced startPlay for room" << id;
  412. RecorderWidget *rec = qobject_cast<RecorderWidget *>(playerWidget);
  413. if (rec) {
  414. rec->setLiveName(id);
  415. }
  416. AVPlayerWidget *av = qobject_cast<AVPlayerWidget*>(playerWidget);
  417. if (av) {
  418. // 连接异步播放的错误信号(如果尚未连接)
  419. static bool signalsConnected = false;
  420. if (!signalsConnected) {
  421. connect(av, &AVPlayerWidget::playError, this, [this](const QString &error) {
  422. qDebug() << "[MainPanel] Play error:" << error;
  423. // 显示错误提示
  424. BubbleTip::showTip(this, error, BubbleTip::Bottom, 5000, BubbleTip::Error);
  425. });
  426. connect(av, &AVPlayerWidget::playLoadingStarted, this, [this]() {
  427. qDebug() << "[MainPanel] Play loading started";
  428. // 可以在这里添加全局加载状态指示
  429. });
  430. connect(av, &AVPlayerWidget::playLoadingFinished, this, [this]() {
  431. qDebug() << "[MainPanel] Play loading finished";
  432. // 可以在这里隐藏全局加载状态指示
  433. });
  434. signalsConnected = true;
  435. }
  436. av->stopPlay();
  437. av->setPlayRoomId(id);
  438. av->startPlayAsync(); // 使用异步播放方法
  439. return;
  440. }
  441. }
  442. void MainPanel::initAudioDeviceSelectors()
  443. {
  444. // TODO: 枚举音频设备、填充 m_micWidget 和 m_speakerWidget;枚举编码器填充 m_encoderWidget
  445. }
  446. void MainPanel::showRecorderStandalone()
  447. {
  448. if (m_recorderFrame) {
  449. m_recorderFrame->raise();
  450. m_recorderFrame->activateWindow();
  451. return;
  452. }
  453. m_recorderFrame = new TMainWindow();
  454. m_recorderFrame->setAttribute(Qt::WA_DeleteOnClose);
  455. m_recorderFrame->setWindowTitle(tr("录制器"));
  456. m_recorderStandalone = new RecorderWidget(m_recorderFrame);
  457. m_recorderFrame->setCentralWidget(m_recorderStandalone);
  458. m_recorderFrame->resize(960, 600);
  459. m_recorderFrame->show();
  460. connect(m_recorderFrame, &QObject::destroyed, this, [this]() {
  461. m_recorderFrame = nullptr;
  462. m_recorderStandalone = nullptr;
  463. });
  464. }
  465. void MainPanel::showPlayerStandalone()
  466. {
  467. if (m_playerFrame) {
  468. m_playerFrame->raise();
  469. m_playerFrame->activateWindow();
  470. return;
  471. }
  472. m_playerFrame = new TMainWindow();
  473. m_playerFrame->setAttribute(Qt::WA_DeleteOnClose);
  474. m_playerFrame->setWindowTitle(tr("播放器"));
  475. m_avPlayerStandalone = new AVPlayerWidget(m_playerFrame);
  476. m_playerFrame->setCentralWidget(m_avPlayerStandalone);
  477. m_playerFrame->resize(960, 600);
  478. m_playerFrame->show();
  479. connect(m_playerFrame, &QObject::destroyed, this, [this]() {
  480. m_playerFrame = nullptr;
  481. m_avPlayerStandalone = nullptr;
  482. });
  483. }
  484. void MainPanel::showChatStandalone()
  485. {
  486. // 显示独立聊天窗口 - 调用 showChatInFrame 实现
  487. showChatInFrame();
  488. }
  489. void MainPanel::showChatInFrame()
  490. {
  491. if (!m_chatTabWidget) return;
  492. if (m_isStreaming) return; // 推流时保持仅浮窗
  493. // - if (m_chatFrame) {
  494. // - m_chatFrame->raise();
  495. // - m_chatFrame->activateWindow();
  496. // - return;
  497. // - }
  498. if (m_chatFrame) {
  499. // 确保 m_chatTabWidget 已正确作为中央部件挂载
  500. if (m_chatFrame->centralWidget() != m_chatTabWidget && m_chatTabWidget) {
  501. m_chatTabWidget->hide();
  502. if (m_chatTabWidget->parent() != m_chatFrame) {
  503. m_chatTabWidget->setParent(m_chatFrame);
  504. }
  505. m_chatFrame->setCentralWidget(m_chatTabWidget);
  506. m_chatTabWidget->show();
  507. }
  508. if (!m_chatFrame->isVisible())
  509. m_chatFrame->show();
  510. m_chatFrame->raise();
  511. m_chatFrame->activateWindow();
  512. return;
  513. }
  514. // 从嵌入容器移除
  515. if (m_chatContainer) {
  516. if (auto l = qobject_cast<QVBoxLayout*>(m_chatContainer->layout())) {
  517. l->removeWidget(m_chatTabWidget);
  518. }
  519. }
  520. m_chatFrame = new TMainWindow();
  521. m_chatFrame->setAttribute(Qt::WA_DeleteOnClose);
  522. m_chatFrame->setWindowTitle(tr("聊天"));
  523. m_chatFrame->installEventFilter(this);
  524. m_chatTabWidget->setParent(m_chatFrame);
  525. m_chatFrame->setCentralWidget(m_chatTabWidget);
  526. m_chatFrame->resize(380, 540);
  527. m_chatFrame->show();
  528. // - connect(m_chatFrame, &QObject::destroyed, this, [this]() {
  529. // - m_chatFrame = nullptr;
  530. // - onChatWindowCloseRequested();
  531. // - });
  532. connect(m_chatFrame, &QObject::destroyed, this, [this]() { m_chatFrame = nullptr; });
  533. }
  534. void MainPanel::showChatEmbedded()
  535. {
  536. if (!m_chatTabWidget || !m_chatContainer) return;
  537. // 如存在独立窗口包装,先将 m_chatTabWidget 移回容器再关闭窗口,避免被父窗口销毁
  538. if (m_chatFrame) {
  539. // 断开临时回调,避免 destroyed 中的副作用
  540. disconnect(m_chatFrame, nullptr, this, nullptr);
  541. if (m_chatTabWidget->parent() == m_chatFrame) {
  542. // 使用 takeCentralWidget 而非 setCentralWidget(nullptr) 避免误删中央部件
  543. if (m_chatFrame->centralWidget()) {
  544. QWidget *w = m_chatFrame->takeCentralWidget();
  545. Q_UNUSED(w);
  546. }
  547. }
  548. m_chatTabWidget->hide();
  549. m_chatTabWidget->setParent(m_chatContainer);
  550. m_chatTabWidget->setWindowTitle(QString());
  551. if (auto l = qobject_cast<QVBoxLayout *>(m_chatContainer->layout())) {
  552. if (l->indexOf(m_chatTabWidget) < 0)
  553. l->addWidget(m_chatTabWidget);
  554. } else {
  555. auto l2 = new QVBoxLayout(m_chatContainer);
  556. l2->setContentsMargins(0, 0, 0, 0);
  557. l2->addWidget(m_chatTabWidget);
  558. }
  559. m_chatTabWidget->show();
  560. // 现在安全地关闭独立窗口
  561. m_chatFrame->close();
  562. m_chatFrame = nullptr;
  563. } else {
  564. // 无独立窗口,仅确保嵌入
  565. m_chatTabWidget->hide();
  566. m_chatTabWidget->setParent(m_chatContainer);
  567. m_chatTabWidget->setWindowTitle(QString());
  568. if (auto l = qobject_cast<QVBoxLayout *>(m_chatContainer->layout())) {
  569. if (l->indexOf(m_chatTabWidget) < 0)
  570. l->addWidget(m_chatTabWidget);
  571. } else {
  572. auto l2 = new QVBoxLayout(m_chatContainer);
  573. l2->setContentsMargins(0, 0, 0, 0);
  574. l2->addWidget(m_chatTabWidget);
  575. }
  576. m_chatTabWidget->show();
  577. }
  578. }
  579. void MainPanel::onRecordButtonClicked()
  580. {
  581. // TODO: 录制控制逻辑
  582. }
  583. void MainPanel::onStreamButtonClicked()
  584. {
  585. RecorderWidget *rec = qobject_cast<RecorderWidget*>(playerWidget);
  586. if (!rec) {
  587. qDebug() << "[MainPanel] 当前不处于录制器模式,无法推流";
  588. return;
  589. }
  590. if (!m_isStreaming) {
  591. rec->startStreaming(); // 成功与否由信号驱动 UI
  592. } else {
  593. rec->stopStreaming();
  594. }
  595. }
  596. void MainPanel::onChatWindowCloseRequested()
  597. {
  598. // 如果处在推流状态,认为是“隐藏聊天”而不是关闭程序
  599. if (m_isStreaming) {
  600. // 显式隐藏独立聊天窗口,统一行为到 eventFilter 的隐藏逻辑
  601. if (m_chatFrame && m_chatFrame->isVisible()) {
  602. m_chatFrame->hide();
  603. }
  604. if (m_chatButton) m_chatButton->setText(tr("显示聊天"));
  605. return; // ChatWindow 本身不要销毁
  606. }
  607. // 非推流:回归嵌入显示
  608. showChatEmbedded();
  609. if (m_chatButton) m_chatButton->setText(tr("聊天"));
  610. applyModeLayout();
  611. }
  612. void MainPanel::onChatButtonClicked()
  613. {
  614. if (!m_chatTabWidget) return;
  615. if (m_isStreaming) {
  616. // 推流时:只在独立聊天窗口上 显示/隐藏 切换,避免频繁 reparent
  617. if (!m_chatFrame) {
  618. // 如果当前在嵌入容器里,先从布局移除
  619. if (m_chatContainer) {
  620. if (auto l = qobject_cast<QVBoxLayout*>(m_chatContainer->layout())) {
  621. l->removeWidget(m_chatTabWidget);
  622. }
  623. }
  624. m_chatFrame = new TMainWindow();
  625. m_chatFrame->setAttribute(Qt::WA_DeleteOnClose);
  626. m_chatFrame->setWindowTitle(tr("聊天"));
  627. m_chatFrame->installEventFilter(this);
  628. m_chatTabWidget->setParent(m_chatFrame);
  629. m_chatFrame->setCentralWidget(m_chatTabWidget);
  630. m_chatTabWidget->show();
  631. m_chatFrame->resize(380, 540);
  632. m_chatFrame->show();
  633. m_chatFrame->raise();
  634. m_chatFrame->activateWindow();
  635. connect(m_chatFrame, &QObject::destroyed, this, [this]() { m_chatFrame = nullptr; });
  636. if (m_chatButton) m_chatButton->setText(tr("隐藏聊天"));
  637. return; // 结束推流分支处理
  638. } else {
  639. // 已有独立窗口:切换显示/隐藏
  640. if (m_chatTabWidget->parent() != m_chatFrame) {
  641. m_chatTabWidget->setParent(m_chatFrame);
  642. if (m_chatFrame->centralWidget() != m_chatTabWidget)
  643. m_chatFrame->setCentralWidget(m_chatTabWidget);
  644. }
  645. m_chatTabWidget->show();
  646. if (m_chatFrame->isVisible()) {
  647. m_chatFrame->hide();
  648. if (m_chatButton) m_chatButton->setText(tr("显示聊天"));
  649. } else {
  650. m_chatFrame->show();
  651. m_chatFrame->raise();
  652. m_chatFrame->activateWindow();
  653. if (m_chatButton) m_chatButton->setText(tr("隐藏聊天"));
  654. }
  655. return; // 结束推流分支处理
  656. }
  657. }
  658. // 非推流:在弹出与嵌入间切换
  659. if (m_chatFrame) {
  660. // 已经是弹出状态,切回嵌入
  661. showChatEmbedded();
  662. if (m_chatButton)
  663. m_chatButton->setText(tr("聊天"));
  664. applyModeLayout();
  665. } else {
  666. // 目前为嵌入状态,弹出为独立窗口
  667. showChatInFrame();
  668. if (m_chatButton)
  669. m_chatButton->setText(tr("嵌入聊天"));
  670. applyModeLayout();
  671. }
  672. }
  673. // ===== 浮动工具栏 =====
  674. void MainPanel::showFloatingToolbar()
  675. {
  676. // 改为使用 m_compactFrame 承载,无需再在主窗口中悬浮
  677. if (!m_compactFrame)
  678. return;
  679. m_compactFrame->show();
  680. m_compactFrame->raise();
  681. }
  682. void MainPanel::hideFloatingToolbar()
  683. {
  684. if (m_compactFrame)
  685. m_compactFrame->hide();
  686. }
  687. // 新增:根据模式与窗口大小自适应布局
  688. void MainPanel::applyModeLayout()
  689. {
  690. // 若控件未初始化,直接返回
  691. if (!splitter || !playerContainer) return;
  692. // 推流期间,主窗口处于隐藏/极简状态,这里只保证分割区域隐藏
  693. if (m_isStreaming) {
  694. if (m_rightWidget) m_rightWidget->hide();
  695. splitter->hide();
  696. return;
  697. }
  698. const bool isRecorder = qobject_cast<RecorderWidget*>(playerWidget) != nullptr;
  699. const bool isPlayer = qobject_cast<AVPlayerWidget*>(playerWidget) != nullptr;
  700. // 保证分割窗口可见,用于布局
  701. splitter->show();
  702. const int panelW = width() > 0 ? width() : 1200;
  703. if (isPlayer) {
  704. // 播放模式:根据聊天是否嵌入来决定右侧面板显示
  705. const bool embeddedChat = (m_chatTabWidget && m_chatContainer && m_chatTabWidget->parent() == m_chatContainer);
  706. if (embeddedChat) {
  707. if (m_rightWidget) m_rightWidget->show();
  708. int chatW = panelW / 3;
  709. chatW = qBound(300, chatW, 420);
  710. const int leftW = qMax(0, panelW - chatW);
  711. QList<int> sizes; sizes << leftW << chatW;
  712. splitter->setSizes(sizes);
  713. } else {
  714. if (m_rightWidget) m_rightWidget->hide();
  715. QList<int> sizes; sizes << panelW << 0;
  716. splitter->setSizes(sizes);
  717. }
  718. return;
  719. }
  720. // 录制模式(非推流):展示右侧面板,根据聊天当前状态决定布局
  721. if (m_rightWidget) m_rightWidget->show();
  722. // 检查聊天当前是否为嵌入状态
  723. const bool embeddedChat = (m_chatTabWidget && m_chatContainer && m_chatTabWidget->parent() == m_chatContainer);
  724. if (embeddedChat) {
  725. // 聊天为嵌入状态,设置分割布局
  726. int chatW = panelW / 3;
  727. chatW = qBound(300, chatW, 420);
  728. const int leftW = qMax(0, panelW - chatW);
  729. QList<int> sizes; sizes << leftW << chatW;
  730. splitter->setSizes(sizes);
  731. } else {
  732. // 聊天为独立窗口状态,隐藏右侧面板
  733. if (m_rightWidget) m_rightWidget->hide();
  734. QList<int> sizes; sizes << panelW << 0;
  735. splitter->setSizes(sizes);
  736. }
  737. }
  738. void MainPanel::resizeEvent(QResizeEvent* event)
  739. {
  740. QWidget::resizeEvent(event);
  741. // 随窗口大小调整左右比例
  742. applyModeLayout();
  743. }
  744. bool MainPanel::eventFilter(QObject *watched, QEvent *event)
  745. {
  746. // 拦截聊天独立窗口的关闭事件
  747. if (watched == m_chatFrame && event->type() == QEvent::Close) {
  748. // 推流期间:关闭操作改为隐藏窗口,避免销毁与频繁 reparent 导致异常
  749. if (m_isStreaming) {
  750. event->ignore();
  751. if (m_chatFrame) {
  752. m_chatFrame->hide();
  753. }
  754. if (m_chatButton)
  755. m_chatButton->setText(tr("显示聊天"));
  756. return true; // 事件已处理,不再继续关闭
  757. }
  758. // 非推流:允许关闭,但先把 m_chatTabWidget 放回嵌入容器,避免被父窗口销毁
  759. if (m_chatTabWidget && m_chatContainer && m_chatTabWidget->parent() == m_chatFrame) {
  760. // 取走中央部件,避免 setCentralWidget(nullptr) 触发潜在删除
  761. if (m_chatFrame->centralWidget()) {
  762. QWidget *w = m_chatFrame->takeCentralWidget();
  763. Q_UNUSED(w);
  764. }
  765. m_chatTabWidget->hide();
  766. m_chatTabWidget->setParent(m_chatContainer);
  767. m_chatTabWidget->setWindowTitle(QString());
  768. if (auto l = qobject_cast<QVBoxLayout *>(m_chatContainer->layout())) {
  769. if (l->indexOf(m_chatTabWidget) < 0)
  770. l->addWidget(m_chatTabWidget);
  771. } else {
  772. auto l2 = new QVBoxLayout(m_chatContainer);
  773. l2->setContentsMargins(0, 0, 0, 0);
  774. l2->addWidget(m_chatTabWidget);
  775. }
  776. m_chatTabWidget->show();
  777. if (m_chatButton)
  778. m_chatButton->setText(tr("聊天"));
  779. applyModeLayout();
  780. }
  781. return false; // 继续关闭流程
  782. }
  783. return QWidget::eventFilter(watched, event);
  784. }
  785. void MainPanel::startStreamingTest(const QString &roomId, bool enableRecording)
  786. {
  787. qDebug() << "开始推流拉流测试,房间ID:" << roomId << ",启用录制:" << enableRecording;
  788. // 1. 创建或获取RecorderWidget实例用于推流
  789. if (!m_recorderStandalone) {
  790. m_recorderStandalone = new RecorderWidget();
  791. // 连接推流相关信号
  792. connect(m_recorderStandalone, &RecorderWidget::streamingStarted, this, [this]() {
  793. qDebug() << "推流已开始";
  794. });
  795. connect(m_recorderStandalone, &RecorderWidget::streamingStopped, this, [this]() {
  796. qDebug() << "推流已停止";
  797. });
  798. connect(m_recorderStandalone, &RecorderWidget::errorOccurred, this, [this](const QString &error) {
  799. qDebug() << "推流错误:" << error;
  800. });
  801. }
  802. // 2. 创建或获取AVPlayerWidget实例用于拉流
  803. if (!m_avPlayerStandalone) {
  804. m_avPlayerStandalone = new AVPlayerWidget();
  805. // 连接拉流相关信号
  806. connect(m_avPlayerStandalone, &AVPlayerWidget::playStateChanged, this, [this](bool isPlaying) {
  807. qDebug() << "播放状态变化:" << (isPlaying ? "播放中" : "已停止");
  808. });
  809. connect(m_avPlayerStandalone, &AVPlayerWidget::playError, this, [this](const QString &error) {
  810. qDebug() << "播放错误:" << error;
  811. });
  812. connect(m_avPlayerStandalone, &AVPlayerWidget::playLoadingStarted, this, [this]() {
  813. qDebug() << "开始加载流媒体";
  814. });
  815. connect(m_avPlayerStandalone, &AVPlayerWidget::playLoadingFinished, this, [this]() {
  816. qDebug() << "流媒体加载完成";
  817. });
  818. }
  819. // 3. 配置推流参数
  820. m_recorderStandalone->setLiveName(roomId);
  821. // 4. 配置拉流参数
  822. m_avPlayerStandalone->setPlayRoomId(roomId);
  823. // 5. 显示推流和拉流窗口
  824. m_recorderStandalone->show();
  825. m_recorderStandalone->resize(800, 600);
  826. m_recorderStandalone->move(0, 0);
  827. m_recorderStandalone->setWindowTitle(QString("推流测试 - 房间ID: %1").arg(roomId));
  828. m_avPlayerStandalone->show();
  829. m_avPlayerStandalone->resize(800, 600);
  830. m_recorderStandalone->move(800, 600);
  831. m_avPlayerStandalone->setWindowTitle(QString("拉流测试 - 房间ID: %1").arg(roomId));
  832. // 6. 启动推流
  833. if (m_recorderStandalone->startStreaming()) {
  834. qDebug() << "推流启动成功";
  835. } else {
  836. qDebug() << "推流启动失败";
  837. }
  838. // 7. 延迟启动拉流,等待推流稳定
  839. QTimer::singleShot(5000, this, [this]() {
  840. if (m_avPlayerStandalone) {
  841. m_avPlayerStandalone->startPlayAsync();
  842. qDebug() << "延迟启动拉流";
  843. }
  844. });
  845. // 8. 如果启用录制,同时开始录制
  846. if (enableRecording) {
  847. QTimer::singleShot(1000, this, [this]() {
  848. if (m_recorderStandalone) {
  849. m_recorderStandalone->startRecording();
  850. qDebug() << "同时启动录制功能";
  851. }
  852. });
  853. }
  854. }
  855. void MainPanel::stopStreamingTest()
  856. {
  857. qDebug() << "停止推流拉流测试";
  858. // 1. 停止推流
  859. if (m_recorderStandalone) {
  860. m_recorderStandalone->stopStreaming();
  861. m_recorderStandalone->stopRecording(); // 同时停止录制
  862. m_recorderStandalone->hide();
  863. }
  864. // 2. 停止拉流
  865. if (m_avPlayerStandalone) {
  866. m_avPlayerStandalone->stop();
  867. m_avPlayerStandalone->hide();
  868. }
  869. qDebug() << "推流拉流测试已停止";
  870. }