mirror of
https://github.com/bggRGjQaUbCoE/PiliPlus.git
synced 2026-05-26 02:58:39 +00:00
opt: live room
Closes #427 Signed-off-by: bggRGjQaUbCoE <githubaccount56556@proton.me>
This commit is contained in:
@@ -57,15 +57,16 @@ class LiveRoomController extends GetxController {
|
|||||||
// 硬解
|
// 硬解
|
||||||
enableHA: true,
|
enableHA: true,
|
||||||
autoplay: true,
|
autoplay: true,
|
||||||
|
direction: isPortrait.value ? 'vertical' : 'horizontal',
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool? isPortrait;
|
final RxBool isPortrait = false.obs;
|
||||||
|
|
||||||
Future queryLiveInfo() async {
|
Future queryLiveInfo() async {
|
||||||
var res = await LiveHttp.liveRoomInfo(roomId: roomId, qn: currentQn);
|
var res = await LiveHttp.liveRoomInfo(roomId: roomId, qn: currentQn);
|
||||||
if (res['status']) {
|
if (res['status']) {
|
||||||
isPortrait = res['data'].isPortrait;
|
isPortrait.value = res['data'].isPortrait ?? false;
|
||||||
List<CodecItem> codec =
|
List<CodecItem> codec =
|
||||||
res['data'].playurlInfo.playurl.stream.first.format.first.codec;
|
res['data'].playurlInfo.playurl.stream.first.format.first.codec;
|
||||||
CodecItem item = codec.first;
|
CodecItem item = codec.first;
|
||||||
|
|||||||
@@ -4,7 +4,9 @@ import 'dart:math';
|
|||||||
|
|
||||||
import 'package:PiliPlus/http/live.dart';
|
import 'package:PiliPlus/http/live.dart';
|
||||||
import 'package:PiliPlus/pages/live_room/widgets/chat.dart';
|
import 'package:PiliPlus/pages/live_room/widgets/chat.dart';
|
||||||
|
import 'package:PiliPlus/utils/extension.dart';
|
||||||
import 'package:PiliPlus/utils/utils.dart';
|
import 'package:PiliPlus/utils/utils.dart';
|
||||||
|
import 'package:cached_network_image/cached_network_image.dart';
|
||||||
import 'package:canvas_danmaku/canvas_danmaku.dart';
|
import 'package:canvas_danmaku/canvas_danmaku.dart';
|
||||||
import 'package:floating/floating.dart';
|
import 'package:floating/floating.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
@@ -33,6 +35,7 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
late final PlPlayerController plPlayerController;
|
late final PlPlayerController plPlayerController;
|
||||||
late Future? _futureBuilder;
|
late Future? _futureBuilder;
|
||||||
late Future? _futureBuilderFuture;
|
late Future? _futureBuilderFuture;
|
||||||
|
bool get isFullScreen => plPlayerController.isFullScreen.value;
|
||||||
|
|
||||||
bool isShowCover = true;
|
bool isShowCover = true;
|
||||||
bool isPlay = true;
|
bool isPlay = true;
|
||||||
@@ -74,11 +77,6 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
_updateFontSize();
|
_updateFontSize();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
// WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
||||||
// if (context.orientation == Orientation.landscape) {
|
|
||||||
// plPlayerController.triggerFullScreen(status: true);
|
|
||||||
// }
|
|
||||||
// });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void _updateFontSize() async {
|
void _updateFontSize() async {
|
||||||
@@ -89,7 +87,7 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
if (_liveRoomController.controller != null) {
|
if (_liveRoomController.controller != null) {
|
||||||
_liveRoomController.controller!.updateOption(
|
_liveRoomController.controller!.updateOption(
|
||||||
_liveRoomController.controller!.option.copyWith(
|
_liveRoomController.controller!.option.copyWith(
|
||||||
fontSize: _getFontSize(plPlayerController.isFullScreen.value),
|
fontSize: _getFontSize(isFullScreen),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -101,7 +99,7 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
: 15 * plPlayerController.fontSizeFSVal;
|
: 15 * plPlayerController.fontSizeFSVal;
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> videoSourceInit() async {
|
void videoSourceInit() {
|
||||||
_futureBuilder = _liveRoomController.queryLiveInfoH5();
|
_futureBuilder = _liveRoomController.queryLiveInfoH5();
|
||||||
plPlayerController = _liveRoomController.plPlayerController;
|
plPlayerController = _liveRoomController.plPlayerController;
|
||||||
}
|
}
|
||||||
@@ -117,7 +115,6 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
_node.dispose();
|
_node.dispose();
|
||||||
plPlayerController.dispose();
|
plPlayerController.dispose();
|
||||||
_ctr.dispose();
|
_ctr.dispose();
|
||||||
_liveRoomController.scrollController.removeListener(() {});
|
|
||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -133,56 +130,69 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
|
|
||||||
final GlobalKey videoPlayerKey = GlobalKey();
|
final GlobalKey videoPlayerKey = GlobalKey();
|
||||||
final GlobalKey playerKey = GlobalKey();
|
final GlobalKey playerKey = GlobalKey();
|
||||||
|
double? padding;
|
||||||
|
|
||||||
Widget videoPlayerPanel([Color? fill]) {
|
Widget videoPlayerPanel([Color? fill]) {
|
||||||
return FutureBuilder(
|
return PopScope(
|
||||||
key: videoPlayerKey,
|
canPop: !isFullScreen,
|
||||||
future: _futureBuilderFuture,
|
onPopInvokedWithResult: (bool didPop, Object? result) {
|
||||||
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
if (isFullScreen) {
|
||||||
if (snapshot.hasData && snapshot.data['status']) {
|
plPlayerController.triggerFullScreen(status: false);
|
||||||
return PLVideoPlayer(
|
|
||||||
key: playerKey,
|
|
||||||
fill: fill,
|
|
||||||
plPlayerController: plPlayerController,
|
|
||||||
bottomControl: BottomControl(
|
|
||||||
plPlayerController: plPlayerController,
|
|
||||||
liveRoomCtr: _liveRoomController,
|
|
||||||
floating: floating,
|
|
||||||
onRefresh: () {
|
|
||||||
_futureBuilderFuture = _liveRoomController.queryLiveInfo();
|
|
||||||
},
|
|
||||||
),
|
|
||||||
danmuWidget: Obx(
|
|
||||||
() => AnimatedOpacity(
|
|
||||||
opacity: plPlayerController.isOpenDanmu.value ? 1 : 0,
|
|
||||||
duration: const Duration(milliseconds: 100),
|
|
||||||
child: DanmakuScreen(
|
|
||||||
createdController: (DanmakuController e) {
|
|
||||||
plPlayerController.danmakuController =
|
|
||||||
_liveRoomController.controller = e;
|
|
||||||
},
|
|
||||||
option: DanmakuOption(
|
|
||||||
fontSize:
|
|
||||||
_getFontSize(plPlayerController.isFullScreen.value),
|
|
||||||
fontWeight: plPlayerController.fontWeight,
|
|
||||||
area: plPlayerController.showArea,
|
|
||||||
opacity: plPlayerController.opacityVal,
|
|
||||||
hideTop: plPlayerController.blockTypes.contains(5),
|
|
||||||
hideScroll: plPlayerController.blockTypes.contains(2),
|
|
||||||
hideBottom: plPlayerController.blockTypes.contains(4),
|
|
||||||
duration: plPlayerController.danmakuDurationVal ~/
|
|
||||||
plPlayerController.playbackSpeed,
|
|
||||||
strokeWidth: plPlayerController.strokeWidth,
|
|
||||||
lineHeight: plPlayerController.danmakuLineHeight,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return const SizedBox();
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
child: Listener(
|
||||||
|
onPointerDown: (_) {
|
||||||
|
_node.unfocus();
|
||||||
|
},
|
||||||
|
child: FutureBuilder(
|
||||||
|
key: videoPlayerKey,
|
||||||
|
future: _futureBuilderFuture,
|
||||||
|
builder: (BuildContext context, AsyncSnapshot snapshot) {
|
||||||
|
if (snapshot.hasData && snapshot.data['status']) {
|
||||||
|
return PLVideoPlayer(
|
||||||
|
key: playerKey,
|
||||||
|
fill: fill,
|
||||||
|
plPlayerController: plPlayerController,
|
||||||
|
bottomControl: BottomControl(
|
||||||
|
plPlayerController: plPlayerController,
|
||||||
|
liveRoomCtr: _liveRoomController,
|
||||||
|
floating: floating,
|
||||||
|
onRefresh: () {
|
||||||
|
_futureBuilderFuture = _liveRoomController.queryLiveInfo();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
danmuWidget: Obx(
|
||||||
|
() => AnimatedOpacity(
|
||||||
|
opacity: plPlayerController.isOpenDanmu.value ? 1 : 0,
|
||||||
|
duration: const Duration(milliseconds: 100),
|
||||||
|
child: DanmakuScreen(
|
||||||
|
createdController: (DanmakuController e) {
|
||||||
|
plPlayerController.danmakuController =
|
||||||
|
_liveRoomController.controller = e;
|
||||||
|
},
|
||||||
|
option: DanmakuOption(
|
||||||
|
fontSize: _getFontSize(isFullScreen),
|
||||||
|
fontWeight: plPlayerController.fontWeight,
|
||||||
|
area: plPlayerController.showArea,
|
||||||
|
opacity: plPlayerController.opacityVal,
|
||||||
|
hideTop: plPlayerController.blockTypes.contains(5),
|
||||||
|
hideScroll: plPlayerController.blockTypes.contains(2),
|
||||||
|
hideBottom: plPlayerController.blockTypes.contains(4),
|
||||||
|
duration: plPlayerController.danmakuDurationVal ~/
|
||||||
|
plPlayerController.playbackSpeed,
|
||||||
|
strokeWidth: plPlayerController.strokeWidth,
|
||||||
|
lineHeight: plPlayerController.danmakuLineHeight,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
return const SizedBox();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -191,42 +201,41 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
color: Colors.black,
|
color: Colors.black,
|
||||||
child: Stack(
|
child: Stack(
|
||||||
children: [
|
children: [
|
||||||
Positioned.fill(
|
|
||||||
child: Opacity(
|
|
||||||
opacity: 0.6,
|
|
||||||
child: Image.asset(
|
|
||||||
'assets/images/live/default_bg.webp',
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Obx(
|
Obx(
|
||||||
() => _liveRoomController
|
() => isFullScreen
|
||||||
.roomInfoH5.value.roomInfo?.appBackground?.isNotEmpty ==
|
? const SizedBox.shrink()
|
||||||
true
|
: Positioned.fill(
|
||||||
? Positioned.fill(
|
|
||||||
child: Opacity(
|
child: Opacity(
|
||||||
opacity: 0.6,
|
opacity: 0.6,
|
||||||
child: NetworkImgLayer(
|
child: _liveRoomController.roomInfoH5.value.roomInfo
|
||||||
width: Get.width,
|
?.appBackground?.isNotEmpty ==
|
||||||
height: Get.height,
|
true
|
||||||
type: 'bg',
|
? CachedNetworkImage(
|
||||||
src: _liveRoomController
|
fit: BoxFit.cover,
|
||||||
.roomInfoH5.value.roomInfo!.appBackground,
|
width: Get.width,
|
||||||
),
|
height: Get.height,
|
||||||
|
imageUrl: _liveRoomController.roomInfoH5.value
|
||||||
|
.roomInfo!.appBackground!.http2https,
|
||||||
|
)
|
||||||
|
: Image.asset(
|
||||||
|
'assets/images/live/default_bg.webp',
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
)
|
),
|
||||||
: const SizedBox(),
|
|
||||||
),
|
),
|
||||||
isPortrait
|
isPortrait
|
||||||
? Scaffold(
|
? Obx(
|
||||||
backgroundColor: Colors.transparent,
|
() {
|
||||||
body: Column(
|
if (_liveRoomController.isPortrait.value) {
|
||||||
children: [
|
if (padding == null) {
|
||||||
_buildAppBar,
|
final padding = MediaQuery.paddingOf(context);
|
||||||
..._buildBodyP,
|
this.padding = padding.bottom + padding.top;
|
||||||
],
|
}
|
||||||
),
|
return _buildPP;
|
||||||
|
}
|
||||||
|
return _buildPH;
|
||||||
|
},
|
||||||
)
|
)
|
||||||
: Column(
|
: Column(
|
||||||
children: [
|
children: [
|
||||||
@@ -239,6 +248,66 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget get _buildPH => Scaffold(
|
||||||
|
appBar: _buildAppBar,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
body: Column(
|
||||||
|
children: _buildBodyP,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
Widget get _buildPP => Scaffold(
|
||||||
|
appBar: _buildAppBar,
|
||||||
|
backgroundColor: Colors.transparent,
|
||||||
|
body: Stack(
|
||||||
|
children: [
|
||||||
|
Column(
|
||||||
|
children: [
|
||||||
|
Obx(
|
||||||
|
() => Container(
|
||||||
|
color: Colors.black,
|
||||||
|
width: Get.width,
|
||||||
|
height: isFullScreen
|
||||||
|
? Get.height -
|
||||||
|
(context.orientation == Orientation.landscape
|
||||||
|
? 0
|
||||||
|
: MediaQuery.paddingOf(context).top)
|
||||||
|
: Get.height - 56 - 85 - padding!,
|
||||||
|
child: videoPlayerPanel(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
Obx(
|
||||||
|
() => isFullScreen
|
||||||
|
? const SizedBox.shrink()
|
||||||
|
: Positioned(
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
bottom: 125 + MediaQuery.paddingOf(context).bottom,
|
||||||
|
child: SizedBox(
|
||||||
|
height: 125,
|
||||||
|
child: _buildChatWidget,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Obx(
|
||||||
|
() => isFullScreen
|
||||||
|
? const SizedBox.shrink()
|
||||||
|
: Positioned(
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
bottom: 0,
|
||||||
|
child: Padding(
|
||||||
|
padding: EdgeInsets.only(bottom: 0),
|
||||||
|
child: _buildInputWidget,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
@@ -262,93 +331,90 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
|
|
||||||
Color get _color => Color(0xFFEEEEEE);
|
Color get _color => Color(0xFFEEEEEE);
|
||||||
|
|
||||||
Widget get _buildAppBar => Obx(
|
PreferredSizeWidget get _buildAppBar => AppBar(
|
||||||
() => AppBar(
|
backgroundColor: Colors.transparent,
|
||||||
backgroundColor: Colors.transparent,
|
foregroundColor: Colors.white,
|
||||||
foregroundColor: Colors.white,
|
toolbarHeight: isFullScreen ? 0 : null,
|
||||||
titleTextStyle: TextStyle(color: Colors.white),
|
titleTextStyle: TextStyle(color: Colors.white),
|
||||||
toolbarHeight: plPlayerController.isFullScreen.value ? 0 : null,
|
title: FutureBuilder(
|
||||||
title: FutureBuilder(
|
future: _futureBuilder,
|
||||||
future: _futureBuilder,
|
builder: (context, snapshot) {
|
||||||
builder: (context, snapshot) {
|
if (snapshot.data == null) {
|
||||||
if (snapshot.data == null) {
|
return const SizedBox();
|
||||||
return const SizedBox();
|
}
|
||||||
}
|
Map data = snapshot.data as Map;
|
||||||
Map data = snapshot.data as Map;
|
if (data['status']) {
|
||||||
if (data['status']) {
|
return Obx(
|
||||||
return Obx(
|
() => Row(
|
||||||
() => Row(
|
children: [
|
||||||
children: [
|
GestureDetector(
|
||||||
GestureDetector(
|
onTap: () {
|
||||||
onTap: () {
|
_node.unfocus();
|
||||||
_node.unfocus();
|
dynamic uid =
|
||||||
dynamic uid = _liveRoomController
|
_liveRoomController.roomInfoH5.value.roomInfo?.uid;
|
||||||
.roomInfoH5.value.roomInfo?.uid;
|
Get.toNamed(
|
||||||
Get.toNamed(
|
'/member?mid=$uid',
|
||||||
'/member?mid=$uid',
|
arguments: {
|
||||||
arguments: {
|
'heroTag': Utils.makeHeroTag(uid),
|
||||||
'heroTag': Utils.makeHeroTag(uid),
|
},
|
||||||
},
|
);
|
||||||
|
},
|
||||||
|
child: NetworkImgLayer(
|
||||||
|
width: 34,
|
||||||
|
height: 34,
|
||||||
|
type: 'avatar',
|
||||||
|
src: _liveRoomController
|
||||||
|
.roomInfoH5.value.anchorInfo!.baseInfo!.face,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
_liveRoomController
|
||||||
|
.roomInfoH5.value.anchorInfo!.baseInfo!.uname!,
|
||||||
|
style: const TextStyle(fontSize: 14),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 1),
|
||||||
|
if (_liveRoomController.roomInfoH5.value.watchedShow !=
|
||||||
|
null)
|
||||||
|
Text(
|
||||||
|
_liveRoomController.roomInfoH5.value
|
||||||
|
.watchedShow!['text_large'] ??
|
||||||
|
'',
|
||||||
|
style: const TextStyle(fontSize: 12),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const Spacer(),
|
||||||
|
//刷新
|
||||||
|
IconButton(
|
||||||
|
tooltip: '刷新',
|
||||||
|
onPressed: () {
|
||||||
|
_futureBuilderFuture =
|
||||||
|
_liveRoomController.queryLiveInfo();
|
||||||
|
// videoSourceInit();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.refresh),
|
||||||
|
),
|
||||||
|
//内置浏览器打开
|
||||||
|
IconButton(
|
||||||
|
tooltip: '浏览器打开',
|
||||||
|
onPressed: () {
|
||||||
|
Utils.inAppWebview(
|
||||||
|
'https://live.bilibili.com/h5/${_liveRoomController.roomId}',
|
||||||
|
off: true,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
child: NetworkImgLayer(
|
icon: const Icon(Icons.open_in_browser)),
|
||||||
width: 34,
|
],
|
||||||
height: 34,
|
),
|
||||||
type: 'avatar',
|
);
|
||||||
src: _liveRoomController
|
} else {
|
||||||
.roomInfoH5.value.anchorInfo!.baseInfo!.face,
|
return const SizedBox();
|
||||||
),
|
}
|
||||||
),
|
},
|
||||||
const SizedBox(width: 10),
|
|
||||||
Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
_liveRoomController
|
|
||||||
.roomInfoH5.value.anchorInfo!.baseInfo!.uname!,
|
|
||||||
style: const TextStyle(fontSize: 14),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 1),
|
|
||||||
if (_liveRoomController
|
|
||||||
.roomInfoH5.value.watchedShow !=
|
|
||||||
null)
|
|
||||||
Text(
|
|
||||||
_liveRoomController.roomInfoH5.value
|
|
||||||
.watchedShow!['text_large'] ??
|
|
||||||
'',
|
|
||||||
style: const TextStyle(fontSize: 12),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
const Spacer(),
|
|
||||||
//刷新
|
|
||||||
IconButton(
|
|
||||||
tooltip: '刷新',
|
|
||||||
onPressed: () {
|
|
||||||
_futureBuilderFuture =
|
|
||||||
_liveRoomController.queryLiveInfo();
|
|
||||||
// videoSourceInit();
|
|
||||||
},
|
|
||||||
icon: const Icon(Icons.refresh),
|
|
||||||
),
|
|
||||||
//内置浏览器打开
|
|
||||||
IconButton(
|
|
||||||
tooltip: '浏览器打开',
|
|
||||||
onPressed: () {
|
|
||||||
Utils.inAppWebview(
|
|
||||||
'https://live.bilibili.com/h5/${_liveRoomController.roomId}',
|
|
||||||
off: true,
|
|
||||||
);
|
|
||||||
},
|
|
||||||
icon: const Icon(Icons.open_in_browser)),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
} else {
|
|
||||||
return const SizedBox();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -359,33 +425,14 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
child: Row(
|
child: Row(
|
||||||
children: [
|
children: [
|
||||||
Obx(
|
Obx(
|
||||||
() => PopScope(
|
() => Container(
|
||||||
canPop: plPlayerController.isFullScreen.value != true,
|
color: Colors.black,
|
||||||
onPopInvokedWithResult: (bool didPop, Object? result) {
|
width: isFullScreen ? Get.size.width : videoWidth,
|
||||||
if (plPlayerController.isFullScreen.value == true) {
|
height: isFullScreen ? Get.size.height : Get.size.width * 9 / 16,
|
||||||
plPlayerController.triggerFullScreen(status: false);
|
child: MediaQuery.removePadding(
|
||||||
}
|
removeRight: true,
|
||||||
},
|
context: context,
|
||||||
child: Listener(
|
child: videoPlayerPanel(Colors.transparent),
|
||||||
onPointerDown: (_) {
|
|
||||||
_node.unfocus();
|
|
||||||
},
|
|
||||||
child: Container(
|
|
||||||
color: plPlayerController.isFullScreen.value
|
|
||||||
? Colors.black
|
|
||||||
: null,
|
|
||||||
width: plPlayerController.isFullScreen.value
|
|
||||||
? Get.size.width
|
|
||||||
: videoWidth,
|
|
||||||
height: plPlayerController.isFullScreen.value
|
|
||||||
? Get.size.height
|
|
||||||
: Get.size.width * 9 / 16,
|
|
||||||
child: MediaQuery.removePadding(
|
|
||||||
removeRight: true,
|
|
||||||
context: context,
|
|
||||||
child: videoPlayerPanel(Colors.transparent),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -409,26 +456,11 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
|
|
||||||
List<Widget> get _buildBodyP => [
|
List<Widget> get _buildBodyP => [
|
||||||
Obx(
|
Obx(
|
||||||
() => PopScope(
|
() => Container(
|
||||||
canPop: plPlayerController.isFullScreen.value != true,
|
color: Colors.black,
|
||||||
onPopInvokedWithResult: (bool didPop, Object? result) {
|
width: Get.size.width,
|
||||||
if (plPlayerController.isFullScreen.value == true) {
|
height: isFullScreen ? Get.size.height : Get.size.width * 9 / 16,
|
||||||
plPlayerController.triggerFullScreen(status: false);
|
child: videoPlayerPanel(),
|
||||||
}
|
|
||||||
},
|
|
||||||
child: Listener(
|
|
||||||
onPointerDown: (_) {
|
|
||||||
_node.unfocus();
|
|
||||||
},
|
|
||||||
child: Container(
|
|
||||||
color: Colors.black,
|
|
||||||
width: Get.size.width,
|
|
||||||
height: plPlayerController.isFullScreen.value
|
|
||||||
? Get.size.height
|
|
||||||
: Get.size.width * 9 / 16,
|
|
||||||
child: videoPlayerPanel(),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
..._buildBottomWidget,
|
..._buildBottomWidget,
|
||||||
@@ -437,92 +469,91 @@ class _LiveRoomPageState extends State<LiveRoomPage>
|
|||||||
final GlobalKey chatKey = GlobalKey();
|
final GlobalKey chatKey = GlobalKey();
|
||||||
|
|
||||||
List<Widget> get _buildBottomWidget => [
|
List<Widget> get _buildBottomWidget => [
|
||||||
Expanded(
|
Expanded(child: _buildChatWidget),
|
||||||
child: Listener(
|
_buildInputWidget,
|
||||||
onPointerDown: (_) {
|
];
|
||||||
_node.unfocus();
|
|
||||||
},
|
Widget get _buildChatWidget => Listener(
|
||||||
child: Padding(
|
onPointerDown: (_) {
|
||||||
padding: const EdgeInsets.symmetric(vertical: 16),
|
_node.unfocus();
|
||||||
child: LiveRoomChat(
|
},
|
||||||
key: chatKey,
|
child: Padding(
|
||||||
roomId: _roomId,
|
padding: const EdgeInsets.symmetric(vertical: 16),
|
||||||
liveRoomController: _liveRoomController,
|
child: LiveRoomChat(
|
||||||
),
|
key: chatKey,
|
||||||
),
|
roomId: _roomId,
|
||||||
|
liveRoomController: _liveRoomController,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Container(
|
);
|
||||||
padding: EdgeInsets.only(
|
|
||||||
left: 10,
|
Widget get _buildInputWidget => Container(
|
||||||
top: 10,
|
padding: EdgeInsets.only(
|
||||||
right: 10,
|
left: 10,
|
||||||
bottom: 25 + MediaQuery.of(context).padding.bottom,
|
top: 10,
|
||||||
|
right: 10,
|
||||||
|
bottom: 25 + MediaQuery.of(context).padding.bottom,
|
||||||
|
),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
borderRadius: BorderRadius.only(
|
||||||
|
topLeft: Radius.circular(20),
|
||||||
|
topRight: Radius.circular(20),
|
||||||
),
|
),
|
||||||
decoration: BoxDecoration(
|
border: Border(
|
||||||
borderRadius: BorderRadius.only(
|
top: BorderSide(color: Color(0x1AFFFFFF)),
|
||||||
topLeft: Radius.circular(20),
|
|
||||||
topRight: Radius.circular(20),
|
|
||||||
),
|
|
||||||
border: Border(
|
|
||||||
top: BorderSide(color: Color(0x1AFFFFFF)),
|
|
||||||
),
|
|
||||||
color: Color(0x1AFFFFFF),
|
|
||||||
),
|
),
|
||||||
child: Row(
|
color: Color(0x1AFFFFFF),
|
||||||
children: [
|
),
|
||||||
Obx(
|
child: Row(
|
||||||
() => IconButton(
|
children: [
|
||||||
onPressed: () {
|
Obx(
|
||||||
plPlayerController.isOpenDanmu.value =
|
() => IconButton(
|
||||||
!plPlayerController.isOpenDanmu.value;
|
|
||||||
GStorage.setting.put(SettingBoxKey.enableShowDanmaku,
|
|
||||||
plPlayerController.isOpenDanmu.value);
|
|
||||||
},
|
|
||||||
icon: Icon(
|
|
||||||
plPlayerController.isOpenDanmu.value
|
|
||||||
? Icons.subtitles_outlined
|
|
||||||
: Icons.subtitles_off_outlined,
|
|
||||||
color: _color,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Expanded(
|
|
||||||
child: TextField(
|
|
||||||
focusNode: _node,
|
|
||||||
controller: _ctr,
|
|
||||||
textInputAction: TextInputAction.send,
|
|
||||||
cursorColor: _color,
|
|
||||||
style: TextStyle(color: _color),
|
|
||||||
onSubmitted: (value) {
|
|
||||||
if (value.isNotEmpty) {
|
|
||||||
_onSendMsg(value);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
decoration: InputDecoration(
|
|
||||||
border: InputBorder.none,
|
|
||||||
hintText: '发送弹幕',
|
|
||||||
hintStyle: TextStyle(
|
|
||||||
color: Colors.white.withOpacity(0.6),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
IconButton(
|
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
if (_ctr.text.isNotEmpty) {
|
plPlayerController.isOpenDanmu.value =
|
||||||
_onSendMsg(_ctr.text);
|
!plPlayerController.isOpenDanmu.value;
|
||||||
}
|
GStorage.setting.put(SettingBoxKey.enableShowDanmaku,
|
||||||
|
plPlayerController.isOpenDanmu.value);
|
||||||
},
|
},
|
||||||
icon: Icon(
|
icon: Icon(
|
||||||
Icons.send,
|
plPlayerController.isOpenDanmu.value
|
||||||
|
? Icons.subtitles_outlined
|
||||||
|
: Icons.subtitles_off_outlined,
|
||||||
color: _color,
|
color: _color,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
),
|
||||||
),
|
Expanded(
|
||||||
)
|
child: TextField(
|
||||||
];
|
focusNode: _node,
|
||||||
|
controller: _ctr,
|
||||||
|
textInputAction: TextInputAction.send,
|
||||||
|
cursorColor: _color,
|
||||||
|
style: TextStyle(color: _color),
|
||||||
|
onSubmitted: (value) {
|
||||||
|
if (value.isNotEmpty) {
|
||||||
|
_onSendMsg(value);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
decoration: InputDecoration(
|
||||||
|
border: InputBorder.none,
|
||||||
|
hintText: '发送弹幕',
|
||||||
|
hintStyle: TextStyle(
|
||||||
|
color: Colors.white.withOpacity(0.6),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
onPressed: () {
|
||||||
|
if (_ctr.text.isNotEmpty) {
|
||||||
|
_onSendMsg(_ctr.text);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
icon: Icon(Icons.send, color: _color),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
void _onSendMsg(msg) async {
|
void _onSendMsg(msg) async {
|
||||||
if (!_isLogin) {
|
if (!_isLogin) {
|
||||||
|
|||||||
Reference in New Issue
Block a user