ente/lib/ui/detail_page.dart

125 lines
3.4 KiB
Dart
Raw Normal View History

2020-04-23 20:00:20 +00:00
import 'dart:typed_data';
2020-03-24 19:59:36 +00:00
import 'package:flutter/material.dart';
2020-04-23 20:00:20 +00:00
import 'package:logger/logger.dart';
2020-03-28 18:18:27 +00:00
import 'package:myapp/core/lru_map.dart';
2020-03-30 14:28:46 +00:00
import 'package:myapp/models/photo.dart';
2020-04-14 16:49:33 +00:00
import 'package:photo_view/photo_view.dart';
2020-04-17 08:17:37 +00:00
import 'extents_page_view.dart';
2020-04-23 20:00:20 +00:00
import 'loading_widget.dart';
2020-04-24 12:40:24 +00:00
import 'package:myapp/utils/share_util.dart';
2020-03-24 19:59:36 +00:00
2020-04-17 08:17:37 +00:00
class DetailPage extends StatefulWidget {
final List<Photo> photos;
final int selectedIndex;
2020-03-24 19:59:36 +00:00
2020-04-17 08:17:37 +00:00
DetailPage(this.photos, this.selectedIndex, {Key key}) : super(key: key);
@override
_DetailPageState createState() => _DetailPageState();
}
class _DetailPageState extends State<DetailPage> {
bool _shouldDisableScroll = false;
int _selectedIndex = 0;
2020-04-23 20:00:20 +00:00
final _cachedImages = LRUMap<int, ZoomableImage>(5);
2020-03-24 19:59:36 +00:00
@override
Widget build(BuildContext context) {
_selectedIndex = widget.selectedIndex;
2020-04-23 20:00:20 +00:00
Logger().i("Loading " + widget.photos[_selectedIndex].toString());
var pageController = PageController(initialPage: _selectedIndex);
2020-03-24 19:59:36 +00:00
return Scaffold(
2020-04-23 20:00:20 +00:00
appBar: _buildAppBar(),
2020-03-24 19:59:36 +00:00
body: Center(
child: Container(
2020-04-17 08:17:37 +00:00
child: ExtentsPageView.extents(
itemBuilder: (context, index) {
2020-04-23 20:00:20 +00:00
if (_cachedImages.get(index) != null) {
return _cachedImages.get(index);
}
final image = ZoomableImage(
photo: widget.photos[index],
shouldDisableScroll: (value) {
setState(() {
_shouldDisableScroll = value;
});
},
);
_cachedImages.put(index, image);
return image;
2020-04-17 08:17:37 +00:00
},
onPageChanged: (int index) {
_selectedIndex = index;
2020-04-17 08:17:37 +00:00
},
physics: _shouldDisableScroll
? NeverScrollableScrollPhysics()
: PageScrollPhysics(),
2020-04-17 10:11:18 +00:00
controller: pageController,
itemCount: widget.photos.length,
2020-04-17 08:17:37 +00:00
),
2020-03-24 19:59:36 +00:00
),
),
);
}
2020-04-23 20:00:20 +00:00
AppBar _buildAppBar() {
return AppBar(
actions: <Widget>[
IconButton(
icon: Icon(Icons.share),
2020-04-24 12:40:24 +00:00
onPressed: () async {
share(widget.photos[_selectedIndex]);
2020-04-23 20:00:20 +00:00
},
)
],
);
}
}
class ZoomableImage extends StatelessWidget {
final Function(bool) shouldDisableScroll;
const ZoomableImage({
Key key,
@required this.photo,
this.shouldDisableScroll,
}) : super(key: key);
final Photo photo;
@override
Widget build(BuildContext context) {
2020-04-24 12:40:24 +00:00
Logger().i("Building " + photo.toString());
2020-04-23 20:00:20 +00:00
if (ImageLruCache.getData(photo.generatedId) != null) {
return _buildPhotoView(ImageLruCache.getData(photo.generatedId));
}
return FutureBuilder<Uint8List>(
2020-04-24 12:40:24 +00:00
future: photo.getBytes(),
2020-04-23 20:00:20 +00:00
builder: (_, snapshot) {
if (snapshot.hasData) {
return _buildPhotoView(snapshot.data);
} else if (snapshot.hasError) {
2020-04-24 12:40:24 +00:00
return Text(snapshot.error.toString());
2020-04-23 20:00:20 +00:00
} else {
return loadWidget;
}
},
);
}
Widget _buildPhotoView(Uint8List imageData) {
2020-04-17 08:17:37 +00:00
ValueChanged<PhotoViewScaleState> scaleStateChangedCallback = (value) {
2020-04-23 20:00:20 +00:00
if (shouldDisableScroll != null) {
shouldDisableScroll(value != PhotoViewScaleState.initial);
2020-04-17 08:17:37 +00:00
}
};
return PhotoView(
2020-04-23 20:00:20 +00:00
imageProvider: Image.memory(imageData).image,
2020-04-17 08:17:37 +00:00
scaleStateChangedCallback: scaleStateChangedCallback,
2020-04-17 20:44:13 +00:00
minScale: PhotoViewComputedScale.contained,
2020-04-17 08:17:37 +00:00
);
2020-03-24 19:59:36 +00:00
}
}