Clean up everything
This commit is contained in:
parent
cccbd7557e
commit
f4ef4eeea9
|
@ -42,7 +42,9 @@ class Avatar extends StatelessWidget {
|
||||||
src,
|
src,
|
||||||
)
|
)
|
||||||
: null,
|
: null,
|
||||||
backgroundColor: name?.color ?? Theme.of(context).secondaryHeaderColor,
|
backgroundColor: mxContent.mxc.isEmpty
|
||||||
|
? name?.color ?? Theme.of(context).secondaryHeaderColor
|
||||||
|
: Theme.of(context).secondaryHeaderColor,
|
||||||
child: mxContent.mxc.isEmpty
|
child: mxContent.mxc.isEmpty
|
||||||
? Text(fallbackLetters, style: TextStyle(color: Colors.white))
|
? Text(fallbackLetters, style: TextStyle(color: Colors.white))
|
||||||
: null,
|
: null,
|
||||||
|
|
|
@ -12,11 +12,13 @@ class ContentBanner extends StatelessWidget {
|
||||||
final double height;
|
final double height;
|
||||||
final IconData defaultIcon;
|
final IconData defaultIcon;
|
||||||
final bool loading;
|
final bool loading;
|
||||||
|
final Function onEdit;
|
||||||
|
|
||||||
const ContentBanner(this.mxContent,
|
const ContentBanner(this.mxContent,
|
||||||
{this.height = 400,
|
{this.height = 400,
|
||||||
this.defaultIcon = Icons.people_outline,
|
this.defaultIcon = Icons.people_outline,
|
||||||
this.loading = false,
|
this.loading = false,
|
||||||
|
this.onEdit,
|
||||||
Key key})
|
Key key})
|
||||||
: super(key: key);
|
: super(key: key);
|
||||||
|
|
||||||
|
@ -42,7 +44,17 @@ class ContentBanner extends StatelessWidget {
|
||||||
: null,
|
: null,
|
||||||
child: Container(
|
child: Container(
|
||||||
height: 200,
|
height: 200,
|
||||||
|
alignment: Alignment.center,
|
||||||
|
decoration: BoxDecoration(
|
||||||
color: Theme.of(context).secondaryHeaderColor,
|
color: Theme.of(context).secondaryHeaderColor,
|
||||||
|
),
|
||||||
|
child: Stack(
|
||||||
|
children: <Widget>[
|
||||||
|
Positioned(
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
top: 0,
|
||||||
|
bottom: 0,
|
||||||
child: !loading
|
child: !loading
|
||||||
? mxContent.mxc?.isNotEmpty ?? false
|
? mxContent.mxc?.isNotEmpty ?? false
|
||||||
? kIsWeb
|
? kIsWeb
|
||||||
|
@ -55,13 +67,24 @@ class ContentBanner extends StatelessWidget {
|
||||||
imageUrl: src,
|
imageUrl: src,
|
||||||
height: 200,
|
height: 200,
|
||||||
fit: BoxFit.cover,
|
fit: BoxFit.cover,
|
||||||
placeholder: (c, s) =>
|
|
||||||
Center(child: CircularProgressIndicator()),
|
|
||||||
errorWidget: (c, s, o) => Icon(Icons.error, size: 200),
|
|
||||||
)
|
)
|
||||||
: Icon(defaultIcon, size: 200)
|
: Icon(defaultIcon, size: 200)
|
||||||
: Icon(defaultIcon, size: 200),
|
: Icon(defaultIcon, size: 200),
|
||||||
),
|
),
|
||||||
|
if (this.onEdit != null)
|
||||||
|
Container(
|
||||||
|
margin: EdgeInsets.all(8),
|
||||||
|
alignment: Alignment.bottomRight,
|
||||||
|
child: FloatingActionButton(
|
||||||
|
mini: true,
|
||||||
|
backgroundColor: Theme.of(context).primaryColor,
|
||||||
|
child: Icon(Icons.file_upload),
|
||||||
|
onPressed: onEdit,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import 'package:famedlysdk/famedlysdk.dart';
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
import 'package:fluffychat/components/message_content.dart';
|
import 'package:fluffychat/utils/event_extension.dart';
|
||||||
import 'package:fluffychat/utils/date_time_extension.dart';
|
import 'package:fluffychat/utils/date_time_extension.dart';
|
||||||
import 'package:fluffychat/utils/app_route.dart';
|
import 'package:fluffychat/utils/app_route.dart';
|
||||||
import 'package:fluffychat/utils/room_name_calculator.dart';
|
import 'package:fluffychat/utils/room_extension.dart';
|
||||||
import 'package:fluffychat/views/chat.dart';
|
import 'package:fluffychat/views/chat.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:toast/toast.dart';
|
import 'package:toast/toast.dart';
|
||||||
|
@ -91,7 +91,7 @@ class ChatListItem extends StatelessWidget {
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
Expanded(
|
Expanded(
|
||||||
child: Text(
|
child: Text(
|
||||||
RoomNameCalculator(room).name,
|
room.getLocalizedDisplayname(context),
|
||||||
maxLines: 1,
|
maxLines: 1,
|
||||||
overflow: TextOverflow.ellipsis,
|
overflow: TextOverflow.ellipsis,
|
||||||
),
|
),
|
||||||
|
@ -117,9 +117,16 @@ class ChatListItem extends StatelessWidget {
|
||||||
color: Theme.of(context).primaryColor,
|
color: Theme.of(context).primaryColor,
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
: MessageContent(
|
: Text(
|
||||||
room.lastEvent,
|
room.lastEvent.getLocalizedBody(context,
|
||||||
textOnly: true,
|
withSenderNamePrefix: true, hideQuotes: true),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
style: TextStyle(
|
||||||
|
decoration: room.lastEvent.redacted
|
||||||
|
? TextDecoration.lineThrough
|
||||||
|
: null,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
SizedBox(width: 8),
|
SizedBox(width: 8),
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
import 'package:bubble/bubble.dart';
|
import 'package:bubble/bubble.dart';
|
||||||
import 'package:famedlysdk/famedlysdk.dart';
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:fluffychat/utils/event_extension.dart';
|
||||||
import '../message_content.dart';
|
import 'package:link_text/link_text.dart';
|
||||||
|
|
||||||
class StateMessage extends StatelessWidget {
|
class StateMessage extends StatelessWidget {
|
||||||
final Event event;
|
final Event event;
|
||||||
|
@ -23,7 +23,13 @@ class StateMessage extends StatelessWidget {
|
||||||
color: Colors.black,
|
color: Colors.black,
|
||||||
elevation: 0,
|
elevation: 0,
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
child: MessageContent(event, textColor: Colors.white),
|
child: LinkText(
|
||||||
|
text: event.getLocalizedBody(context),
|
||||||
|
textStyle: TextStyle(
|
||||||
|
color: Colors.white,
|
||||||
|
decoration: event.redacted ? TextDecoration.lineThrough : null,
|
||||||
|
),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
|
@ -4,7 +4,8 @@ import 'dart:io';
|
||||||
import 'package:firebase_messaging/firebase_messaging.dart';
|
import 'package:firebase_messaging/firebase_messaging.dart';
|
||||||
import 'package:famedlysdk/famedlysdk.dart';
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
import 'package:fluffychat/utils/app_route.dart';
|
import 'package:fluffychat/utils/app_route.dart';
|
||||||
import 'package:fluffychat/utils/room_name_calculator.dart';
|
import 'package:fluffychat/utils/event_extension.dart';
|
||||||
|
import 'package:fluffychat/utils/room_extension.dart';
|
||||||
import 'package:fluffychat/utils/sqflite_store.dart';
|
import 'package:fluffychat/utils/sqflite_store.dart';
|
||||||
import 'package:fluffychat/views/chat.dart';
|
import 'package:fluffychat/views/chat.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
|
@ -264,28 +265,12 @@ class MatrixState extends State<Matrix> {
|
||||||
: "$unreadEvents unread messages";
|
: "$unreadEvents unread messages";
|
||||||
|
|
||||||
// Calculate the body
|
// Calculate the body
|
||||||
String body;
|
final String body = event.getLocalizedBody(context,
|
||||||
switch (event.messageType) {
|
withSenderNamePrefix: true, hideQuotes: true);
|
||||||
case MessageTypes.Image:
|
|
||||||
body = "${event.sender.calcDisplayname()} sent a picture";
|
|
||||||
break;
|
|
||||||
case MessageTypes.File:
|
|
||||||
body = "${event.sender.calcDisplayname()} sent a file";
|
|
||||||
break;
|
|
||||||
case MessageTypes.Audio:
|
|
||||||
body = "${event.sender.calcDisplayname()} sent an audio";
|
|
||||||
break;
|
|
||||||
case MessageTypes.Video:
|
|
||||||
body = "${event.sender.calcDisplayname()} sent a video";
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
body = "${event.sender.calcDisplayname()}: ${event.body}";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// The person object for the android message style notification
|
// The person object for the android message style notification
|
||||||
final person = Person(
|
final person = Person(
|
||||||
name: RoomNameCalculator(room).name,
|
name: room.getLocalizedDisplayname(context),
|
||||||
icon: room.avatar.mxc.isEmpty
|
icon: room.avatar.mxc.isEmpty
|
||||||
? null
|
? null
|
||||||
: await downloadAndSaveContent(
|
: await downloadAndSaveContent(
|
||||||
|
@ -320,7 +305,10 @@ class MatrixState extends State<Matrix> {
|
||||||
var platformChannelSpecifics = NotificationDetails(
|
var platformChannelSpecifics = NotificationDetails(
|
||||||
androidPlatformChannelSpecifics, iOSPlatformChannelSpecifics);
|
androidPlatformChannelSpecifics, iOSPlatformChannelSpecifics);
|
||||||
await _flutterLocalNotificationsPlugin.show(
|
await _flutterLocalNotificationsPlugin.show(
|
||||||
0, RoomNameCalculator(room).name, body, platformChannelSpecifics,
|
0,
|
||||||
|
room.getLocalizedDisplayname(context),
|
||||||
|
body,
|
||||||
|
platformChannelSpecifics,
|
||||||
payload: roomId);
|
payload: roomId);
|
||||||
} catch (exception) {
|
} catch (exception) {
|
||||||
print("[Push] Error while processing notification: " +
|
print("[Push] Error while processing notification: " +
|
||||||
|
@ -356,7 +344,7 @@ class MatrixState extends State<Matrix> {
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
if (widget.client == null) {
|
if (widget.client == null) {
|
||||||
client = Client(widget.clientName, debug: false);
|
client = Client(widget.clientName, debug: true);
|
||||||
if (!kIsWeb) {
|
if (!kIsWeb) {
|
||||||
_initWithStore();
|
_initWithStore();
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -2,6 +2,7 @@ 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/utils/app_route.dart';
|
import 'package:fluffychat/utils/app_route.dart';
|
||||||
|
import 'package:fluffychat/utils/event_extension.dart';
|
||||||
import 'package:fluffychat/views/content_web_view.dart';
|
import 'package:fluffychat/views/content_web_view.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
@ -13,41 +14,17 @@ import 'matrix.dart';
|
||||||
class MessageContent extends StatelessWidget {
|
class MessageContent extends StatelessWidget {
|
||||||
final Event event;
|
final Event event;
|
||||||
final Color textColor;
|
final Color textColor;
|
||||||
final bool textOnly;
|
|
||||||
|
|
||||||
const MessageContent(this.event, {this.textColor, this.textOnly = false});
|
const MessageContent(this.event, {this.textColor});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final int maxLines = textOnly ? 1 : null;
|
|
||||||
|
|
||||||
final Widget unknown = Text(
|
|
||||||
"${event.sender.calcDisplayname()} sent a ${event.typeKey} event",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration: event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
switch (event.type) {
|
switch (event.type) {
|
||||||
case EventTypes.Message:
|
case EventTypes.Message:
|
||||||
case EventTypes.Sticker:
|
case EventTypes.Sticker:
|
||||||
switch (event.messageType) {
|
switch (event.messageType) {
|
||||||
case MessageTypes.Image:
|
case MessageTypes.Image:
|
||||||
case MessageTypes.Sticker:
|
case MessageTypes.Sticker:
|
||||||
if (textOnly) {
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} sent a picture",
|
|
||||||
maxLines: maxLines,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration:
|
|
||||||
event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
final int size = 400;
|
final int size = 400;
|
||||||
final String src = MxContent(event.content["url"]).getThumbnail(
|
final String src = MxContent(event.content["url"]).getThumbnail(
|
||||||
Matrix.of(context).client,
|
Matrix.of(context).client,
|
||||||
|
@ -78,17 +55,6 @@ class MessageContent extends StatelessWidget {
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
case MessageTypes.Audio:
|
case MessageTypes.Audio:
|
||||||
if (textOnly) {
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} sent an audio message",
|
|
||||||
maxLines: maxLines,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration:
|
|
||||||
event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return Container(
|
return Container(
|
||||||
width: 200,
|
width: 200,
|
||||||
child: RaisedButton(
|
child: RaisedButton(
|
||||||
|
@ -114,17 +80,6 @@ class MessageContent extends StatelessWidget {
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
case MessageTypes.Video:
|
case MessageTypes.Video:
|
||||||
if (textOnly) {
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} sent a video message",
|
|
||||||
maxLines: maxLines,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration:
|
|
||||||
event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return Container(
|
return Container(
|
||||||
width: 200,
|
width: 200,
|
||||||
child: RaisedButton(
|
child: RaisedButton(
|
||||||
|
@ -150,17 +105,6 @@ class MessageContent extends StatelessWidget {
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
case MessageTypes.File:
|
case MessageTypes.File:
|
||||||
if (textOnly) {
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} sent a file",
|
|
||||||
maxLines: maxLines,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration:
|
|
||||||
event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return Container(
|
return Container(
|
||||||
width: 200,
|
width: 200,
|
||||||
child: RaisedButton(
|
child: RaisedButton(
|
||||||
|
@ -183,217 +127,29 @@ class MessageContent extends StatelessWidget {
|
||||||
case MessageTypes.Location:
|
case MessageTypes.Location:
|
||||||
case MessageTypes.None:
|
case MessageTypes.None:
|
||||||
case MessageTypes.Notice:
|
case MessageTypes.Notice:
|
||||||
final String senderPrefix =
|
|
||||||
textOnly && event.senderId != event.room.directChatMatrixID
|
|
||||||
? event.senderId == Matrix.of(context).client.userID
|
|
||||||
? "You: "
|
|
||||||
: "${event.sender.calcDisplayname()}: "
|
|
||||||
: "";
|
|
||||||
final String body = event.redacted
|
|
||||||
? "Redacted by ${event.redactedBecause.sender.calcDisplayname()}"
|
|
||||||
: senderPrefix + event.body;
|
|
||||||
if (textOnly) {
|
|
||||||
return Text(
|
|
||||||
body,
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration:
|
|
||||||
event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return LinkText(
|
|
||||||
text: body,
|
|
||||||
textStyle: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
decoration: event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
case MessageTypes.Emote:
|
case MessageTypes.Emote:
|
||||||
if (textOnly) {
|
|
||||||
return Text(
|
|
||||||
"* " + event.body,
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
decoration:
|
|
||||||
event.redacted ? TextDecoration.lineThrough : null,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return LinkText(
|
return LinkText(
|
||||||
text: "* " + event.body,
|
text: event.getLocalizedBody(context),
|
||||||
textStyle: TextStyle(
|
textStyle: TextStyle(
|
||||||
color: textColor,
|
color: textColor,
|
||||||
fontStyle: FontStyle.italic,
|
|
||||||
decoration: event.redacted ? TextDecoration.lineThrough : null,
|
decoration: event.redacted ? TextDecoration.lineThrough : null,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return unknown;
|
|
||||||
case EventTypes.RoomCreate:
|
|
||||||
return Text(
|
return Text(
|
||||||
"${event.sender.calcDisplayname()} has created the chat",
|
event.getLocalizedBody(context),
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomAvatar:
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the chat avatar",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomName:
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the chat name to '${event.content['name']}'",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomMember: // Display what has changed
|
|
||||||
String text = "Failed to parse member event";
|
|
||||||
// Has the membership changed?
|
|
||||||
final String newMembership = event.content["membership"] ?? "";
|
|
||||||
final String oldMembership =
|
|
||||||
event.unsigned["prev_content"] is Map<String, dynamic>
|
|
||||||
? event.unsigned["prev_content"]["membership"] ?? ""
|
|
||||||
: "";
|
|
||||||
if (newMembership != oldMembership) {
|
|
||||||
if (oldMembership == "invite" && newMembership == "join") {
|
|
||||||
text =
|
|
||||||
"${event.stateKeyUser.calcDisplayname()} has accepted the invitation";
|
|
||||||
} else if (oldMembership == "leave" && newMembership == "join") {
|
|
||||||
text =
|
|
||||||
"${event.stateKeyUser.calcDisplayname()} has joined the chat";
|
|
||||||
} else if (oldMembership == "join" && newMembership == "ban") {
|
|
||||||
text =
|
|
||||||
"${event.sender.calcDisplayname()} has kicked and banned ${event.stateKeyUser.calcDisplayname()}";
|
|
||||||
} else if (oldMembership == "join" &&
|
|
||||||
newMembership == "leave" &&
|
|
||||||
event.stateKey != event.senderId) {
|
|
||||||
text =
|
|
||||||
"${event.sender.calcDisplayname()} has kicked ${event.stateKeyUser.calcDisplayname()}";
|
|
||||||
} else if (oldMembership == "join" &&
|
|
||||||
newMembership == "leave" &&
|
|
||||||
event.stateKey == event.senderId) {
|
|
||||||
text = "${event.stateKeyUser.calcDisplayname()} has left the room";
|
|
||||||
} else if (oldMembership == "invite" && newMembership == "ban") {
|
|
||||||
text =
|
|
||||||
"${event.sender.calcDisplayname()} has banned ${event.stateKeyUser.calcDisplayname()}";
|
|
||||||
} else if (oldMembership == "leave" && newMembership == "ban") {
|
|
||||||
text =
|
|
||||||
"${event.sender.calcDisplayname()} has banned ${event.stateKeyUser.calcDisplayname()}";
|
|
||||||
} else if (oldMembership == "ban" && newMembership == "leave") {
|
|
||||||
text =
|
|
||||||
"${event.sender.calcDisplayname()} has unbanned ${event.stateKeyUser.calcDisplayname()}";
|
|
||||||
} else if (newMembership == "invite") {
|
|
||||||
text =
|
|
||||||
"${event.sender.calcDisplayname()} has invited ${event.stateKeyUser.calcDisplayname()}";
|
|
||||||
} else if (newMembership == "join") {
|
|
||||||
text = "${event.stateKeyUser.calcDisplayname()} has joined";
|
|
||||||
}
|
|
||||||
} else if (newMembership == "join") {
|
|
||||||
final String newAvatar = event.content["avatar_url"] ?? "";
|
|
||||||
final String oldAvatar =
|
|
||||||
event.unsigned["prev_content"] is Map<String, dynamic>
|
|
||||||
? event.unsigned["prev_content"]["avatar_url"] ?? ""
|
|
||||||
: "";
|
|
||||||
|
|
||||||
final String newDisplayname = event.content["displayname"] ?? "";
|
|
||||||
final String oldDisplayname =
|
|
||||||
event.unsigned["prev_content"] is Map<String, dynamic>
|
|
||||||
? event.unsigned["prev_content"]["displayname"] ?? ""
|
|
||||||
: "";
|
|
||||||
|
|
||||||
// Has the user avatar changed?
|
|
||||||
if (newAvatar != oldAvatar) {
|
|
||||||
text =
|
|
||||||
"${event.stateKeyUser.calcDisplayname()} has changed the profile avatar";
|
|
||||||
}
|
|
||||||
// Has the user avatar changed?
|
|
||||||
else if (newDisplayname != oldDisplayname) {
|
|
||||||
text =
|
|
||||||
"${event.stateKeyUser.calcDisplayname()} has changed the displayname to '$newDisplayname'";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return Text(
|
|
||||||
text,
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomTopic:
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the chat topic to '${event.content['topic']}'",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomPowerLevels:
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the power levels of the chat",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.HistoryVisibility:
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the history visibility of the chat to '${event.content['history_visibility']}'",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomJoinRules:
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the join rules of the chat to '${event.content['join_rule']}'",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
case EventTypes.RoomCanonicalAlias:
|
|
||||||
if (event.content['canonical_alias']?.isEmpty ?? true) {
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has removed the canonical alias.",
|
|
||||||
maxLines: maxLines,
|
|
||||||
style: TextStyle(
|
|
||||||
color: textColor,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return Text(
|
|
||||||
"${event.sender.calcDisplayname()} has changed the canonical alias to: ${event.content['canonical_alias']}",
|
|
||||||
maxLines: maxLines,
|
|
||||||
overflow: textOnly ? TextOverflow.ellipsis : null,
|
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: textColor,
|
color: textColor,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
default:
|
default:
|
||||||
return unknown;
|
return Text(
|
||||||
|
"${event.sender.calcDisplayname()} sent a ${event.typeKey} event",
|
||||||
|
style: TextStyle(
|
||||||
|
color: textColor,
|
||||||
|
decoration: event.redacted ? TextDecoration.lineThrough : null,
|
||||||
|
),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
224
lib/utils/event_extension.dart
Normal file
224
lib/utils/event_extension.dart
Normal file
|
@ -0,0 +1,224 @@
|
||||||
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'room_state_enums_extensions.dart';
|
||||||
|
|
||||||
|
extension LocalizedBody on Event {
|
||||||
|
static Set<MessageTypes> textOnlyMessageTypes = {
|
||||||
|
MessageTypes.Text,
|
||||||
|
MessageTypes.Reply,
|
||||||
|
MessageTypes.Notice,
|
||||||
|
MessageTypes.Emote,
|
||||||
|
MessageTypes.None,
|
||||||
|
};
|
||||||
|
|
||||||
|
getLocalizedBody(BuildContext context,
|
||||||
|
{bool withSenderNamePrefix = false, hideQuotes = false}) {
|
||||||
|
if (this.redacted) {
|
||||||
|
return "Redacted by ${this.redactedBecause.sender.calcDisplayname()}";
|
||||||
|
}
|
||||||
|
String localizedBody = body;
|
||||||
|
final String senderName = this.sender.calcDisplayname();
|
||||||
|
switch (this.type) {
|
||||||
|
case EventTypes.Sticker:
|
||||||
|
localizedBody = "$senderName sent a sticker";
|
||||||
|
break;
|
||||||
|
case EventTypes.Redaction:
|
||||||
|
localizedBody = "$senderName redacted an event";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomAliases:
|
||||||
|
localizedBody = "$senderName changed the room aliases";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomCanonicalAlias:
|
||||||
|
localizedBody = "$senderName changed the room invite link";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomCreate:
|
||||||
|
localizedBody = "$senderName created the room";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomJoinRules:
|
||||||
|
JoinRules joinRules = JoinRules.values.firstWhere(
|
||||||
|
(r) =>
|
||||||
|
r.toString().replaceAll("JoinRules.", "") ==
|
||||||
|
content["join_rule"],
|
||||||
|
orElse: () => null);
|
||||||
|
if (joinRules == null) {
|
||||||
|
localizedBody = "$senderName changed the join rules";
|
||||||
|
} else {
|
||||||
|
localizedBody =
|
||||||
|
"$senderName changed the join rules to: ${joinRules.getLocalizedString(context)}";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomMember:
|
||||||
|
String text = "Failed to parse member event";
|
||||||
|
final String targetName = this.stateKeyUser.calcDisplayname();
|
||||||
|
// Has the membership changed?
|
||||||
|
final String newMembership = this.content["membership"] ?? "";
|
||||||
|
final String oldMembership =
|
||||||
|
this.unsigned["prev_content"] is Map<String, dynamic>
|
||||||
|
? this.unsigned["prev_content"]["membership"] ?? ""
|
||||||
|
: "";
|
||||||
|
if (newMembership != oldMembership) {
|
||||||
|
if (oldMembership == "invite" && newMembership == "join") {
|
||||||
|
text = "$targetName has accepted the invitation";
|
||||||
|
} else if (oldMembership == "leave" && newMembership == "join") {
|
||||||
|
text = "$targetName has joined the chat";
|
||||||
|
} else if (oldMembership == "join" && newMembership == "ban") {
|
||||||
|
text = "$senderName has kicked and banned $targetName";
|
||||||
|
} else if (oldMembership == "join" &&
|
||||||
|
newMembership == "leave" &&
|
||||||
|
this.stateKey != this.senderId) {
|
||||||
|
text = "$senderName has kicked $targetName";
|
||||||
|
} else if (oldMembership == "join" &&
|
||||||
|
newMembership == "leave" &&
|
||||||
|
this.stateKey == this.senderId) {
|
||||||
|
text = "$senderName has left the room";
|
||||||
|
} else if (oldMembership == "invite" && newMembership == "ban") {
|
||||||
|
text = "$senderName has banned $targetName";
|
||||||
|
} else if (oldMembership == "leave" && newMembership == "ban") {
|
||||||
|
text = "$senderName has banned $targetName";
|
||||||
|
} else if (oldMembership == "ban" && newMembership == "leave") {
|
||||||
|
text = "$senderName has unbanned $targetName";
|
||||||
|
} else if (newMembership == "invite") {
|
||||||
|
text = "$senderName has invited $targetName";
|
||||||
|
} else if (newMembership == "join") {
|
||||||
|
text = "$targetName has joined";
|
||||||
|
}
|
||||||
|
} else if (newMembership == "join") {
|
||||||
|
final String newAvatar = this.content["avatar_url"] ?? "";
|
||||||
|
final String oldAvatar =
|
||||||
|
this.unsigned["prev_content"] is Map<String, dynamic>
|
||||||
|
? this.unsigned["prev_content"]["avatar_url"] ?? ""
|
||||||
|
: "";
|
||||||
|
|
||||||
|
final String newDisplayname = this.content["displayname"] ?? "";
|
||||||
|
final String oldDisplayname =
|
||||||
|
this.unsigned["prev_content"] is Map<String, dynamic>
|
||||||
|
? this.unsigned["prev_content"]["displayname"] ?? ""
|
||||||
|
: "";
|
||||||
|
|
||||||
|
// Has the user avatar changed?
|
||||||
|
if (newAvatar != oldAvatar) {
|
||||||
|
text = "$targetName has changed the profile avatar";
|
||||||
|
}
|
||||||
|
// Has the user avatar changed?
|
||||||
|
else if (newDisplayname != oldDisplayname) {
|
||||||
|
text =
|
||||||
|
"${this.stateKeyUser.id} has changed the displayname to '$newDisplayname'";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
localizedBody = text;
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomPowerLevels:
|
||||||
|
localizedBody = "$senderName changed the group permissions";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomName:
|
||||||
|
localizedBody =
|
||||||
|
"$senderName changed the group name to: '${content["name"]}'";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomTopic:
|
||||||
|
localizedBody =
|
||||||
|
"$senderName changed the group name to: '${content["topic"]}'";
|
||||||
|
break;
|
||||||
|
case EventTypes.RoomAvatar:
|
||||||
|
localizedBody = "$senderName changed the group avatar";
|
||||||
|
break;
|
||||||
|
case EventTypes.GuestAccess:
|
||||||
|
GuestAccess guestAccess = GuestAccess.values.firstWhere(
|
||||||
|
(r) =>
|
||||||
|
r.toString().replaceAll("GuestAccess.", "") ==
|
||||||
|
content["guest_access"],
|
||||||
|
orElse: () => null);
|
||||||
|
if (guestAccess == null) {
|
||||||
|
localizedBody = "$senderName changed the guest access rules";
|
||||||
|
} else {
|
||||||
|
localizedBody =
|
||||||
|
"$senderName changed the guest access rules to: ${guestAccess.getLocalizedString(context)}";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case EventTypes.HistoryVisibility:
|
||||||
|
HistoryVisibility historyVisibility = HistoryVisibility.values
|
||||||
|
.firstWhere(
|
||||||
|
(r) =>
|
||||||
|
r.toString().replaceAll("HistoryVisibility.", "") ==
|
||||||
|
content["history_visibility"],
|
||||||
|
orElse: () => null);
|
||||||
|
if (historyVisibility == null) {
|
||||||
|
localizedBody = "$senderName changed the history visibility";
|
||||||
|
} else {
|
||||||
|
localizedBody =
|
||||||
|
"$senderName changed the history visibility to: ${historyVisibility.getLocalizedString(context)}";
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case EventTypes.Encryption:
|
||||||
|
localizedBody = "$senderName activated end to end encryption";
|
||||||
|
break;
|
||||||
|
case EventTypes.Encrypted:
|
||||||
|
localizedBody = "Could not decrypt message";
|
||||||
|
break;
|
||||||
|
case EventTypes.CallInvite:
|
||||||
|
localizedBody = body;
|
||||||
|
break;
|
||||||
|
case EventTypes.CallAnswer:
|
||||||
|
localizedBody = body;
|
||||||
|
break;
|
||||||
|
case EventTypes.CallCandidates:
|
||||||
|
localizedBody = body;
|
||||||
|
break;
|
||||||
|
case EventTypes.CallHangup:
|
||||||
|
localizedBody = body;
|
||||||
|
break;
|
||||||
|
case EventTypes.Unknown:
|
||||||
|
localizedBody = body;
|
||||||
|
break;
|
||||||
|
case EventTypes.Message:
|
||||||
|
switch (this.messageType) {
|
||||||
|
case MessageTypes.Image:
|
||||||
|
localizedBody = "$senderName sent a picture";
|
||||||
|
break;
|
||||||
|
case MessageTypes.File:
|
||||||
|
localizedBody = "$senderName sent a file";
|
||||||
|
break;
|
||||||
|
case MessageTypes.Audio:
|
||||||
|
localizedBody = "$senderName sent an audio";
|
||||||
|
break;
|
||||||
|
case MessageTypes.Video:
|
||||||
|
localizedBody = "$senderName sent a video";
|
||||||
|
break;
|
||||||
|
case MessageTypes.Location:
|
||||||
|
localizedBody = "$senderName shared the location";
|
||||||
|
break;
|
||||||
|
case MessageTypes.Sticker:
|
||||||
|
localizedBody = "$senderName sent a sticker";
|
||||||
|
break;
|
||||||
|
case MessageTypes.Emote:
|
||||||
|
localizedBody = "* $body";
|
||||||
|
break;
|
||||||
|
case MessageTypes.Text:
|
||||||
|
case MessageTypes.Notice:
|
||||||
|
case MessageTypes.None:
|
||||||
|
case MessageTypes.Reply:
|
||||||
|
localizedBody = body;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
localizedBody =
|
||||||
|
"Unknown event '${this.type.toString().replaceAll("EventTypes.", "")}'";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the sender name prefix
|
||||||
|
if (withSenderNamePrefix &&
|
||||||
|
this.type == EventTypes.Message &&
|
||||||
|
textOnlyMessageTypes.contains(this.messageType)) {
|
||||||
|
localizedBody = "$senderName: $localizedBody";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hide quotes
|
||||||
|
if (hideQuotes) {
|
||||||
|
List<String> lines = localizedBody.split("\n");
|
||||||
|
lines.removeWhere((s) => s.startsWith("> "));
|
||||||
|
localizedBody = lines.join("\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
return localizedBody;
|
||||||
|
}
|
||||||
|
}
|
13
lib/utils/room_extension.dart
Normal file
13
lib/utils/room_extension.dart
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
extension LocalizedRoomDisplayname on Room {
|
||||||
|
String getLocalizedDisplayname(BuildContext context) {
|
||||||
|
if ((this.name?.isEmpty ?? true) &&
|
||||||
|
(this.canonicalAlias?.isEmpty ?? true) &&
|
||||||
|
!this.isDirectChat) {
|
||||||
|
return "Group with ${this.displayname}";
|
||||||
|
}
|
||||||
|
return this.displayname;
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,16 +0,0 @@
|
||||||
import 'package:famedlysdk/famedlysdk.dart';
|
|
||||||
|
|
||||||
class RoomNameCalculator {
|
|
||||||
final Room room;
|
|
||||||
|
|
||||||
const RoomNameCalculator(this.room);
|
|
||||||
|
|
||||||
String get name {
|
|
||||||
if ((room.name?.isEmpty ?? true) &&
|
|
||||||
(room.canonicalAlias?.isEmpty ?? true) &&
|
|
||||||
!room.isDirectChat) {
|
|
||||||
return "Group with ${room.displayname}";
|
|
||||||
}
|
|
||||||
return room.displayname;
|
|
||||||
}
|
|
||||||
}
|
|
45
lib/utils/room_state_enums_extensions.dart
Normal file
45
lib/utils/room_state_enums_extensions.dart
Normal file
|
@ -0,0 +1,45 @@
|
||||||
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
|
extension HistoryVisibilityDisplayString on HistoryVisibility {
|
||||||
|
String getLocalizedString(BuildContext context) {
|
||||||
|
switch (this) {
|
||||||
|
case HistoryVisibility.invited:
|
||||||
|
return "From the invitation";
|
||||||
|
case HistoryVisibility.joined:
|
||||||
|
return "From joining";
|
||||||
|
case HistoryVisibility.shared:
|
||||||
|
return "Visible for all participants";
|
||||||
|
case HistoryVisibility.world_readable:
|
||||||
|
return "Visible for everyone";
|
||||||
|
default:
|
||||||
|
return this.toString().replaceAll("HistoryVisibility.", "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extension GuestAccessDisplayString on GuestAccess {
|
||||||
|
String getLocalizedString(BuildContext context) {
|
||||||
|
switch (this) {
|
||||||
|
case GuestAccess.can_join:
|
||||||
|
return "Guests can join";
|
||||||
|
case GuestAccess.forbidden:
|
||||||
|
return "Guests are forbidden";
|
||||||
|
default:
|
||||||
|
return this.toString().replaceAll("GuestAccess.", "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
extension JoinRulesDisplayString on JoinRules {
|
||||||
|
String getLocalizedString(BuildContext context) {
|
||||||
|
switch (this) {
|
||||||
|
case JoinRules.public:
|
||||||
|
return "Anyone can join";
|
||||||
|
case JoinRules.invite:
|
||||||
|
return "Invited users only";
|
||||||
|
default:
|
||||||
|
return this.toString().replaceAll("JoinRules.", "");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -7,7 +7,7 @@ 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/list_items/message.dart';
|
import 'package:fluffychat/components/list_items/message.dart';
|
||||||
import 'package:fluffychat/components/matrix.dart';
|
import 'package:fluffychat/components/matrix.dart';
|
||||||
import 'package:fluffychat/utils/room_name_calculator.dart';
|
import 'package:fluffychat/utils/room_extension.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:image_picker/image_picker.dart';
|
import 'package:image_picker/image_picker.dart';
|
||||||
|
@ -187,7 +187,7 @@ class _ChatState extends State<Chat> {
|
||||||
mainAxisSize: MainAxisSize.min,
|
mainAxisSize: MainAxisSize.min,
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
Text(RoomNameCalculator(room).name),
|
Text(room.getLocalizedDisplayname(context)),
|
||||||
AnimatedContainer(
|
AnimatedContainer(
|
||||||
duration: Duration(milliseconds: 500),
|
duration: Duration(milliseconds: 500),
|
||||||
height: typingText.isEmpty ? 0 : 20,
|
height: typingText.isEmpty ? 0 : 20,
|
||||||
|
@ -237,14 +237,14 @@ class _ChatState extends State<Chat> {
|
||||||
controller: _scrollController,
|
controller: _scrollController,
|
||||||
itemBuilder: (BuildContext context, int i) => i == 0
|
itemBuilder: (BuildContext context, int i) => i == 0
|
||||||
? AnimatedContainer(
|
? AnimatedContainer(
|
||||||
height: seenByText.isEmpty ? 0 : 36,
|
height: seenByText.isEmpty ? 0 : 24,
|
||||||
duration: seenByText.isEmpty
|
duration: seenByText.isEmpty
|
||||||
? Duration(milliseconds: 0)
|
? Duration(milliseconds: 0)
|
||||||
: Duration(milliseconds: 500),
|
: Duration(milliseconds: 500),
|
||||||
alignment: timeline.events.first.senderId ==
|
alignment: timeline.events.first.senderId ==
|
||||||
client.userID
|
client.userID
|
||||||
? Alignment.centerRight
|
? Alignment.topRight
|
||||||
: Alignment.centerLeft,
|
: Alignment.topLeft,
|
||||||
child: Text(
|
child: Text(
|
||||||
seenByText,
|
seenByText,
|
||||||
maxLines: 1,
|
maxLines: 1,
|
||||||
|
@ -253,7 +253,11 @@ class _ChatState extends State<Chat> {
|
||||||
color: Theme.of(context).primaryColor,
|
color: Theme.of(context).primaryColor,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
padding: EdgeInsets.all(8),
|
padding: EdgeInsets.only(
|
||||||
|
left: 8,
|
||||||
|
right: 8,
|
||||||
|
bottom: 8,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
: Message(timeline.events[i - 1],
|
: Message(timeline.events[i - 1],
|
||||||
nextEvent:
|
nextEvent:
|
||||||
|
|
|
@ -8,13 +8,14 @@ import 'package:fluffychat/components/content_banner.dart';
|
||||||
import 'package:fluffychat/components/list_items/participant_list_item.dart';
|
import 'package:fluffychat/components/list_items/participant_list_item.dart';
|
||||||
import 'package:fluffychat/components/matrix.dart';
|
import 'package:fluffychat/components/matrix.dart';
|
||||||
import 'package:fluffychat/utils/app_route.dart';
|
import 'package:fluffychat/utils/app_route.dart';
|
||||||
import 'package:fluffychat/utils/room_name_calculator.dart';
|
import 'package:fluffychat/utils/room_extension.dart';
|
||||||
|
import 'package:fluffychat/utils/room_state_enums_extensions.dart';
|
||||||
import 'package:fluffychat/views/chat_list.dart';
|
import 'package:fluffychat/views/chat_list.dart';
|
||||||
import 'package:fluffychat/views/invitation_selection.dart';
|
import 'package:fluffychat/views/invitation_selection.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:image_picker/image_picker.dart';
|
import 'package:image_picker/image_picker.dart';
|
||||||
import 'package:link_text/link_text.dart';
|
import 'package:link_text/link_text.dart';
|
||||||
|
import 'package:share/share.dart';
|
||||||
import 'package:toast/toast.dart';
|
import 'package:toast/toast.dart';
|
||||||
|
|
||||||
class ChatDetails extends StatefulWidget {
|
class ChatDetails extends StatefulWidget {
|
||||||
|
@ -43,6 +44,40 @@ class _ChatDetailsState extends State<ChatDetails> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setCanonicalAliasAction(context, s) async {
|
||||||
|
final String domain = widget.room.client.userID.split(":")[1];
|
||||||
|
final String canonicalAlias = "%23" + s + "%3A" + domain;
|
||||||
|
final Event aliasEvent = widget.room.getState("m.room.aliases", domain);
|
||||||
|
final List aliases =
|
||||||
|
aliasEvent != null ? aliasEvent.content["aliases"] ?? [] : [];
|
||||||
|
if (aliases.indexWhere((s) => s == canonicalAlias) == -1) {
|
||||||
|
List<String> newAliases = List.from(aliases);
|
||||||
|
newAliases.add(canonicalAlias);
|
||||||
|
final response = await Matrix.of(context).tryRequestWithLoadingDialog(
|
||||||
|
widget.room.client.jsonRequest(
|
||||||
|
type: HTTPType.GET,
|
||||||
|
action: "/client/r0/directory/room/$canonicalAlias",
|
||||||
|
),
|
||||||
|
);
|
||||||
|
if (response == false) {
|
||||||
|
final success = await Matrix.of(context).tryRequestWithLoadingDialog(
|
||||||
|
widget.room.client.jsonRequest(
|
||||||
|
type: HTTPType.PUT,
|
||||||
|
action: "/client/r0/directory/room/$canonicalAlias",
|
||||||
|
data: {"room_id": widget.room.id}),
|
||||||
|
);
|
||||||
|
if (success == false) return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await Matrix.of(context).tryRequestWithLoadingDialog(
|
||||||
|
widget.room.client.jsonRequest(
|
||||||
|
type: HTTPType.PUT,
|
||||||
|
action:
|
||||||
|
"/client/r0/rooms/${widget.room.id}/state/m.room.canonical_alias",
|
||||||
|
data: {"alias": "#$s:$domain"}),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
void setTopicAction(BuildContext context, String displayname) async {
|
void setTopicAction(BuildContext context, String displayname) async {
|
||||||
setState(() => topicEditMode = false);
|
setState(() => topicEditMode = false);
|
||||||
final MatrixState matrix = Matrix.of(context);
|
final MatrixState matrix = Matrix.of(context);
|
||||||
|
@ -117,8 +152,16 @@ class _ChatDetailsState extends State<ChatDetails> {
|
||||||
),
|
),
|
||||||
secondScaffold: Scaffold(
|
secondScaffold: Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(RoomNameCalculator(widget.room).name),
|
title: Text(widget.room.getLocalizedDisplayname(context)),
|
||||||
actions: <Widget>[ChatSettingsPopupMenu(widget.room, false)],
|
actions: <Widget>[
|
||||||
|
if (widget.room.canonicalAlias?.isNotEmpty ?? false)
|
||||||
|
IconButton(
|
||||||
|
icon: Icon(Icons.share),
|
||||||
|
onPressed: () => Share.share(
|
||||||
|
"https://matrix.to/#/${widget.room.canonicalAlias}"),
|
||||||
|
),
|
||||||
|
ChatSettingsPopupMenu(widget.room, false)
|
||||||
|
],
|
||||||
),
|
),
|
||||||
body: ListView.builder(
|
body: ListView.builder(
|
||||||
itemCount: members.length + 1 + (canRequestMoreMembers ? 1 : 0),
|
itemCount: members.length + 1 + (canRequestMoreMembers ? 1 : 0),
|
||||||
|
@ -126,28 +169,11 @@ class _ChatDetailsState extends State<ChatDetails> {
|
||||||
? Column(
|
? Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.stretch,
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
children: <Widget>[
|
children: <Widget>[
|
||||||
ContentBanner(widget.room.avatar),
|
ContentBanner(widget.room.avatar,
|
||||||
|
onEdit: widget.room.canSendEvent("m.room.avatar")
|
||||||
|
? () => setAvatarAction(context)
|
||||||
|
: null),
|
||||||
Divider(height: 1),
|
Divider(height: 1),
|
||||||
if (widget.room.canSendEvent("m.room.avatar") && !kIsWeb)
|
|
||||||
ListTile(
|
|
||||||
title: Text("Upload group avatar"),
|
|
||||||
leading: Icon(Icons.camera),
|
|
||||||
onTap: () => setAvatarAction(context),
|
|
||||||
),
|
|
||||||
if (widget.room.canSendEvent("m.room.name"))
|
|
||||||
ListTile(
|
|
||||||
leading: Icon(Icons.edit),
|
|
||||||
title: TextField(
|
|
||||||
textInputAction: TextInputAction.done,
|
|
||||||
onSubmitted: (s) => setDisplaynameAction(context, s),
|
|
||||||
decoration: InputDecoration(
|
|
||||||
border: InputBorder.none,
|
|
||||||
labelText: "Set group name",
|
|
||||||
labelStyle: TextStyle(color: Colors.black),
|
|
||||||
hintText: (RoomNameCalculator(widget.room).name),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
topicEditMode
|
topicEditMode
|
||||||
? ListTile(
|
? ListTile(
|
||||||
title: TextField(
|
title: TextField(
|
||||||
|
@ -168,6 +194,13 @@ class _ChatDetailsState extends State<ChatDetails> {
|
||||||
),
|
),
|
||||||
)
|
)
|
||||||
: ListTile(
|
: ListTile(
|
||||||
|
leading: widget.room.canSendEvent("m.room.topic")
|
||||||
|
? CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.grey,
|
||||||
|
child: Icon(Icons.edit),
|
||||||
|
)
|
||||||
|
: null,
|
||||||
title: Text("Group description:",
|
title: Text("Group description:",
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: Theme.of(context).primaryColor,
|
color: Theme.of(context).primaryColor,
|
||||||
|
@ -185,6 +218,173 @@ class _ChatDetailsState extends State<ChatDetails> {
|
||||||
? () => setState(() => topicEditMode = true)
|
? () => setState(() => topicEditMode = true)
|
||||||
: null,
|
: null,
|
||||||
),
|
),
|
||||||
|
Divider(thickness: 8),
|
||||||
|
ListTile(
|
||||||
|
title: Text(
|
||||||
|
"Settings",
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context).primaryColor,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (widget.room.canSendEvent("m.room.name"))
|
||||||
|
ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.grey,
|
||||||
|
child: Icon(Icons.people),
|
||||||
|
),
|
||||||
|
title: TextField(
|
||||||
|
textInputAction: TextInputAction.done,
|
||||||
|
onSubmitted: (s) => setDisplaynameAction(context, s),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
border: InputBorder.none,
|
||||||
|
labelText: "Change the name of the group",
|
||||||
|
labelStyle: TextStyle(color: Colors.black),
|
||||||
|
hintText:
|
||||||
|
widget.room.getLocalizedDisplayname(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (widget.room.canSendEvent("m.room.canonical_alias") &&
|
||||||
|
widget.room.joinRules == JoinRules.public)
|
||||||
|
ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.grey,
|
||||||
|
child: Icon(Icons.link),
|
||||||
|
),
|
||||||
|
title: TextField(
|
||||||
|
textInputAction: TextInputAction.done,
|
||||||
|
onSubmitted: (s) =>
|
||||||
|
setCanonicalAliasAction(context, s),
|
||||||
|
decoration: InputDecoration(
|
||||||
|
border: InputBorder.none,
|
||||||
|
labelText: "Set invitation link",
|
||||||
|
labelStyle: TextStyle(color: Colors.black),
|
||||||
|
hintText: widget.room.canonicalAlias
|
||||||
|
?.replaceAll("#", "") ??
|
||||||
|
"alias",
|
||||||
|
prefixText: "#",
|
||||||
|
suffixText: widget.room.client.userID.split(":")[1],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
PopupMenuButton(
|
||||||
|
child: ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.grey,
|
||||||
|
child: Icon(Icons.public)),
|
||||||
|
title: Text("Who is allowed to join this group"),
|
||||||
|
subtitle: Text(
|
||||||
|
widget.room.joinRules.getLocalizedString(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onSelected: (JoinRules joinRule) =>
|
||||||
|
Matrix.of(context).tryRequestWithLoadingDialog(
|
||||||
|
widget.room.setJoinRules(joinRule),
|
||||||
|
),
|
||||||
|
itemBuilder: (BuildContext context) =>
|
||||||
|
<PopupMenuEntry<JoinRules>>[
|
||||||
|
if (widget.room.canChangeJoinRules)
|
||||||
|
PopupMenuItem<JoinRules>(
|
||||||
|
value: JoinRules.public,
|
||||||
|
child: Text(
|
||||||
|
JoinRules.public.getLocalizedString(context)),
|
||||||
|
),
|
||||||
|
if (widget.room.canChangeJoinRules)
|
||||||
|
PopupMenuItem<JoinRules>(
|
||||||
|
value: JoinRules.invite,
|
||||||
|
child: Text(
|
||||||
|
JoinRules.invite.getLocalizedString(context)),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
PopupMenuButton(
|
||||||
|
child: ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.grey,
|
||||||
|
child: Icon(Icons.visibility),
|
||||||
|
),
|
||||||
|
title: Text("Visibility of the chat history"),
|
||||||
|
subtitle: Text(
|
||||||
|
widget.room.historyVisibility
|
||||||
|
.getLocalizedString(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onSelected: (HistoryVisibility historyVisibility) =>
|
||||||
|
Matrix.of(context).tryRequestWithLoadingDialog(
|
||||||
|
widget.room.setHistoryVisibility(historyVisibility),
|
||||||
|
),
|
||||||
|
itemBuilder: (BuildContext context) =>
|
||||||
|
<PopupMenuEntry<HistoryVisibility>>[
|
||||||
|
if (widget.room.canChangeHistoryVisibility)
|
||||||
|
PopupMenuItem<HistoryVisibility>(
|
||||||
|
value: HistoryVisibility.invited,
|
||||||
|
child: Text(HistoryVisibility.invited
|
||||||
|
.getLocalizedString(context)),
|
||||||
|
),
|
||||||
|
if (widget.room.canChangeHistoryVisibility)
|
||||||
|
PopupMenuItem<HistoryVisibility>(
|
||||||
|
value: HistoryVisibility.joined,
|
||||||
|
child: Text(HistoryVisibility.joined
|
||||||
|
.getLocalizedString(context)),
|
||||||
|
),
|
||||||
|
if (widget.room.canChangeHistoryVisibility)
|
||||||
|
PopupMenuItem<HistoryVisibility>(
|
||||||
|
value: HistoryVisibility.shared,
|
||||||
|
child: Text(HistoryVisibility.shared
|
||||||
|
.getLocalizedString(context)),
|
||||||
|
),
|
||||||
|
if (widget.room.canChangeHistoryVisibility)
|
||||||
|
PopupMenuItem<HistoryVisibility>(
|
||||||
|
value: HistoryVisibility.world_readable,
|
||||||
|
child: Text(HistoryVisibility.world_readable
|
||||||
|
.getLocalizedString(context)),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
if (widget.room.joinRules == JoinRules.public)
|
||||||
|
PopupMenuButton(
|
||||||
|
child: ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
backgroundColor: Colors.white,
|
||||||
|
foregroundColor: Colors.grey,
|
||||||
|
child: Icon(Icons.info_outline),
|
||||||
|
),
|
||||||
|
title: Text("Are guest users allowed to join"),
|
||||||
|
subtitle: Text(
|
||||||
|
widget.room.guestAccess.getLocalizedString(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
onSelected: (GuestAccess guestAccess) =>
|
||||||
|
Matrix.of(context).tryRequestWithLoadingDialog(
|
||||||
|
widget.room.setGuestAccess(guestAccess),
|
||||||
|
),
|
||||||
|
itemBuilder: (BuildContext context) =>
|
||||||
|
<PopupMenuEntry<GuestAccess>>[
|
||||||
|
if (widget.room.canChangeGuestAccess)
|
||||||
|
PopupMenuItem<GuestAccess>(
|
||||||
|
value: GuestAccess.can_join,
|
||||||
|
child: Text(
|
||||||
|
GuestAccess.can_join
|
||||||
|
.getLocalizedString(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (widget.room.canChangeGuestAccess)
|
||||||
|
PopupMenuItem<GuestAccess>(
|
||||||
|
value: GuestAccess.forbidden,
|
||||||
|
child: Text(
|
||||||
|
GuestAccess.forbidden
|
||||||
|
.getLocalizedString(context),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
Divider(thickness: 8),
|
||||||
ListTile(
|
ListTile(
|
||||||
title: Text(
|
title: Text(
|
||||||
"$actualMembersCount participant" +
|
"$actualMembersCount participant" +
|
||||||
|
|
|
@ -7,7 +7,6 @@ import 'package:fluffychat/components/matrix.dart';
|
||||||
import 'package:fluffychat/utils/app_route.dart';
|
import 'package:fluffychat/utils/app_route.dart';
|
||||||
import 'package:fluffychat/views/chat_list.dart';
|
import 'package:fluffychat/views/chat_list.dart';
|
||||||
import 'package:fluffychat/views/sign_up.dart';
|
import 'package:fluffychat/views/sign_up.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:image_picker/image_picker.dart';
|
import 'package:image_picker/image_picker.dart';
|
||||||
import 'package:toast/toast.dart';
|
import 'package:toast/toast.dart';
|
||||||
|
@ -44,11 +43,7 @@ class _SettingsState extends State<Settings> {
|
||||||
final MatrixState matrix = Matrix.of(context);
|
final MatrixState matrix = Matrix.of(context);
|
||||||
final Map<String, dynamic> success =
|
final Map<String, dynamic> success =
|
||||||
await matrix.tryRequestWithLoadingDialog(
|
await matrix.tryRequestWithLoadingDialog(
|
||||||
matrix.client.jsonRequest(
|
matrix.client.setDisplayname(displayname),
|
||||||
type: HTTPType.PUT,
|
|
||||||
action: "/client/r0/profile/${matrix.client.userID}/displayname",
|
|
||||||
data: {"displayname": displayname},
|
|
||||||
),
|
|
||||||
);
|
);
|
||||||
if (success != null && success.isEmpty) {
|
if (success != null && success.isEmpty) {
|
||||||
Toast.show(
|
Toast.show(
|
||||||
|
@ -110,22 +105,7 @@ class _SettingsState extends State<Settings> {
|
||||||
profile?.avatarUrl ?? MxContent(""),
|
profile?.avatarUrl ?? MxContent(""),
|
||||||
defaultIcon: Icons.account_circle,
|
defaultIcon: Icons.account_circle,
|
||||||
loading: profile == null,
|
loading: profile == null,
|
||||||
),
|
onEdit: () => setAvatarAction(context),
|
||||||
ListTile(
|
|
||||||
title: Text(
|
|
||||||
"Profile",
|
|
||||||
style: TextStyle(
|
|
||||||
color: Theme.of(context).primaryColor,
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
kIsWeb
|
|
||||||
? Container()
|
|
||||||
: ListTile(
|
|
||||||
title: Text("Upload avatar"),
|
|
||||||
leading: Icon(Icons.camera),
|
|
||||||
onTap: () => setAvatarAction(context),
|
|
||||||
),
|
),
|
||||||
ListTile(
|
ListTile(
|
||||||
leading: Icon(Icons.edit),
|
leading: Icon(Icons.edit),
|
||||||
|
@ -141,6 +121,7 @@ class _SettingsState extends State<Settings> {
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
Divider(thickness: 8),
|
||||||
ListTile(
|
ListTile(
|
||||||
title: Text(
|
title: Text(
|
||||||
"About",
|
"About",
|
||||||
|
@ -170,6 +151,7 @@ class _SettingsState extends State<Settings> {
|
||||||
title: Text("Source code"),
|
title: Text("Source code"),
|
||||||
onTap: () => launch(
|
onTap: () => launch(
|
||||||
"https://gitlab.com/ChristianPauly/fluffychat-flutter")),
|
"https://gitlab.com/ChristianPauly/fluffychat-flutter")),
|
||||||
|
Divider(thickness: 8),
|
||||||
ListTile(
|
ListTile(
|
||||||
title: Text(
|
title: Text(
|
||||||
"Logout",
|
"Logout",
|
||||||
|
|
13
pubspec.lock
13
pubspec.lock
|
@ -82,8 +82,8 @@ packages:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
path: "."
|
path: "."
|
||||||
ref: c8633111e5f016cc3dd95f644a4e8767be5559f6
|
ref: "2545995bbe96a1d96fe176ab666f4dd03d591aa6"
|
||||||
resolved-ref: c8633111e5f016cc3dd95f644a4e8767be5559f6
|
resolved-ref: "2545995bbe96a1d96fe176ab666f4dd03d591aa6"
|
||||||
url: "https://gitlab.com/famedly/famedlysdk.git"
|
url: "https://gitlab.com/famedly/famedlysdk.git"
|
||||||
source: git
|
source: git
|
||||||
version: "0.0.1"
|
version: "0.0.1"
|
||||||
|
@ -263,6 +263,13 @@ packages:
|
||||||
url: "https://pub.dartlang.org"
|
url: "https://pub.dartlang.org"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.0.5"
|
version: "2.0.5"
|
||||||
|
share:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: share
|
||||||
|
url: "https://pub.dartlang.org"
|
||||||
|
source: hosted
|
||||||
|
version: "0.6.3+5"
|
||||||
sky_engine:
|
sky_engine:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description: flutter
|
description: flutter
|
||||||
|
@ -402,5 +409,5 @@ packages:
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.2.0"
|
version: "2.2.0"
|
||||||
sdks:
|
sdks:
|
||||||
dart: ">=2.6.0 <3.0.0"
|
dart: ">=2.7.0 <3.0.0"
|
||||||
flutter: ">=1.12.13+hotfix.5 <2.0.0"
|
flutter: ">=1.12.13+hotfix.5 <2.0.0"
|
||||||
|
|
|
@ -27,7 +27,7 @@ dependencies:
|
||||||
famedlysdk:
|
famedlysdk:
|
||||||
git:
|
git:
|
||||||
url: https://gitlab.com/famedly/famedlysdk.git
|
url: https://gitlab.com/famedly/famedlysdk.git
|
||||||
ref: c8633111e5f016cc3dd95f644a4e8767be5559f6
|
ref: 2545995bbe96a1d96fe176ab666f4dd03d591aa6
|
||||||
|
|
||||||
localstorage: ^3.0.1+4
|
localstorage: ^3.0.1+4
|
||||||
bubble: ^1.1.9+1
|
bubble: ^1.1.9+1
|
||||||
|
@ -44,6 +44,7 @@ dependencies:
|
||||||
link_text: ^0.1.1
|
link_text: ^0.1.1
|
||||||
path_provider: ^1.5.1
|
path_provider: ^1.5.1
|
||||||
webview_flutter: ^0.3.19+4
|
webview_flutter: ^0.3.19+4
|
||||||
|
share: ^0.6.3+5
|
||||||
|
|
||||||
dev_dependencies:
|
dev_dependencies:
|
||||||
flutter_test:
|
flutter_test:
|
||||||
|
|
Loading…
Reference in a new issue