|
|
|
@ -21,6 +21,7 @@ class ChatInputRow extends StatelessWidget {
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
final theme = Theme.of(context);
|
|
|
|
|
|
|
|
|
|
const height = 48.0;
|
|
|
|
|
|
|
|
|
|
if (!controller.room.otherPartyCanReceiveMessages) {
|
|
|
|
@ -39,231 +40,290 @@ class ChatInputRow extends StatelessWidget {
|
|
|
|
|
return Row(
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.end,
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
const SizedBox(width: 4),
|
|
|
|
|
AnimatedContainer(
|
|
|
|
|
duration: FluffyThemes.animationDuration,
|
|
|
|
|
curve: FluffyThemes.animationCurve,
|
|
|
|
|
width: controller.sendController.text.isNotEmpty ? 0 : height,
|
|
|
|
|
height: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
decoration: const BoxDecoration(),
|
|
|
|
|
clipBehavior: Clip.hardEdge,
|
|
|
|
|
child: PopupMenuButton<String>(
|
|
|
|
|
useRootNavigator: true,
|
|
|
|
|
enabled: !controller.selectMode,
|
|
|
|
|
icon: const Icon(Icons.add_circle_outline),
|
|
|
|
|
iconColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
onSelected: controller.onAddPopupMenuButtonSelected,
|
|
|
|
|
itemBuilder: (BuildContext context) => <PopupMenuEntry<String>>[
|
|
|
|
|
if (PlatformInfos.isMobile)
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'location',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.gps_fixed_outlined),
|
|
|
|
|
children: controller.selectMode
|
|
|
|
|
? <Widget>[
|
|
|
|
|
if (controller.selectedEvents
|
|
|
|
|
.every((event) => event.status == EventStatus.error))
|
|
|
|
|
SizedBox(
|
|
|
|
|
height: height,
|
|
|
|
|
child: TextButton(
|
|
|
|
|
style: TextButton.styleFrom(
|
|
|
|
|
foregroundColor: theme.colorScheme.error,
|
|
|
|
|
),
|
|
|
|
|
onPressed: controller.deleteErrorEventsAction,
|
|
|
|
|
child: Row(
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
const Icon(Icons.delete),
|
|
|
|
|
Text(L10n.of(context).delete),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).shareLocation),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'image',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.photo_outlined),
|
|
|
|
|
)
|
|
|
|
|
else
|
|
|
|
|
SizedBox(
|
|
|
|
|
height: height,
|
|
|
|
|
child: TextButton(
|
|
|
|
|
onPressed: controller.forwardEventsAction,
|
|
|
|
|
child: Row(
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
const Icon(Icons.keyboard_arrow_left_outlined),
|
|
|
|
|
Text(L10n.of(context).forward),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).sendImage),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'video',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.video_camera_back_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).sendVideo),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
controller.selectedEvents.length == 1
|
|
|
|
|
? controller.selectedEvents.first
|
|
|
|
|
.getDisplayEvent(controller.timeline!)
|
|
|
|
|
.status
|
|
|
|
|
.isSent
|
|
|
|
|
? SizedBox(
|
|
|
|
|
height: height,
|
|
|
|
|
child: TextButton(
|
|
|
|
|
onPressed: controller.replyAction,
|
|
|
|
|
child: Row(
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
Text(L10n.of(context).reply),
|
|
|
|
|
const Icon(Icons.keyboard_arrow_right),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
: SizedBox(
|
|
|
|
|
height: height,
|
|
|
|
|
child: TextButton(
|
|
|
|
|
onPressed: controller.sendAgainAction,
|
|
|
|
|
child: Row(
|
|
|
|
|
children: <Widget>[
|
|
|
|
|
Text(L10n.of(context).tryToSendAgain),
|
|
|
|
|
const SizedBox(width: 4),
|
|
|
|
|
const Icon(Icons.send_outlined, size: 16),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
)
|
|
|
|
|
: const SizedBox.shrink(),
|
|
|
|
|
]
|
|
|
|
|
: <Widget>[
|
|
|
|
|
const SizedBox(width: 4),
|
|
|
|
|
AnimatedContainer(
|
|
|
|
|
duration: FluffyThemes.animationDuration,
|
|
|
|
|
curve: FluffyThemes.animationCurve,
|
|
|
|
|
width: controller.sendController.text.isNotEmpty ? 0 : height,
|
|
|
|
|
height: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
decoration: const BoxDecoration(),
|
|
|
|
|
clipBehavior: Clip.hardEdge,
|
|
|
|
|
child: PopupMenuButton<String>(
|
|
|
|
|
useRootNavigator: true,
|
|
|
|
|
icon: const Icon(Icons.add_circle_outline),
|
|
|
|
|
iconColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
onSelected: controller.onAddPopupMenuButtonSelected,
|
|
|
|
|
itemBuilder: (BuildContext context) =>
|
|
|
|
|
<PopupMenuEntry<String>>[
|
|
|
|
|
if (PlatformInfos.isMobile)
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'location',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor:
|
|
|
|
|
theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.gps_fixed_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).shareLocation),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'image',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.photo_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).sendImage),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'video',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.video_camera_back_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).sendVideo),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'file',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.attachment_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).sendFile),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'file',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.attachment_outlined),
|
|
|
|
|
if (PlatformInfos.isMobile)
|
|
|
|
|
AnimatedContainer(
|
|
|
|
|
duration: FluffyThemes.animationDuration,
|
|
|
|
|
curve: FluffyThemes.animationCurve,
|
|
|
|
|
width: controller.sendController.text.isNotEmpty ? 0 : height,
|
|
|
|
|
height: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
decoration: const BoxDecoration(),
|
|
|
|
|
clipBehavior: Clip.hardEdge,
|
|
|
|
|
child: PopupMenuButton(
|
|
|
|
|
useRootNavigator: true,
|
|
|
|
|
icon: const Icon(Icons.camera_alt_outlined),
|
|
|
|
|
onSelected: controller.onAddPopupMenuButtonSelected,
|
|
|
|
|
iconColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
itemBuilder: (context) => [
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'camera-video',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor:
|
|
|
|
|
theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.videocam_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).recordAVideo),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'camera',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor:
|
|
|
|
|
theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.camera_alt_outlined),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).takeAPhoto),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).sendFile),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
if (PlatformInfos.isMobile)
|
|
|
|
|
AnimatedContainer(
|
|
|
|
|
duration: FluffyThemes.animationDuration,
|
|
|
|
|
curve: FluffyThemes.animationCurve,
|
|
|
|
|
width: controller.sendController.text.isNotEmpty ? 0 : height,
|
|
|
|
|
height: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
decoration: const BoxDecoration(),
|
|
|
|
|
clipBehavior: Clip.hardEdge,
|
|
|
|
|
child: PopupMenuButton(
|
|
|
|
|
enabled: !controller.selectMode,
|
|
|
|
|
useRootNavigator: true,
|
|
|
|
|
icon: const Icon(Icons.camera_alt_outlined),
|
|
|
|
|
onSelected: controller.onAddPopupMenuButtonSelected,
|
|
|
|
|
iconColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
itemBuilder: (context) => [
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'camera-video',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.videocam_outlined),
|
|
|
|
|
Container(
|
|
|
|
|
height: height,
|
|
|
|
|
width: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
child: IconButton(
|
|
|
|
|
tooltip: L10n.of(context).emojis,
|
|
|
|
|
color: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
icon: PageTransitionSwitcher(
|
|
|
|
|
transitionBuilder: (
|
|
|
|
|
Widget child,
|
|
|
|
|
Animation<double> primaryAnimation,
|
|
|
|
|
Animation<double> secondaryAnimation,
|
|
|
|
|
) {
|
|
|
|
|
return SharedAxisTransition(
|
|
|
|
|
animation: primaryAnimation,
|
|
|
|
|
secondaryAnimation: secondaryAnimation,
|
|
|
|
|
transitionType: SharedAxisTransitionType.scaled,
|
|
|
|
|
fillColor: Colors.transparent,
|
|
|
|
|
child: child,
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
child: Icon(
|
|
|
|
|
controller.showEmojiPicker
|
|
|
|
|
? Icons.keyboard
|
|
|
|
|
: Icons.add_reaction_outlined,
|
|
|
|
|
key: ValueKey(controller.showEmojiPicker),
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).recordAVideo),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
),
|
|
|
|
|
onPressed: controller.emojiPickerAction,
|
|
|
|
|
),
|
|
|
|
|
PopupMenuItem<String>(
|
|
|
|
|
value: 'camera',
|
|
|
|
|
child: ListTile(
|
|
|
|
|
leading: CircleAvatar(
|
|
|
|
|
backgroundColor: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
foregroundColor: theme.colorScheme.primaryContainer,
|
|
|
|
|
child: const Icon(Icons.camera_alt_outlined),
|
|
|
|
|
),
|
|
|
|
|
if (Matrix.of(context).isMultiAccount &&
|
|
|
|
|
Matrix.of(context).hasComplexBundles &&
|
|
|
|
|
Matrix.of(context).currentBundle!.length > 1)
|
|
|
|
|
Container(
|
|
|
|
|
width: height,
|
|
|
|
|
height: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
child: _ChatAccountPicker(controller),
|
|
|
|
|
),
|
|
|
|
|
Expanded(
|
|
|
|
|
child: Padding(
|
|
|
|
|
padding: const EdgeInsets.symmetric(vertical: 0.0),
|
|
|
|
|
child: InputBar(
|
|
|
|
|
room: controller.room,
|
|
|
|
|
minLines: 1,
|
|
|
|
|
maxLines: 8,
|
|
|
|
|
autofocus: !PlatformInfos.isMobile,
|
|
|
|
|
keyboardType: TextInputType.multiline,
|
|
|
|
|
textInputAction:
|
|
|
|
|
AppConfig.sendOnEnter == true && PlatformInfos.isMobile
|
|
|
|
|
? TextInputAction.send
|
|
|
|
|
: null,
|
|
|
|
|
onSubmitted: controller.onInputBarSubmitted,
|
|
|
|
|
onSubmitImage: controller.sendImageFromClipBoard,
|
|
|
|
|
focusNode: controller.inputFocus,
|
|
|
|
|
controller: controller.sendController,
|
|
|
|
|
decoration: InputDecoration(
|
|
|
|
|
contentPadding: const EdgeInsets.only(
|
|
|
|
|
left: 6.0,
|
|
|
|
|
right: 6.0,
|
|
|
|
|
bottom: 6.0,
|
|
|
|
|
top: 3.0,
|
|
|
|
|
),
|
|
|
|
|
hintText: L10n.of(context).writeAMessage,
|
|
|
|
|
hintMaxLines: 1,
|
|
|
|
|
border: InputBorder.none,
|
|
|
|
|
enabledBorder: InputBorder.none,
|
|
|
|
|
filled: false,
|
|
|
|
|
),
|
|
|
|
|
title: Text(L10n.of(context).takeAPhoto),
|
|
|
|
|
contentPadding: const EdgeInsets.all(0),
|
|
|
|
|
onChanged: controller.onInputBarChanged,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
Container(
|
|
|
|
|
height: height,
|
|
|
|
|
width: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
child: IconButton(
|
|
|
|
|
tooltip: L10n.of(context).emojis,
|
|
|
|
|
color: theme.colorScheme.onPrimaryContainer,
|
|
|
|
|
icon: PageTransitionSwitcher(
|
|
|
|
|
transitionBuilder: (
|
|
|
|
|
Widget child,
|
|
|
|
|
Animation<double> primaryAnimation,
|
|
|
|
|
Animation<double> secondaryAnimation,
|
|
|
|
|
) {
|
|
|
|
|
return SharedAxisTransition(
|
|
|
|
|
animation: primaryAnimation,
|
|
|
|
|
secondaryAnimation: secondaryAnimation,
|
|
|
|
|
transitionType: SharedAxisTransitionType.scaled,
|
|
|
|
|
fillColor: Colors.transparent,
|
|
|
|
|
child: child,
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
child: Icon(
|
|
|
|
|
controller.showEmojiPicker
|
|
|
|
|
? Icons.keyboard
|
|
|
|
|
: Icons.add_reaction_outlined,
|
|
|
|
|
key: ValueKey(controller.showEmojiPicker),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
onPressed:
|
|
|
|
|
controller.selectMode ? null : controller.emojiPickerAction,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
if (Matrix.of(context).isMultiAccount &&
|
|
|
|
|
Matrix.of(context).hasComplexBundles &&
|
|
|
|
|
Matrix.of(context).currentBundle!.length > 1)
|
|
|
|
|
Container(
|
|
|
|
|
width: height,
|
|
|
|
|
height: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
child: _ChatAccountPicker(controller),
|
|
|
|
|
),
|
|
|
|
|
Expanded(
|
|
|
|
|
child: Padding(
|
|
|
|
|
padding: const EdgeInsets.symmetric(vertical: 0.0),
|
|
|
|
|
child: InputBar(
|
|
|
|
|
room: controller.room,
|
|
|
|
|
minLines: 1,
|
|
|
|
|
readOnly: controller.selectMode,
|
|
|
|
|
maxLines: 8,
|
|
|
|
|
autofocus: !PlatformInfos.isMobile,
|
|
|
|
|
keyboardType: TextInputType.multiline,
|
|
|
|
|
textInputAction:
|
|
|
|
|
AppConfig.sendOnEnter == true && PlatformInfos.isMobile
|
|
|
|
|
? TextInputAction.send
|
|
|
|
|
: null,
|
|
|
|
|
onSubmitted: controller.onInputBarSubmitted,
|
|
|
|
|
onSubmitImage: controller.sendImageFromClipBoard,
|
|
|
|
|
focusNode: controller.inputFocus,
|
|
|
|
|
controller: controller.sendController,
|
|
|
|
|
decoration: InputDecoration(
|
|
|
|
|
contentPadding: const EdgeInsets.only(
|
|
|
|
|
left: 6.0,
|
|
|
|
|
right: 6.0,
|
|
|
|
|
bottom: 6.0,
|
|
|
|
|
top: 3.0,
|
|
|
|
|
),
|
|
|
|
|
hintText: L10n.of(context).writeAMessage,
|
|
|
|
|
hintMaxLines: 1,
|
|
|
|
|
border: InputBorder.none,
|
|
|
|
|
enabledBorder: InputBorder.none,
|
|
|
|
|
filled: false,
|
|
|
|
|
Container(
|
|
|
|
|
height: height,
|
|
|
|
|
width: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
child: PlatformInfos.platformCanRecord &&
|
|
|
|
|
controller.sendController.text.isEmpty
|
|
|
|
|
? FloatingActionButton.small(
|
|
|
|
|
tooltip: L10n.of(context).voiceMessage,
|
|
|
|
|
onPressed: controller.voiceMessageAction,
|
|
|
|
|
elevation: 0,
|
|
|
|
|
heroTag: null,
|
|
|
|
|
shape: RoundedRectangleBorder(
|
|
|
|
|
borderRadius: BorderRadius.circular(height),
|
|
|
|
|
),
|
|
|
|
|
backgroundColor: theme.bubbleColor,
|
|
|
|
|
foregroundColor: theme.onBubbleColor,
|
|
|
|
|
child: const Icon(Icons.mic_none_outlined),
|
|
|
|
|
)
|
|
|
|
|
: FloatingActionButton.small(
|
|
|
|
|
tooltip: L10n.of(context).send,
|
|
|
|
|
onPressed: controller.send,
|
|
|
|
|
elevation: 0,
|
|
|
|
|
heroTag: null,
|
|
|
|
|
shape: RoundedRectangleBorder(
|
|
|
|
|
borderRadius: BorderRadius.circular(height),
|
|
|
|
|
),
|
|
|
|
|
backgroundColor: theme.bubbleColor,
|
|
|
|
|
foregroundColor: theme.onBubbleColor,
|
|
|
|
|
child: const Icon(Icons.send_outlined),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
onChanged: controller.onInputBarChanged,
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
Opacity(
|
|
|
|
|
opacity: controller.selectMode ? 0.66 : 1,
|
|
|
|
|
child: Container(
|
|
|
|
|
height: height,
|
|
|
|
|
width: height,
|
|
|
|
|
alignment: Alignment.center,
|
|
|
|
|
child: PlatformInfos.platformCanRecord &&
|
|
|
|
|
controller.sendController.text.isEmpty
|
|
|
|
|
? FloatingActionButton.small(
|
|
|
|
|
tooltip: L10n.of(context).voiceMessage,
|
|
|
|
|
onPressed: controller.selectMode
|
|
|
|
|
? null
|
|
|
|
|
: controller.voiceMessageAction,
|
|
|
|
|
elevation: 0,
|
|
|
|
|
heroTag: null,
|
|
|
|
|
shape: RoundedRectangleBorder(
|
|
|
|
|
borderRadius: BorderRadius.circular(height),
|
|
|
|
|
),
|
|
|
|
|
backgroundColor: theme.bubbleColor,
|
|
|
|
|
foregroundColor: theme.onBubbleColor,
|
|
|
|
|
child: const Icon(Icons.mic_none_outlined),
|
|
|
|
|
)
|
|
|
|
|
: FloatingActionButton.small(
|
|
|
|
|
tooltip: L10n.of(context).send,
|
|
|
|
|
onPressed: controller.selectMode ? null : controller.send,
|
|
|
|
|
elevation: 0,
|
|
|
|
|
heroTag: null,
|
|
|
|
|
shape: RoundedRectangleBorder(
|
|
|
|
|
borderRadius: BorderRadius.circular(height),
|
|
|
|
|
),
|
|
|
|
|
backgroundColor: theme.bubbleColor,
|
|
|
|
|
foregroundColor: theme.onBubbleColor,
|
|
|
|
|
child: const Icon(Icons.send_outlined),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
],
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|