2020-01-03 17:23:40 +01:00
|
|
|
import 'dart:async';
|
|
|
|
import 'dart:io';
|
2020-02-16 15:57:50 +01:00
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
import 'package:famedlysdk/famedlysdk.dart';
|
2020-02-22 08:27:08 +01:00
|
|
|
import 'package:fluffychat/components/dialogs/simple_dialogs.dart';
|
2020-05-05 10:30:24 +02:00
|
|
|
import 'package:fluffychat/utils/firebase_controller.dart';
|
2020-01-01 19:10:13 +01:00
|
|
|
import 'package:flutter/foundation.dart';
|
|
|
|
import 'package:flutter/material.dart';
|
|
|
|
import 'package:localstorage/localstorage.dart';
|
2020-04-08 17:43:07 +02:00
|
|
|
import 'package:url_launcher/url_launcher.dart';
|
2020-01-01 19:10:13 +01:00
|
|
|
|
2020-05-07 07:52:40 +02:00
|
|
|
import '../l10n/l10n.dart';
|
2020-02-22 08:27:08 +01:00
|
|
|
import '../utils/beautify_string_extension.dart';
|
2020-02-16 15:57:50 +01:00
|
|
|
import '../utils/famedlysdk_store.dart';
|
2020-04-08 17:43:07 +02:00
|
|
|
import 'avatar.dart';
|
2020-05-17 15:28:27 +02:00
|
|
|
import '../views/key_verification.dart';
|
|
|
|
import '../utils/app_route.dart';
|
2020-02-16 15:57:50 +01:00
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
class Matrix extends StatefulWidget {
|
2020-04-08 17:43:07 +02:00
|
|
|
static const String callNamespace = 'chat.fluffy.jitsi_call';
|
2020-04-12 10:35:45 +02:00
|
|
|
static const String defaultHomeserver = 'tchncs.de';
|
2020-04-08 17:43:07 +02:00
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
final Widget child;
|
|
|
|
|
|
|
|
final String clientName;
|
|
|
|
|
|
|
|
final Client client;
|
|
|
|
|
2020-05-13 15:58:59 +02:00
|
|
|
final Store store;
|
|
|
|
|
|
|
|
Matrix({this.child, this.clientName, this.client, this.store, Key key})
|
|
|
|
: super(key: key);
|
2020-01-01 19:10:13 +01:00
|
|
|
|
|
|
|
@override
|
|
|
|
MatrixState createState() => MatrixState();
|
|
|
|
|
|
|
|
/// Returns the (nearest) Client instance of your application.
|
|
|
|
static MatrixState of(BuildContext context) {
|
2020-05-13 15:58:59 +02:00
|
|
|
var newState =
|
2020-01-01 19:10:13 +01:00
|
|
|
(context.dependOnInheritedWidgetOfExactType<_InheritedMatrix>()).data;
|
2020-05-05 10:30:24 +02:00
|
|
|
newState.context = FirebaseController.context = context;
|
2020-01-01 19:10:13 +01:00
|
|
|
return newState;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class MatrixState extends State<Matrix> {
|
|
|
|
Client client;
|
2020-05-13 15:58:59 +02:00
|
|
|
Store store;
|
|
|
|
@override
|
2020-01-01 19:10:13 +01:00
|
|
|
BuildContext context;
|
|
|
|
|
2020-04-09 09:51:52 +02:00
|
|
|
Map<String, dynamic> get shareContent => _shareContent;
|
|
|
|
set shareContent(Map<String, dynamic> content) {
|
|
|
|
_shareContent = content;
|
|
|
|
onShareContentChanged.add(_shareContent);
|
|
|
|
}
|
|
|
|
|
|
|
|
Map<String, dynamic> _shareContent;
|
|
|
|
|
|
|
|
final StreamController<Map<String, dynamic>> onShareContentChanged =
|
|
|
|
StreamController.broadcast();
|
2020-01-08 14:19:15 +01:00
|
|
|
|
|
|
|
String activeRoomId;
|
2020-04-03 20:24:25 +02:00
|
|
|
File wallpaper;
|
2020-05-09 13:36:41 +02:00
|
|
|
bool renderHtml = false;
|
2020-01-03 17:23:40 +01:00
|
|
|
|
2020-04-08 17:43:07 +02:00
|
|
|
String jitsiInstance = 'https://meet.jit.si/';
|
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
void clean() async {
|
|
|
|
if (!kIsWeb) return;
|
|
|
|
|
2020-05-13 15:58:59 +02:00
|
|
|
final storage = LocalStorage('LocalStorage');
|
2020-01-01 19:10:13 +01:00
|
|
|
await storage.ready;
|
2020-01-02 22:31:39 +01:00
|
|
|
await storage.deleteItem(widget.clientName);
|
2020-01-01 19:10:13 +01:00
|
|
|
}
|
|
|
|
|
2020-01-08 14:19:15 +01:00
|
|
|
void _initWithStore() async {
|
2020-05-13 15:58:59 +02:00
|
|
|
var initLoginState = client.onLoginStateChanged.stream.first;
|
|
|
|
client.database = await getDatabase(client, store);
|
|
|
|
client.connect();
|
2020-02-16 15:57:50 +01:00
|
|
|
if (await initLoginState == LoginState.logged && !kIsWeb) {
|
2020-05-05 10:30:24 +02:00
|
|
|
await FirebaseController.setupFirebase(
|
|
|
|
client,
|
|
|
|
widget.clientName,
|
|
|
|
);
|
2020-01-08 14:19:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-19 16:23:13 +01:00
|
|
|
Map<String, dynamic> getAuthByPassword(String password, String session) => {
|
2020-05-13 15:58:59 +02:00
|
|
|
'type': 'm.login.password',
|
|
|
|
'identifier': {
|
|
|
|
'type': 'm.id.user',
|
|
|
|
'user': client.userID,
|
2020-02-19 16:23:13 +01:00
|
|
|
},
|
2020-05-13 15:58:59 +02:00
|
|
|
'user': client.userID,
|
|
|
|
'password': password,
|
|
|
|
'session': session,
|
2020-02-19 16:23:13 +01:00
|
|
|
};
|
|
|
|
|
2020-02-22 08:27:08 +01:00
|
|
|
StreamSubscription onRoomKeyRequestSub;
|
2020-05-17 15:28:27 +02:00
|
|
|
StreamSubscription onKeyVerificationRequestSub;
|
2020-04-08 17:43:07 +02:00
|
|
|
StreamSubscription onJitsiCallSub;
|
|
|
|
|
|
|
|
void onJitsiCall(EventUpdate eventUpdate) {
|
|
|
|
final event = Event.fromJson(
|
|
|
|
eventUpdate.content, client.getRoomById(eventUpdate.roomID));
|
|
|
|
if (DateTime.now().millisecondsSinceEpoch -
|
|
|
|
event.time.millisecondsSinceEpoch >
|
|
|
|
1000 * 60 * 5) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
final senderName = event.sender.calcDisplayname();
|
|
|
|
final senderAvatar = event.sender.avatarUrl;
|
|
|
|
showDialog(
|
|
|
|
context: context,
|
|
|
|
builder: (context) => AlertDialog(
|
2020-05-07 07:52:40 +02:00
|
|
|
title: Text(L10n.of(context).videoCall),
|
2020-04-08 17:43:07 +02:00
|
|
|
content: Column(
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
children: <Widget>[
|
2020-04-09 10:16:38 +02:00
|
|
|
ListTile(
|
|
|
|
contentPadding: EdgeInsets.all(0),
|
|
|
|
leading: Avatar(senderAvatar, senderName),
|
|
|
|
title: Text(
|
|
|
|
senderName,
|
|
|
|
style: TextStyle(fontSize: 18),
|
|
|
|
),
|
|
|
|
subtitle:
|
|
|
|
event.room.isDirectChat ? null : Text(event.room.displayname),
|
|
|
|
),
|
2020-04-08 17:43:07 +02:00
|
|
|
Divider(),
|
|
|
|
Row(
|
|
|
|
children: <Widget>[
|
|
|
|
Spacer(),
|
|
|
|
FloatingActionButton(
|
|
|
|
backgroundColor: Colors.red,
|
|
|
|
child: Icon(Icons.phone_missed),
|
|
|
|
onPressed: () => Navigator.of(context).pop(),
|
|
|
|
),
|
|
|
|
Spacer(),
|
|
|
|
FloatingActionButton(
|
|
|
|
backgroundColor: Colors.green,
|
|
|
|
child: Icon(Icons.phone),
|
|
|
|
onPressed: () {
|
|
|
|
Navigator.of(context).pop();
|
|
|
|
launch(event.body);
|
|
|
|
},
|
|
|
|
),
|
|
|
|
Spacer(),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
2020-02-22 08:27:08 +01:00
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
@override
|
|
|
|
void initState() {
|
2020-05-13 15:58:59 +02:00
|
|
|
store = widget.store ?? Store();
|
2020-01-01 19:10:13 +01:00
|
|
|
if (widget.client == null) {
|
2020-05-13 15:58:59 +02:00
|
|
|
debugPrint('[Matrix] Init matrix client');
|
2020-04-26 18:15:48 +02:00
|
|
|
client = Client(widget.clientName, debug: false);
|
2020-04-08 17:43:07 +02:00
|
|
|
onJitsiCallSub ??= client.onEvent.stream
|
|
|
|
.where((e) =>
|
2020-04-09 10:21:13 +02:00
|
|
|
e.type == 'timeline' &&
|
2020-04-08 17:43:07 +02:00
|
|
|
e.eventType == 'm.room.message' &&
|
|
|
|
e.content['content']['msgtype'] == Matrix.callNamespace &&
|
|
|
|
e.content['sender'] != client.userID)
|
|
|
|
.listen(onJitsiCall);
|
2020-02-22 08:27:08 +01:00
|
|
|
onRoomKeyRequestSub ??=
|
|
|
|
client.onRoomKeyRequest.stream.listen((RoomKeyRequest request) async {
|
2020-05-13 15:58:59 +02:00
|
|
|
final room = request.room;
|
|
|
|
final sender = room.getUserByMXIDSync(request.sender);
|
2020-02-22 08:27:08 +01:00
|
|
|
if (await SimpleDialogs(context).askConfirmation(
|
2020-05-07 07:52:40 +02:00
|
|
|
titleText: L10n.of(context).requestToReadOlderMessages,
|
2020-02-22 08:27:08 +01:00
|
|
|
contentText:
|
2020-05-13 15:58:59 +02:00
|
|
|
'${sender.id}\n\n${L10n.of(context).device}:\n${request.requestingDevice.deviceId}\n\n${L10n.of(context).identity}:\n${request.requestingDevice.curve25519Key.beautified}',
|
2020-05-07 07:52:40 +02:00
|
|
|
confirmText: L10n.of(context).verify,
|
|
|
|
cancelText: L10n.of(context).deny,
|
2020-02-22 08:27:08 +01:00
|
|
|
)) {
|
|
|
|
await request.forwardKey();
|
|
|
|
}
|
|
|
|
});
|
2020-05-17 15:28:27 +02:00
|
|
|
onKeyVerificationRequestSub ??= client.onKeyVerificationRequest.stream.listen((KeyVerification request) async {
|
|
|
|
if (await SimpleDialogs(context).askConfirmation(
|
|
|
|
titleText: 'New verification request from ${request.userId} and device ${request.deviceId}',
|
|
|
|
contentText: 'Start verification?',
|
|
|
|
)) {
|
|
|
|
await request.acceptVerification();
|
|
|
|
await Navigator.of(context).push(
|
|
|
|
AppRoute.defaultRoute(
|
|
|
|
context,
|
|
|
|
KeyVerificationView(request: request),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
await request.rejectVerification();
|
|
|
|
}
|
|
|
|
});
|
2020-01-26 12:17:54 +01:00
|
|
|
_initWithStore();
|
2020-01-01 19:10:13 +01:00
|
|
|
} else {
|
|
|
|
client = widget.client;
|
2020-05-13 15:58:59 +02:00
|
|
|
client.connect();
|
2020-01-01 19:10:13 +01:00
|
|
|
}
|
2020-05-13 15:58:59 +02:00
|
|
|
if (store != null) {
|
|
|
|
store
|
|
|
|
.getItem('chat.fluffy.jitsi_instance')
|
2020-04-08 17:43:07 +02:00
|
|
|
.then((final instance) => jitsiInstance = instance ?? jitsiInstance);
|
2020-05-13 15:58:59 +02:00
|
|
|
store.getItem('chat.fluffy.wallpaper').then((final path) async {
|
2020-04-08 10:54:17 +02:00
|
|
|
if (path == null) return;
|
2020-04-03 20:24:25 +02:00
|
|
|
final file = File(path);
|
|
|
|
if (await file.exists()) {
|
|
|
|
wallpaper = file;
|
|
|
|
}
|
|
|
|
});
|
2020-05-13 15:58:59 +02:00
|
|
|
store.getItem('chat.fluffy.renderHtml').then((final render) async {
|
|
|
|
renderHtml = render == '1';
|
2020-05-09 13:36:41 +02:00
|
|
|
});
|
2020-04-03 20:24:25 +02:00
|
|
|
}
|
2020-01-01 19:10:13 +01:00
|
|
|
super.initState();
|
|
|
|
}
|
|
|
|
|
2020-01-03 17:23:40 +01:00
|
|
|
@override
|
|
|
|
void dispose() {
|
2020-02-22 08:27:08 +01:00
|
|
|
onRoomKeyRequestSub?.cancel();
|
2020-04-08 17:43:07 +02:00
|
|
|
onJitsiCallSub?.cancel();
|
2020-01-03 17:23:40 +01:00
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
|
2020-01-01 19:10:13 +01:00
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
return _InheritedMatrix(
|
|
|
|
data: this,
|
|
|
|
child: widget.child,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class _InheritedMatrix extends InheritedWidget {
|
|
|
|
final MatrixState data;
|
|
|
|
|
|
|
|
_InheritedMatrix({Key key, this.data, Widget child})
|
|
|
|
: super(key: key, child: child);
|
|
|
|
|
|
|
|
@override
|
|
|
|
bool updateShouldNotify(_InheritedMatrix old) {
|
2020-05-13 15:58:59 +02:00
|
|
|
var update = old.data.client.accessToken != data.client.accessToken ||
|
|
|
|
old.data.client.userID != data.client.userID ||
|
|
|
|
old.data.client.deviceID != data.client.deviceID ||
|
|
|
|
old.data.client.deviceName != data.client.deviceName ||
|
|
|
|
old.data.client.homeserver != data.client.homeserver;
|
2020-01-01 19:10:13 +01:00
|
|
|
return update;
|
|
|
|
}
|
|
|
|
}
|