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';
|
2024-05-03 13:39:52 +08:00
|
|
|
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';
|
2024-04-28 23:05:59 +08:00
|
|
|
import 'package:solian/widgets/chat/call/participant_menu.dart';
|
2024-05-03 13:39:52 +08:00
|
|
|
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
|
2024-04-28 23:05:59 +08:00
|
|
|
? InteractiveParticipantWidget(
|
2024-04-30 20:31:54 +08:00
|
|
|
isFixed: false,
|
|
|
|
participant: _call.focusTrack!,
|
2024-04-28 23:05:59 +08:00
|
|
|
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(
|
2024-04-28 23:05:59 +08:00
|
|
|
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),
|
2024-04-28 23:05:59 +08:00
|
|
|
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) {
|
2024-04-28 23:05:59 +08:00
|
|
|
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)),
|
2024-04-28 23:05:59 +08:00
|
|
|
child: InteractiveParticipantWidget(
|
2024-04-30 20:31:54 +08:00
|
|
|
isFixed: true,
|
2024-04-28 23:05:59 +08:00
|
|
|
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 23:05:59 +08:00
|
|
|
}
|
|
|
|
},
|
|
|
|
),
|
2024-04-28 00:21:16 +08:00
|
|
|
),
|
2024-04-28 23:05:59 +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
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-05-03 13:39:52 +08:00
|
|
|
return IndentScaffold(
|
2024-04-30 20:31:54 +08:00
|
|
|
title: AppLocalizations.of(context)!.chatCall,
|
2024-05-03 13:39:52 +08:00
|
|
|
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
|
|
|
}
|
2024-04-28 23:05:59 +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;
|
2024-04-28 23:05:59 +08:00
|
|
|
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,
|
2024-04-28 23:05:59 +08:00
|
|
|
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),
|
2024-04-28 23:05:59 +08:00
|
|
|
),
|
|
|
|
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,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
},
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|