mirror of
https://gitlab.com/famedly/fluffychat.git
synced 2024-12-22 05:22:34 +01:00
fix: Image viewer
This commit is contained in:
parent
85cb3a6f15
commit
91965b141d
@ -1,5 +1,6 @@
|
|||||||
import 'package:fluffychat/pages/archive.dart';
|
import 'package:fluffychat/pages/archive.dart';
|
||||||
import 'package:fluffychat/pages/homeserver_picker.dart';
|
import 'package:fluffychat/pages/homeserver_picker.dart';
|
||||||
|
import 'package:fluffychat/pages/image_viewer.dart';
|
||||||
import 'package:fluffychat/pages/invitation_selection.dart';
|
import 'package:fluffychat/pages/invitation_selection.dart';
|
||||||
import 'package:fluffychat/pages/settings_emotes.dart';
|
import 'package:fluffychat/pages/settings_emotes.dart';
|
||||||
import 'package:fluffychat/pages/settings_multiple_emotes.dart';
|
import 'package:fluffychat/pages/settings_multiple_emotes.dart';
|
||||||
@ -51,6 +52,11 @@ class AppRoutes {
|
|||||||
widget: ChatList(),
|
widget: ChatList(),
|
||||||
stackedRoutes: [
|
stackedRoutes: [
|
||||||
VWidget(path: ':roomid', widget: Chat(), stackedRoutes: [
|
VWidget(path: ':roomid', widget: Chat(), stackedRoutes: [
|
||||||
|
VWidget(
|
||||||
|
path: 'image/:eventid',
|
||||||
|
widget: ImageViewer(),
|
||||||
|
buildTransition: _fadeTransition,
|
||||||
|
),
|
||||||
VWidget(
|
VWidget(
|
||||||
path: 'encryption',
|
path: 'encryption',
|
||||||
widget: ChatEncryptionSettings(),
|
widget: ChatEncryptionSettings(),
|
||||||
@ -126,6 +132,11 @@ class AppRoutes {
|
|||||||
widget: EmptyPage(),
|
widget: EmptyPage(),
|
||||||
buildTransition: _fadeTransition,
|
buildTransition: _fadeTransition,
|
||||||
),
|
),
|
||||||
|
VWidget(
|
||||||
|
path: 'image/:eventid',
|
||||||
|
widget: ImageViewer(),
|
||||||
|
buildTransition: _fadeTransition,
|
||||||
|
),
|
||||||
VWidget(
|
VWidget(
|
||||||
path: 'encryption',
|
path: 'encryption',
|
||||||
widget: ChatEncryptionSettings(),
|
widget: ChatEncryptionSettings(),
|
||||||
|
@ -8,10 +8,9 @@ import 'package:vrouter/vrouter.dart';
|
|||||||
import '../utils/matrix_sdk_extensions.dart/event_extension.dart';
|
import '../utils/matrix_sdk_extensions.dart/event_extension.dart';
|
||||||
|
|
||||||
class ImageViewer extends StatefulWidget {
|
class ImageViewer extends StatefulWidget {
|
||||||
final Event event;
|
|
||||||
final void Function() onLoaded;
|
final void Function() onLoaded;
|
||||||
|
|
||||||
const ImageViewer(this.event, {Key key, this.onLoaded}) : super(key: key);
|
const ImageViewer({Key key, this.onLoaded}) : super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
ImageViewerController createState() => ImageViewerController();
|
ImageViewerController createState() => ImageViewerController();
|
||||||
@ -19,13 +18,14 @@ class ImageViewer extends StatefulWidget {
|
|||||||
|
|
||||||
class ImageViewerController extends State<ImageViewer> {
|
class ImageViewerController extends State<ImageViewer> {
|
||||||
/// Forward this image to another room.
|
/// Forward this image to another room.
|
||||||
void forwardAction() {
|
void forwardAction(Event event) {
|
||||||
Matrix.of(context).shareContent = widget.event.content;
|
Matrix.of(context).shareContent = event.content;
|
||||||
VRouter.of(context).push('/rooms');
|
VRouter.of(context).push('/rooms');
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Open this file with a system call.
|
/// Open this file with a system call.
|
||||||
void openFileAction() => widget.event.openFile(context, downloadOnly: true);
|
void openFileAction(Event event) =>
|
||||||
|
event.openFile(context, downloadOnly: true);
|
||||||
|
|
||||||
/// Go back if user swiped it away
|
/// Go back if user swiped it away
|
||||||
void onInteractionEnds(ScaleEndDetails endDetails) {
|
void onInteractionEnds(ScaleEndDetails endDetails) {
|
||||||
@ -37,6 +37,16 @@ class ImageViewerController extends State<ImageViewer> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<Event> getEvent() {
|
||||||
|
final roomId = VRouter.of(context).pathParameters['roomid'];
|
||||||
|
final eventId = VRouter.of(context).pathParameters['eventid'];
|
||||||
|
return Matrix.of(context).client.database.getEventById(
|
||||||
|
Matrix.of(context).client.id,
|
||||||
|
eventId,
|
||||||
|
Matrix.of(context).client.getRoomById(roomId),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) => ImageViewerView(this);
|
Widget build(BuildContext context) => ImageViewerView(this);
|
||||||
}
|
}
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
|
|
||||||
import '../image_viewer.dart';
|
import '../image_viewer.dart';
|
||||||
import 'package:fluffychat/widgets/event_content/image_bubble.dart';
|
import 'package:fluffychat/widgets/event_content/image_bubble.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
@ -10,50 +12,72 @@ class ImageViewerView extends StatelessWidget {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return FutureBuilder<Event>(
|
||||||
backgroundColor: Colors.black,
|
future: controller.getEvent(),
|
||||||
extendBodyBehindAppBar: true,
|
builder: (context, snapshot) {
|
||||||
appBar: AppBar(
|
if (!snapshot.hasData) {
|
||||||
elevation: 0,
|
return Scaffold(
|
||||||
leading: IconButton(
|
backgroundColor: Colors.black,
|
||||||
icon: Icon(Icons.close),
|
appBar: AppBar(
|
||||||
onPressed: Navigator.of(context, rootNavigator: false).pop,
|
elevation: 0,
|
||||||
color: Colors.white,
|
backgroundColor: Color(0x44000000),
|
||||||
tooltip: L10n.of(context).close,
|
),
|
||||||
),
|
body: Center(
|
||||||
backgroundColor: Color(0x44000000),
|
child: snapshot.hasError
|
||||||
actions: [
|
? Text(
|
||||||
IconButton(
|
L10n.of(context).oopsSomethingWentWrong,
|
||||||
icon: Icon(Icons.reply_outlined),
|
style: TextStyle(color: Colors.white),
|
||||||
onPressed: controller.forwardAction,
|
textAlign: TextAlign.center,
|
||||||
color: Colors.white,
|
)
|
||||||
tooltip: L10n.of(context).share,
|
: CircularProgressIndicator()),
|
||||||
),
|
);
|
||||||
IconButton(
|
}
|
||||||
icon: Icon(Icons.download_outlined),
|
final event = snapshot.data;
|
||||||
onPressed: controller.openFileAction,
|
return Scaffold(
|
||||||
color: Colors.white,
|
|
||||||
tooltip: L10n.of(context).downloadFile,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
body: InteractiveViewer(
|
|
||||||
minScale: 1.0,
|
|
||||||
maxScale: 10.0,
|
|
||||||
onInteractionEnd: controller.onInteractionEnds,
|
|
||||||
child: Center(
|
|
||||||
child: ImageBubble(
|
|
||||||
controller.widget.event,
|
|
||||||
tapToView: false,
|
|
||||||
onLoaded: controller.widget.onLoaded,
|
|
||||||
fit: BoxFit.contain,
|
|
||||||
backgroundColor: Colors.black,
|
backgroundColor: Colors.black,
|
||||||
maxSize: false,
|
extendBodyBehindAppBar: true,
|
||||||
radius: 0.0,
|
appBar: AppBar(
|
||||||
thumbnailOnly: false,
|
elevation: 0,
|
||||||
),
|
backgroundColor: Color(0x44000000),
|
||||||
),
|
leading: IconButton(
|
||||||
),
|
icon: Icon(Icons.close),
|
||||||
);
|
onPressed: Navigator.of(context, rootNavigator: false).pop,
|
||||||
|
color: Colors.white,
|
||||||
|
tooltip: L10n.of(context).close,
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
IconButton(
|
||||||
|
icon: Icon(Icons.reply_outlined),
|
||||||
|
onPressed: () => controller.forwardAction(event),
|
||||||
|
color: Colors.white,
|
||||||
|
tooltip: L10n.of(context).share,
|
||||||
|
),
|
||||||
|
IconButton(
|
||||||
|
icon: Icon(Icons.download_outlined),
|
||||||
|
onPressed: () => controller.openFileAction(event),
|
||||||
|
color: Colors.white,
|
||||||
|
tooltip: L10n.of(context).downloadFile,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
body: InteractiveViewer(
|
||||||
|
minScale: 1.0,
|
||||||
|
maxScale: 10.0,
|
||||||
|
onInteractionEnd: controller.onInteractionEnds,
|
||||||
|
child: Center(
|
||||||
|
child: ImageBubble(
|
||||||
|
event,
|
||||||
|
tapToView: false,
|
||||||
|
onLoaded: controller.widget.onLoaded,
|
||||||
|
fit: BoxFit.contain,
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
maxSize: false,
|
||||||
|
radius: 0.0,
|
||||||
|
thumbnailOnly: false,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,16 +3,14 @@ import 'package:future_loading_dialog/future_loading_dialog.dart';
|
|||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
import 'package:flutter_cache_manager/flutter_cache_manager.dart';
|
||||||
|
import 'package:vrouter/vrouter.dart';
|
||||||
import 'matrix_file_extension.dart';
|
import 'matrix_file_extension.dart';
|
||||||
import '../../pages/image_viewer.dart';
|
|
||||||
|
|
||||||
extension LocalizedBody on Event {
|
extension LocalizedBody on Event {
|
||||||
void openFile(BuildContext context, {bool downloadOnly = false}) async {
|
void openFile(BuildContext context, {bool downloadOnly = false}) async {
|
||||||
if (!downloadOnly &&
|
if (!downloadOnly &&
|
||||||
[MessageTypes.Image, MessageTypes.Sticker].contains(messageType)) {
|
[MessageTypes.Image, MessageTypes.Sticker].contains(messageType)) {
|
||||||
await Navigator.of(context, rootNavigator: false).push(
|
VRouter.of(context).push('/rooms/${room.id}/image/$eventId');
|
||||||
MaterialPageRoute(builder: (_) => ImageViewer(this)),
|
|
||||||
);
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
final matrixFile = await showFutureLoadingDialog(
|
final matrixFile = await showFutureLoadingDialog(
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
import 'package:famedlysdk/famedlysdk.dart';
|
import 'package:famedlysdk/famedlysdk.dart';
|
||||||
import 'package:fluffychat/pages/image_viewer.dart';
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/foundation.dart';
|
import 'package:flutter/foundation.dart';
|
||||||
import 'package:flutter_blurhash/flutter_blurhash.dart';
|
import 'package:flutter_blurhash/flutter_blurhash.dart';
|
||||||
import 'package:cached_network_image/cached_network_image.dart';
|
import 'package:cached_network_image/cached_network_image.dart';
|
||||||
import 'package:flutter_svg/flutter_svg.dart';
|
import 'package:flutter_svg/flutter_svg.dart';
|
||||||
|
import 'package:vrouter/vrouter.dart';
|
||||||
|
|
||||||
import '../../utils/matrix_sdk_extensions.dart/event_extension.dart';
|
import '../../utils/matrix_sdk_extensions.dart/event_extension.dart';
|
||||||
|
|
||||||
@ -237,23 +237,8 @@ class _ImageBubbleState extends State<ImageBubble> {
|
|||||||
child: InkWell(
|
child: InkWell(
|
||||||
onTap: () {
|
onTap: () {
|
||||||
if (!widget.tapToView) return;
|
if (!widget.tapToView) return;
|
||||||
Navigator.of(context, rootNavigator: false).push(
|
VRouter.of(context).push(
|
||||||
MaterialPageRoute(
|
'/rooms/${widget.event.room.id}/image/${widget.event.eventId}');
|
||||||
builder: (_) => ImageViewer(widget.event, onLoaded: () {
|
|
||||||
// If the original file didn't load yet, we want to do that now.
|
|
||||||
// This is so that the original file displays after going on the image viewer,
|
|
||||||
// waiting for it to load, and then hitting back. This ensures that we always
|
|
||||||
// display the best image available, with requiring as little network as possible
|
|
||||||
if (_file == null) {
|
|
||||||
widget.event.isAttachmentCached().then((cached) {
|
|
||||||
if (cached) {
|
|
||||||
_requestFile();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
child: Hero(
|
child: Hero(
|
||||||
tag: widget.event.eventId,
|
tag: widget.event.eventId,
|
||||||
|
Loading…
Reference in New Issue
Block a user