Implement voice messages

This commit is contained in:
Christian Pauly 2020-03-15 11:27:51 +01:00
parent 62fd512871
commit 9c7b6883f7
9 changed files with 317 additions and 4 deletions

View file

@ -1,3 +1,17 @@
# Version 0.10.0 - 2020-03-??
### New features
- Voice messages
- New message bubble design
# Version 0.9.0 - 2020-03-13
### New features
- Improved design
- End2End encryption for normal messages (not yet files)
- Key sharing
- Device keys verification UI
### Fixes
- Minor bug fixes
# Version 0.8.2 - 2020-02-17 # Version 0.8.2 - 2020-02-17
### Fixes ### Fixes
- SpeedDial labels not visible in light mode - SpeedDial labels not visible in light mode

View file

@ -7,6 +7,7 @@
FlutterApplication and put your custom class here. --> FlutterApplication and put your custom class here. -->
<uses-permission android:name="android.permission.INTERNET" /> <uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.VIBRATE" /> <uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" /> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<application <application

View file

@ -0,0 +1,140 @@
import 'dart:async';
import 'dart:io';
import 'dart:typed_data';
import 'package:famedlysdk/famedlysdk.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter_sound/flutter_sound.dart';
import 'package:intl/intl.dart';
import 'matrix.dart';
class AudioPlayer extends StatefulWidget {
final Color color;
final MxContent content;
const AudioPlayer(this.content, {this.color = Colors.black, Key key})
: super(key: key);
@override
_AudioPlayerState createState() => _AudioPlayerState();
}
enum AudioPlayerStatus { NOT_DOWNLOADED, DOWNLOADING, DOWNLOADED }
class _AudioPlayerState extends State<AudioPlayer> {
AudioPlayerStatus status = AudioPlayerStatus.NOT_DOWNLOADED;
FlutterSound flutterSound = FlutterSound();
StreamSubscription soundSubscription;
static var httpClient = HttpClient();
Uint8List audioFile;
String statusText = "00:00";
double currentPosition = 0;
double maxPosition = 0;
@override
void dispose() {
if (flutterSound.audioState == t_AUDIO_STATE.IS_PLAYING) {
flutterSound.stopPlayer();
}
soundSubscription?.cancel();
super.dispose();
}
_downloadAction() async {
if (status != AudioPlayerStatus.NOT_DOWNLOADED) return;
setState(() => status = AudioPlayerStatus.DOWNLOADING);
String url = widget.content.getDownloadLink(Matrix.of(context).client);
var request = await httpClient.getUrl(Uri.parse(url));
var response = await request.close();
var bytes = await consolidateHttpClientResponseBytes(response);
setState(() {
audioFile = bytes;
status = AudioPlayerStatus.DOWNLOADED;
});
_playAction();
}
_playAction() async {
switch (flutterSound.audioState) {
case t_AUDIO_STATE.IS_PLAYING:
await flutterSound.pausePlayer();
break;
case t_AUDIO_STATE.IS_PAUSED:
await flutterSound.resumePlayer();
break;
case t_AUDIO_STATE.IS_RECORDING:
break;
case t_AUDIO_STATE.IS_STOPPED:
await flutterSound.startPlayerFromBuffer(
audioFile,
codec: t_CODEC.CODEC_AAC,
);
soundSubscription ??= flutterSound.onPlayerStateChanged.listen((e) {
if (e != null) {
DateTime date =
DateTime.fromMillisecondsSinceEpoch(e.currentPosition.toInt());
String txt = DateFormat('mm:ss', 'en_US').format(date);
this.setState(() {
maxPosition = e.duration;
currentPosition = e.currentPosition;
statusText = txt;
});
if (e.duration == e.currentPosition) {
soundSubscription
?.cancel()
?.then((f) => soundSubscription = null);
}
}
});
break;
}
}
@override
Widget build(BuildContext context) {
return Row(
mainAxisSize: MainAxisSize.min,
children: <Widget>[
Container(
width: 40,
child: status == AudioPlayerStatus.DOWNLOADING
? CircularProgressIndicator()
: IconButton(
icon: Icon(
flutterSound.audioState == t_AUDIO_STATE.IS_PLAYING
? Icons.pause
: Icons.play_arrow,
color: widget.color,
),
onPressed: () {
if (status == AudioPlayerStatus.DOWNLOADED) {
_playAction();
} else {
_downloadAction();
}
},
),
),
Slider(
value: currentPosition,
onChanged: (double position) =>
flutterSound.seekToPlayer(position.toInt()),
max: status == AudioPlayerStatus.DOWNLOADED ? maxPosition : 0,
min: 0,
),
Text(
statusText,
style: TextStyle(
color: widget.color,
),
),
],
);
}
}

View file

@ -0,0 +1,97 @@
import 'dart:async';
import 'package:fluffychat/i18n/i18n.dart';
import 'package:flutter/material.dart';
import 'package:flutter_sound/flutter_sound.dart';
import 'package:intl/intl.dart';
class RecordingDialog extends StatefulWidget {
final Function onFinished;
const RecordingDialog({this.onFinished, Key key}) : super(key: key);
@override
_RecordingDialogState createState() => _RecordingDialogState();
}
class _RecordingDialogState extends State<RecordingDialog> {
FlutterSound flutterSound = FlutterSound();
String time = "00:00:00";
StreamSubscription _recorderSubscription;
void startRecording() async {
await flutterSound.startRecorder(
codec: t_CODEC.CODEC_AAC,
);
_recorderSubscription = flutterSound.onRecorderStateChanged.listen((e) {
DateTime date =
DateTime.fromMillisecondsSinceEpoch(e.currentPosition.toInt());
setState(() => time = DateFormat('mm:ss:SS', 'en_US').format(date));
});
}
@override
void initState() {
super.initState();
startRecording();
}
@override
void dispose() {
if (flutterSound.isRecording) flutterSound.stopRecorder();
_recorderSubscription?.cancel();
super.dispose();
}
@override
Widget build(BuildContext context) {
return AlertDialog(
content: Row(
children: <Widget>[
CircleAvatar(
backgroundColor: Colors.red,
radius: 8,
),
SizedBox(width: 8),
Expanded(
child: Text(
"${I18n.of(context).recording}: $time",
style: TextStyle(
fontSize: 18,
),
),
),
],
),
actions: <Widget>[
FlatButton(
child: Text(
I18n.of(context).cancel.toUpperCase(),
style: TextStyle(
color: Theme.of(context).textTheme.body1.color.withAlpha(150),
),
),
onPressed: () => Navigator.of(context).pop(),
),
FlatButton(
child: Row(
children: <Widget>[
Text(I18n.of(context).send.toUpperCase()),
SizedBox(width: 4),
Icon(Icons.send, size: 15),
],
),
onPressed: () async {
await _recorderSubscription?.cancel();
final String result = await flutterSound.stopRecorder();
if (widget.onFinished != null) {
widget.onFinished(result);
}
Navigator.of(context).pop();
},
),
],
);
}
}

View file

@ -1,6 +1,7 @@
import 'package:bubble/bubble.dart'; import 'package:bubble/bubble.dart';
import 'package:cached_network_image/cached_network_image.dart'; import 'package:cached_network_image/cached_network_image.dart';
import 'package:famedlysdk/famedlysdk.dart'; import 'package:famedlysdk/famedlysdk.dart';
import 'package:fluffychat/components/audio_player.dart';
import 'package:fluffychat/i18n/i18n.dart'; import 'package:fluffychat/i18n/i18n.dart';
import 'package:fluffychat/utils/event_extension.dart'; import 'package:fluffychat/utils/event_extension.dart';
import 'package:fluffychat/views/image_viewer.dart'; import 'package:fluffychat/views/image_viewer.dart';
@ -59,6 +60,10 @@ class MessageContent extends StatelessWidget {
), ),
); );
case MessageTypes.Audio: case MessageTypes.Audio:
return AudioPlayer(
MxContent(event.content["url"]),
color: textColor,
);
case MessageTypes.Video: case MessageTypes.Video:
case MessageTypes.File: case MessageTypes.File:
return Container( return Container(

View file

@ -94,6 +94,8 @@ class I18n {
args: [username, targetName], args: [username, targetName],
); );
String get cancel => Intl.message("Cancel");
String changedTheChatAvatar(String username) => Intl.message( String changedTheChatAvatar(String username) => Intl.message(
"$username changed the chat avatar", "$username changed the chat avatar",
name: "changedTheChatAvatar", name: "changedTheChatAvatar",
@ -484,6 +486,8 @@ class I18n {
String get rejoin => Intl.message("Rejoin"); String get rejoin => Intl.message("Rejoin");
String get recording => Intl.message("Recording");
String redactedAnEvent(String username) => Intl.message( String redactedAnEvent(String username) => Intl.message(
"$username redacted an event", "$username redacted an event",
name: "redactedAnEvent", name: "redactedAnEvent",
@ -555,6 +559,8 @@ class I18n {
args: [username, count], args: [username, count],
); );
String get send => Intl.message("Send");
String get sendAMessage => Intl.message("Send a message"); String get sendAMessage => Intl.message("Send a message");
String get sendFile => Intl.message('Send file'); String get sendFile => Intl.message('Send file');
@ -714,6 +720,8 @@ class I18n {
String get visibilityOfTheChatHistory => String get visibilityOfTheChatHistory =>
Intl.message("Visibility of the chat history"); Intl.message("Visibility of the chat history");
String get voiceMessage => Intl.message("Voice message");
String get warningEncryptionInBeta => Intl.message( String get warningEncryptionInBeta => Intl.message(
"End to end encryption is currently in Beta! Use at your own risk!"); "End to end encryption is currently in Beta! Use at your own risk!");

View file

@ -5,6 +5,7 @@ import 'package:famedlysdk/famedlysdk.dart';
import 'package:file_picker/file_picker.dart'; import 'package:file_picker/file_picker.dart';
import 'package:fluffychat/components/adaptive_page_layout.dart'; import 'package:fluffychat/components/adaptive_page_layout.dart';
import 'package:fluffychat/components/chat_settings_popup_menu.dart'; import 'package:fluffychat/components/chat_settings_popup_menu.dart';
import 'package:fluffychat/components/dialogs/recording_dialog.dart';
import 'package:fluffychat/components/dialogs/simple_dialogs.dart'; import 'package:fluffychat/components/dialogs/simple_dialogs.dart';
import 'package:fluffychat/components/encryption_button.dart'; import 'package:fluffychat/components/encryption_button.dart';
import 'package:fluffychat/components/list_items/message.dart'; import 'package:fluffychat/components/list_items/message.dart';
@ -222,6 +223,22 @@ class _ChatState extends State<_Chat> {
); );
} }
void voiceMessageAction(BuildContext context) async {
String result;
await showDialog(
context: context,
builder: (context) => RecordingDialog(
onFinished: (r) => result = r,
));
if (result == null) return;
final File audioFile = File(result);
await Matrix.of(context).tryRequestWithLoadingDialog(
room.sendAudioEvent(
MatrixFile(bytes: audioFile.readAsBytesSync(), path: audioFile.path),
),
);
}
String _getSelectedEventString(BuildContext context) { String _getSelectedEventString(BuildContext context) {
String copyString = ""; String copyString = "";
for (Event event in selectedEvents) { for (Event event in selectedEvents) {
@ -565,6 +582,9 @@ class _ChatState extends State<_Chat> {
if (choice == "camera") { if (choice == "camera") {
openCameraAction(context); openCameraAction(context);
} }
if (choice == "voice") {
voiceMessageAction(context);
}
}, },
itemBuilder: (BuildContext context) => itemBuilder: (BuildContext context) =>
<PopupMenuEntry<String>>[ <PopupMenuEntry<String>>[
@ -607,6 +627,19 @@ class _ChatState extends State<_Chat> {
contentPadding: EdgeInsets.all(0), contentPadding: EdgeInsets.all(0),
), ),
), ),
PopupMenuItem<String>(
value: "voice",
child: ListTile(
leading: CircleAvatar(
backgroundColor: Colors.red,
foregroundColor: Colors.white,
child: Icon(Icons.mic),
),
title: Text(
I18n.of(context).voiceMessage),
contentPadding: EdgeInsets.all(0),
),
),
], ],
), ),
EncryptionButton(room), EncryptionButton(room),
@ -656,6 +689,13 @@ class _ChatState extends State<_Chat> {
), ),
), ),
), ),
if (!kIsWeb && inputText.isEmpty)
IconButton(
icon: Icon(Icons.mic),
onPressed: () =>
voiceMessageAction(context),
),
if (kIsWeb || inputText.isNotEmpty)
IconButton( IconButton(
icon: Icon(Icons.send), icon: Icon(Icons.send),
onPressed: () => send(), onPressed: () => send(),

View file

@ -195,6 +195,13 @@ packages:
url: "https://pub.dartlang.org" url: "https://pub.dartlang.org"
source: hosted source: hosted
version: "0.5.4" version: "0.5.4"
flutter_sound:
dependency: "direct main"
description:
name: flutter_sound
url: "https://pub.dartlang.org"
source: hosted
version: "2.1.1"
flutter_speed_dial: flutter_speed_dial:
dependency: "direct main" dependency: "direct main"
description: description:

View file

@ -52,6 +52,7 @@ dependencies:
flutter_svg: ^0.17.1 flutter_svg: ^0.17.1
flutter_slidable: ^0.5.4 flutter_slidable: ^0.5.4
photo_view: ^0.9.2 photo_view: ^0.9.2
flutter_sound: ^2.1.1
intl: ^0.16.0 intl: ^0.16.0
intl_translation: ^0.17.9 intl_translation: ^0.17.9