Solian/lib/screens/chat/call.dart

188 lines
5.7 KiB
Dart
Raw Normal View History

2024-04-27 01:36:54 +08:00
import 'package:flutter/material.dart';
2024-04-27 13:12:26 +08:00
import 'package:livekit_client/livekit_client.dart';
2024-04-27 01:36:54 +08:00
import 'package:provider/provider.dart';
import 'package:solian/models/call.dart';
2024-04-30 20:31:54 +08:00
import 'package:solian/providers/chat.dart';
import 'package:solian/utils/theme.dart';
import 'package:solian/widgets/chat/call/call_controls.dart';
2024-04-27 13:12:26 +08:00
import 'package:solian/widgets/chat/call/participant.dart';
import 'package:solian/widgets/chat/call/participant_menu.dart';
import 'package:solian/widgets/scaffold.dart';
2024-04-27 01:36:54 +08:00
import 'package:flutter_gen/gen_l10n/app_localizations.dart';
2024-04-27 13:12:26 +08:00
import 'dart:math' as math;
2024-04-27 01:36:54 +08:00
class ChatCall extends StatefulWidget {
final Call call;
const ChatCall({super.key, required this.call});
@override
State<ChatCall> createState() => _ChatCallState();
}
class _ChatCallState extends State<ChatCall> {
2024-04-30 20:31:54 +08:00
bool _isHandled = false;
2024-04-27 13:12:26 +08:00
2024-04-30 20:31:54 +08:00
late ChatProvider _chat;
2024-04-28 21:49:03 +08:00
2024-05-02 00:49:38 +08:00
ChatCallInstance get _call => _chat.currentCall!;
2024-04-27 13:12:26 +08:00
2024-04-30 20:31:54 +08:00
@override
void initState() {
super.initState();
2024-04-28 00:21:16 +08:00
2024-04-30 20:31:54 +08:00
WidgetsBinding.instance.addPostFrameCallback((_) {
2024-05-02 00:49:38 +08:00
_chat.setCallShown(true);
2024-04-27 13:12:26 +08:00
});
}
2024-04-30 20:31:54 +08:00
@override
Widget build(BuildContext context) {
_chat = context.watch<ChatProvider>();
if (!_isHandled) {
_isHandled = true;
2024-05-02 00:49:38 +08:00
if (_chat.handleCallJoin(widget.call, widget.call.channel)) {
_chat.currentCall?.init();
2024-04-27 13:12:26 +08:00
}
}
2024-04-30 20:31:54 +08:00
Widget content;
2024-05-02 00:49:38 +08:00
if (_chat.currentCall == null) {
2024-04-30 20:31:54 +08:00
content = const Center(
child: CircularProgressIndicator(),
);
2024-04-27 13:12:26 +08:00
} else {
2024-04-30 20:31:54 +08:00
content = FutureBuilder(
future: _call.exchangeToken(context),
2024-04-27 01:36:54 +08:00
builder: (context, snapshot) {
if (!snapshot.hasData || snapshot.data == null) {
return const Center(child: CircularProgressIndicator());
}
2024-04-27 13:12:26 +08:00
return Stack(
children: [
Column(
children: [
Expanded(
2024-04-28 00:07:32 +08:00
child: Container(
color: Theme.of(context).colorScheme.surfaceVariant,
2024-04-30 20:31:54 +08:00
child: _call.focusTrack != null
? InteractiveParticipantWidget(
2024-04-30 20:31:54 +08:00
isFixed: false,
participant: _call.focusTrack!,
onTap: () {},
)
2024-04-28 00:07:32 +08:00
: Container(),
),
2024-04-27 13:12:26 +08:00
),
2024-04-30 20:31:54 +08:00
if (_call.room.localParticipant != null)
ControlsWidget(
_call.room,
_call.room.localParticipant!,
),
2024-04-27 13:12:26 +08:00
],
),
Positioned(
left: 0,
right: 0,
top: 0,
child: SizedBox(
height: 128,
2024-04-27 13:12:26 +08:00
child: ListView.builder(
scrollDirection: Axis.horizontal,
2024-04-30 20:31:54 +08:00
itemCount: math.max(0, _call.participantTracks.length),
itemBuilder: (BuildContext context, int index) {
2024-04-30 20:31:54 +08:00
final track = _call.participantTracks[index];
2024-05-01 17:37:34 +08:00
if (track.participant.sid ==
_call.focusTrack?.participant.sid) {
return Container();
}
return Padding(
padding: const EdgeInsets.only(top: 8, left: 8),
child: ClipRRect(
2024-05-01 17:37:34 +08:00
borderRadius:
const BorderRadius.all(Radius.circular(8)),
child: InteractiveParticipantWidget(
2024-04-30 20:31:54 +08:00
isFixed: true,
width: 120,
height: 120,
color: Theme.of(context).cardColor,
participant: track,
onTap: () {
2024-05-01 17:37:34 +08:00
if (track.participant.sid !=
_call.focusTrack?.participant.sid) {
2024-04-30 20:31:54 +08:00
_call.changeFocusTrack(track);
}
},
),
2024-04-28 00:21:16 +08:00
),
);
},
2024-04-27 13:12:26 +08:00
),
),
),
],
);
2024-04-27 01:36:54 +08:00
},
2024-04-30 20:31:54 +08:00
);
}
return IndentScaffold(
2024-04-30 20:31:54 +08:00
title: AppLocalizations.of(context)!.chatCall,
fixedAppBarColor: SolianTheme.isLargeScreen(context),
2024-04-30 20:31:54 +08:00
hideDrawer: true,
2024-05-08 22:01:06 +08:00
body: content,
2024-04-27 01:36:54 +08:00
);
}
2024-04-27 13:12:26 +08:00
@override
void deactivate() {
2024-05-02 00:49:38 +08:00
WidgetsBinding.instance.addPostFrameCallback((_) => _chat.setCallShown(false));
2024-04-27 13:12:26 +08:00
super.deactivate();
}
2024-04-27 01:36:54 +08:00
}
class InteractiveParticipantWidget extends StatelessWidget {
final double? width;
final double? height;
final Color? color;
2024-04-30 20:37:08 +08:00
final bool isFixed;
final ParticipantTrack participant;
final Function() onTap;
const InteractiveParticipantWidget({
super.key,
this.width,
this.height,
this.color,
2024-04-29 20:22:06 +08:00
this.isFixed = false,
required this.participant,
required this.onTap,
});
@override
Widget build(BuildContext context) {
return InkWell(
child: Container(
width: width,
height: height,
color: color,
2024-04-30 20:37:08 +08:00
child: ParticipantWidget.widgetFor(participant, isFixed: isFixed),
),
onTap: () => onTap(),
onLongPress: () {
if (participant.participant is LocalParticipant) return;
showModalBottomSheet(
context: context,
builder: (context) => ParticipantMenu(
participant: participant.participant as RemoteParticipant,
videoTrack: participant.videoTrack,
isScreenShare: participant.isScreenShare,
),
);
},
);
}
}